Compare commits
	
		
			1 Commits
		
	
	
		
			v0.15.1
			...
			extrude-ma
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 2d350a93ed | 
| @ -1,3 +1,3 @@ | |||||||
| [codespell] | [codespell] | ||||||
| ignore-words-list: crate,everytime,inout,co-ordinate,ot,nwo | ignore-words-list: crate,everytime | ||||||
| skip: **/target,node_modules,build,**/Cargo.lock | skip: **/target,node_modules,build | ||||||
|  | |||||||
| @ -1,6 +1,6 @@ | |||||||
| VITE_KC_API_WS_MODELING_URL=wss://api.dev.zoo.dev/ws/modeling/commands | VITE_KC_API_WS_MODELING_URL=wss://api.dev.kittycad.io/ws/modeling/commands | ||||||
| VITE_KC_API_BASE_URL=https://api.dev.zoo.dev | VITE_KC_API_BASE_URL=https://api.dev.kittycad.io | ||||||
| VITE_KC_SITE_BASE_URL=https://dev.zoo.dev | VITE_KC_SITE_BASE_URL=https://dev.kittycad.io | ||||||
| VITE_KC_SKIP_AUTH=false | VITE_KC_SKIP_AUTH=false | ||||||
| VITE_KC_CONNECTION_TIMEOUT_MS=5000 | VITE_KC_CONNECTION_TIMEOUT_MS=5000 | ||||||
| VITE_KC_SENTRY_DSN= | VITE_KC_SENTRY_DSN= | ||||||
|  | |||||||
| @ -1,6 +1,6 @@ | |||||||
| VITE_KC_API_WS_MODELING_URL=wss://api.zoo.dev/ws/modeling/commands | VITE_KC_API_WS_MODELING_URL=wss://api.kittycad.io/ws/modeling/commands | ||||||
| VITE_KC_API_BASE_URL=https://api.zoo.dev | VITE_KC_API_BASE_URL=https://api.kittycad.io | ||||||
| VITE_KC_SITE_BASE_URL=https://zoo.dev | VITE_KC_SITE_BASE_URL=https://kittycad.io | ||||||
| VITE_KC_SKIP_AUTH=false | VITE_KC_SKIP_AUTH=false | ||||||
| VITE_KC_CONNECTION_TIMEOUT_MS=15000 | VITE_KC_CONNECTION_TIMEOUT_MS=15000 | ||||||
| VITE_KC_SENTRY_DSN=https://a814f2f66734989a90367f48feee28ca@o1042111.ingest.sentry.io/4505789425844224 | VITE_KC_SENTRY_DSN=https://a814f2f66734989a90367f48feee28ca@o1042111.ingest.sentry.io/4505789425844224 | ||||||
|  | |||||||
| @ -1,2 +1 @@ | |||||||
| src/wasm-lib/* | src/wasm-lib/* | ||||||
| *.typegen.ts |  | ||||||
|  | |||||||
| @ -17,12 +17,12 @@ | |||||||
|         "never" |         "never" | ||||||
|       ], |       ], | ||||||
|       "react-hooks/exhaustive-deps": "off", |       "react-hooks/exhaustive-deps": "off", | ||||||
|  |       "@typescript-eslint/no-floating-promises": "warn" | ||||||
|     }, |     }, | ||||||
|     "overrides": [ |     "overrides": [ | ||||||
|       { |       { | ||||||
|         "files": ["e2e/**/*.ts"], // Update the pattern based on your file structure |         "files": ["e2e/**/*.ts"], // Update the pattern based on your file structure | ||||||
|         "rules": { |         "rules": { | ||||||
|           "@typescript-eslint/no-floating-promises": "warn", |  | ||||||
|           "testing-library/prefer-screen-queries": "off" |           "testing-library/prefer-screen-queries": "off" | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|  | |||||||
							
								
								
									
										29
									
								
								.github/workflows/announce_release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -1,29 +0,0 @@ | |||||||
| name: Announce Release |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   release: |  | ||||||
|     types: [published] |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   announce_release: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|     - name: Check out code |  | ||||||
|       uses: actions/checkout@v4 |  | ||||||
|  |  | ||||||
|     - name: Set up Python |  | ||||||
|       uses: actions/setup-python@v5 |  | ||||||
|       with: |  | ||||||
|         python-version: '3.x' |  | ||||||
|  |  | ||||||
|     - name: Install dependencies |  | ||||||
|       run: | |  | ||||||
|         python -m pip install --upgrade pip |  | ||||||
|         pip install requests |  | ||||||
|  |  | ||||||
|     - name: Announce Release |  | ||||||
|       env: |  | ||||||
|         DISCORD_WEBHOOK_URL: ${{ secrets.DISCORD_WEBHOOK_URL }} |  | ||||||
|         RELEASE_VERSION: ${{ github.event.release.tag_name }} |  | ||||||
|         RELEASE_BODY: ${{ github.event.release.body}} |  | ||||||
|       run: python public/announce_release.py |  | ||||||
							
								
								
									
										11
									
								
								.github/workflows/cargo-clippy.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -43,6 +43,17 @@ jobs: | |||||||
|       - name: Rust Cache |       - name: Rust Cache | ||||||
|         uses: Swatinem/rust-cache@v2.6.1 |         uses: Swatinem/rust-cache@v2.6.1 | ||||||
|  |  | ||||||
|  |       - name: Install ffmpeg | ||||||
|  |         run: | | ||||||
|  |           sudo apt update | ||||||
|  |           sudo apt install \ | ||||||
|  |             ffmpeg \ | ||||||
|  |             libavformat-dev \ | ||||||
|  |             libavutil-dev \ | ||||||
|  |             libclang-dev \ | ||||||
|  |             libswscale-dev \ | ||||||
|  |             --no-install-recommends | ||||||
|  |  | ||||||
|       - name: Run clippy |       - name: Run clippy | ||||||
|         run: | |         run: | | ||||||
|           cd "${{ matrix.dir }}" |           cd "${{ matrix.dir }}" | ||||||
|  | |||||||
							
								
								
									
										10
									
								
								.github/workflows/cargo-test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -44,6 +44,16 @@ jobs: | |||||||
|       - uses: taiki-e/install-action@nextest |       - uses: taiki-e/install-action@nextest | ||||||
|       - name: Rust Cache |       - name: Rust Cache | ||||||
|         uses: Swatinem/rust-cache@v2.6.1 |         uses: Swatinem/rust-cache@v2.6.1 | ||||||
|  |       - name: Install ffmpeg | ||||||
|  |         run: | | ||||||
|  |           sudo apt update | ||||||
|  |           sudo apt install \ | ||||||
|  |             ffmpeg \ | ||||||
|  |             libavformat-dev \ | ||||||
|  |             libavutil-dev \ | ||||||
|  |             libclang-dev \ | ||||||
|  |             libswscale-dev \ | ||||||
|  |             --no-install-recommends | ||||||
|       - name: cargo test |       - name: cargo test | ||||||
|         shell: bash |         shell: bash | ||||||
|         run: |- |         run: |- | ||||||
|  | |||||||
							
								
								
									
										63
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -46,7 +46,6 @@ jobs: | |||||||
|           workspaces: './src/wasm-lib' |           workspaces: './src/wasm-lib' | ||||||
|  |  | ||||||
|       - run: yarn build:wasm |       - run: yarn build:wasm | ||||||
|       - run: yarn xstate:typegen |  | ||||||
|       - run: yarn tsc |       - run: yarn tsc | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -105,7 +104,7 @@ jobs: | |||||||
|         if: github.event_name == 'schedule' |         if: github.event_name == 'schedule' | ||||||
|         run: | |         run: | | ||||||
|           VERSION=$(date +'%-y.%-m.%-d') yarn bump-jsons |           VERSION=$(date +'%-y.%-m.%-d') yarn bump-jsons | ||||||
|           echo "$(jq --arg url 'https://dl.zoo.dev/releases/modeling-app/nightly/last_update.json' \ |           echo "$(jq --arg url 'https://dl.kittycad.io/releases/modeling-app/nightly/last_update.json' \ | ||||||
|             '.tauri.updater.endpoints[]=$url' src-tauri/tauri.release.conf.json --indent 2)" > src-tauri/tauri.release.conf.json |             '.tauri.updater.endpoints[]=$url' src-tauri/tauri.release.conf.json --indent 2)" > src-tauri/tauri.release.conf.json | ||||||
|  |  | ||||||
|       - uses: actions/upload-artifact@v3 |       - uses: actions/upload-artifact@v3 | ||||||
| @ -124,9 +123,8 @@ jobs: | |||||||
|     needs: [prepare-json-files] |     needs: [prepare-json-files] | ||||||
|     runs-on: ${{ matrix.os }} |     runs-on: ${{ matrix.os }} | ||||||
|     strategy: |     strategy: | ||||||
|       fail-fast: false |  | ||||||
|       matrix: |       matrix: | ||||||
|         os: [macos-14, ubuntu-latest, windows-latest] |         os: [macos-latest, ubuntu-latest, windows-latest] | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v4 | ||||||
|  |  | ||||||
| @ -189,10 +187,10 @@ jobs: | |||||||
|       - name: Fix format |       - name: Fix format | ||||||
|         run: yarn fmt |         run: yarn fmt | ||||||
|  |  | ||||||
|       - name: Install x86 target for Universal builds (MacOS only) |       - name: Install Universal target (MacOS only) | ||||||
|         if: matrix.os == 'macos-14' |         if: matrix.os == 'macos-latest' | ||||||
|         run: | |         run: | | ||||||
|           rustup target add x86_64-apple-darwin |           rustup target add aarch64-apple-darwin | ||||||
|  |  | ||||||
|       - name: Prepare certificate and variables (Windows only) |       - name: Prepare certificate and variables (Windows only) | ||||||
|         if: ${{ matrix.os == 'windows-latest' && env.BUILD_RELEASE == 'true' }} |         if: ${{ matrix.os == 'windows-latest' && env.BUILD_RELEASE == 'true' }} | ||||||
| @ -226,7 +224,7 @@ jobs: | |||||||
|         with: |         with: | ||||||
|           includeRelease: false |           includeRelease: false | ||||||
|           includeDebug: true |           includeDebug: true | ||||||
|           args: ${{ matrix.os == 'macos-14' && '--target universal-apple-darwin' || '' }} |           args: ${{ matrix.os == 'macos-latest' && '--target universal-apple-darwin' || '' }} | ||||||
|  |  | ||||||
|       - name: Build the app (release) and sign |       - name: Build the app (release) and sign | ||||||
|         uses: tauri-apps/tauri-action@v0 |         uses: tauri-apps/tauri-action@v0 | ||||||
| @ -242,12 +240,11 @@ jobs: | |||||||
|           APPLE_TEAM_ID: ${{ secrets.APPLE_TEAM_ID }} |           APPLE_TEAM_ID: ${{ secrets.APPLE_TEAM_ID }} | ||||||
|           TAURI_CONF_ARGS: "--config ${{ matrix.os == 'windows-latest' && 'src-tauri\\tauri.release.conf.json' || 'src-tauri/tauri.release.conf.json' }}" |           TAURI_CONF_ARGS: "--config ${{ matrix.os == 'windows-latest' && 'src-tauri\\tauri.release.conf.json' || 'src-tauri/tauri.release.conf.json' }}" | ||||||
|         with: |         with: | ||||||
|           args: "${{ matrix.os == 'macos-14' && '--target universal-apple-darwin' || '' }} ${{ env.TAURI_CONF_ARGS }}" |           args: "${{ matrix.os == 'macos-latest' && '--target universal-apple-darwin' || '' }} ${{ env.TAURI_CONF_ARGS }}" | ||||||
|  |  | ||||||
|       - uses: actions/upload-artifact@v3 |       - uses: actions/upload-artifact@v3 | ||||||
|         if: matrix.os != 'ubuntu-latest' |  | ||||||
|         env: |         env: | ||||||
|           PREFIX: ${{ matrix.os == 'macos-14' && 'src-tauri/target/universal-apple-darwin' || 'src-tauri/target' }} |           PREFIX: ${{ matrix.os == 'macos-latest' && 'src-tauri/target/universal-apple-darwin' || 'src-tauri/target' }} | ||||||
|           MODE: ${{ env.BUILD_RELEASE == 'true' && 'release' || 'debug' }} |           MODE: ${{ env.BUILD_RELEASE == 'true' && 'release' || 'debug' }} | ||||||
|         with: |         with: | ||||||
|           path: "${{ env.PREFIX }}/${{ env.MODE }}/bundle/*/*" |           path: "${{ env.PREFIX }}/${{ env.MODE }}/bundle/*/*" | ||||||
| @ -255,12 +252,12 @@ jobs: | |||||||
|       - name: Run e2e tests (linux only) |       - name: Run e2e tests (linux only) | ||||||
|         if: matrix.os == 'ubuntu-latest' |         if: matrix.os == 'ubuntu-latest' | ||||||
|         run: | |         run: | | ||||||
|           cargo install tauri-driver@0.1.3 |           cargo install tauri-driver | ||||||
|           source .env.${{ env.BUILD_RELEASE == 'true' && 'production' || 'development' }} |           source .env.${{ env.BUILD_RELEASE == 'true' && 'production' || 'development' }} | ||||||
|           export VITE_KC_API_BASE_URL |           export VITE_KC_API_BASE_URL | ||||||
|           xvfb-run yarn test:e2e:tauri |           xvfb-run yarn test:e2e:tauri | ||||||
|         env: |         env: | ||||||
|           E2E_APPLICATION: "./src-tauri/target/${{ env.BUILD_RELEASE == 'true' && 'release' || 'debug' }}/zoo-modeling-app" |           E2E_APPLICATION: "./src-tauri/target/${{ env.BUILD_RELEASE == 'true' && 'release' || 'debug' }}/kittycad-modeling" | ||||||
|           KITTYCAD_API_TOKEN: ${{ env.BUILD_RELEASE == 'true' && secrets.KITTYCAD_API_TOKEN || secrets.KITTYCAD_API_TOKEN_DEV }} |           KITTYCAD_API_TOKEN: ${{ env.BUILD_RELEASE == 'true' && secrets.KITTYCAD_API_TOKEN || secrets.KITTYCAD_API_TOKEN_DEV }} | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -274,24 +271,26 @@ jobs: | |||||||
|       PUB_DATE: ${{ github.event_name == 'release' && github.event.release.created_at || github.event.repository.updated_at }} |       PUB_DATE: ${{ github.event_name == 'release' && github.event.release.created_at || github.event.repository.updated_at }} | ||||||
|       NOTES: ${{ github.event_name == 'release' && github.event.release.body || format('Nightly build, commit {0}', github.sha) }} |       NOTES: ${{ github.event_name == 'release' && github.event.release.body || format('Nightly build, commit {0}', github.sha) }} | ||||||
|       BUCKET_DIR: ${{ github.event_name == 'release' && 'dl.kittycad.io/releases/modeling-app' || 'dl.kittycad.io/releases/modeling-app/nightly' }} |       BUCKET_DIR: ${{ github.event_name == 'release' && 'dl.kittycad.io/releases/modeling-app' || 'dl.kittycad.io/releases/modeling-app/nightly' }} | ||||||
|       WEBSITE_DIR: ${{ github.event_name == 'release' && 'dl.zoo.dev/releases/modeling-app' || 'dl.zoo.dev/releases/modeling-app/nightly' }} |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/download-artifact@v3 |       - uses: actions/download-artifact@v3 | ||||||
|  |  | ||||||
|       - name: Generate the update static endpoint |       - name: Generate the update static endpoint | ||||||
|         run: | |         run: | | ||||||
|           ls -l artifact/*/*oo* |           ls -l artifact/*/*itty* | ||||||
|           DARWIN_SIG=`cat artifact/macos/*.app.tar.gz.sig` |           DARWIN_SIG=`cat artifact/macos/*.app.tar.gz.sig` | ||||||
|  |           LINUX_SIG=`cat artifact/appimage/*.AppImage.tar.gz.sig` | ||||||
|           WINDOWS_SIG=`cat artifact/msi/*.msi.zip.sig` |           WINDOWS_SIG=`cat artifact/msi/*.msi.zip.sig` | ||||||
|           RELEASE_DIR=https://${WEBSITE_DIR}/${VERSION} |           RELEASE_DIR=https://${BUCKET_DIR}/${VERSION} | ||||||
|           jq --null-input \ |           jq --null-input \ | ||||||
|             --arg version "${VERSION}" \ |             --arg version "${VERSION}" \ | ||||||
|             --arg pub_date "${PUB_DATE}" \ |             --arg pub_date "${PUB_DATE}" \ | ||||||
|             --arg notes "${NOTES}" \ |             --arg notes "${NOTES}" \ | ||||||
|             --arg darwin_sig "$DARWIN_SIG" \ |             --arg darwin_sig "$DARWIN_SIG" \ | ||||||
|             --arg darwin_url "$RELEASE_DIR/macos/Zoo%20Modeling%20App.app.tar.gz" \ |             --arg darwin_url "$RELEASE_DIR/macos/KittyCAD%20Modeling.app.tar.gz" \ | ||||||
|  |             --arg linux_sig "$LINUX_SIG" \ | ||||||
|  |             --arg linux_url "$RELEASE_DIR/appimage/kittycad-modeling_${VERSION_NO_V}_amd64.AppImage.tar.gz" \ | ||||||
|             --arg windows_sig "$WINDOWS_SIG" \ |             --arg windows_sig "$WINDOWS_SIG" \ | ||||||
|             --arg windows_url "$RELEASE_DIR/msi/Zoo%20Modeling%20App_${VERSION_NO_V}_x64_en-US.msi.zip" \ |             --arg windows_url "$RELEASE_DIR/msi/KittyCAD%20Modeling_${VERSION_NO_V}_x64_en-US.msi.zip" \ | ||||||
|             '{ |             '{ | ||||||
|               "version": $version, |               "version": $version, | ||||||
|               "pub_date": $pub_date, |               "pub_date": $pub_date, | ||||||
| @ -305,6 +304,10 @@ jobs: | |||||||
|                   "signature": $darwin_sig, |                   "signature": $darwin_sig, | ||||||
|                   "url": $darwin_url |                   "url": $darwin_url | ||||||
|                 }, |                 }, | ||||||
|  |                 "linux-x86_64": { | ||||||
|  |                   "signature": $linux_sig, | ||||||
|  |                   "url": $linux_url | ||||||
|  |                 }, | ||||||
|                 "windows-x86_64": { |                 "windows-x86_64": { | ||||||
|                   "signature": $windows_sig, |                   "signature": $windows_sig, | ||||||
|                   "url": $windows_url |                   "url": $windows_url | ||||||
| @ -315,13 +318,14 @@ jobs: | |||||||
|  |  | ||||||
|       - name: Generate the download static endpoint |       - name: Generate the download static endpoint | ||||||
|         run: | |         run: | | ||||||
|           RELEASE_DIR=https://${WEBSITE_DIR}/${VERSION} |           RELEASE_DIR=https://${BUCKET_DIR}/${VERSION} | ||||||
|           jq --null-input \ |           jq --null-input \ | ||||||
|             --arg version "${VERSION}" \ |             --arg version "${VERSION}" \ | ||||||
|             --arg pub_date "${PUB_DATE}" \ |             --arg pub_date "${PUB_DATE}" \ | ||||||
|             --arg notes "${NOTES}" \ |             --arg notes "${NOTES}" \ | ||||||
|             --arg darwin_url "$RELEASE_DIR/dmg/Zoo%20Modeling%20App_${VERSION_NO_V}_universal.dmg" \ |             --arg darwin_url "$RELEASE_DIR/dmg/KittyCAD%20Modeling_${VERSION_NO_V}_universal.dmg" \ | ||||||
|             --arg windows_url "$RELEASE_DIR/msi/Zoo%20Modeling%20App_${VERSION_NO_V}_x64_en-US.msi" \ |             --arg linux_url "$RELEASE_DIR/appimage/kittycad-modeling_${VERSION_NO_V}_amd64.AppImage" \ | ||||||
|  |             --arg windows_url "$RELEASE_DIR/msi/KittyCAD%20Modeling_${VERSION_NO_V}_x64_en-US.msi" \ | ||||||
|             '{ |             '{ | ||||||
|               "version": $version, |               "version": $version, | ||||||
|               "pub_date": $pub_date, |               "pub_date": $pub_date, | ||||||
| @ -330,6 +334,9 @@ jobs: | |||||||
|                 "dmg-universal": { |                 "dmg-universal": { | ||||||
|                   "url": $darwin_url |                   "url": $darwin_url | ||||||
|                 }, |                 }, | ||||||
|  |                 "appimage-x86_64": { | ||||||
|  |                   "url": $linux_url | ||||||
|  |                 }, | ||||||
|                 "msi-x86_64": { |                 "msi-x86_64": { | ||||||
|                   "url": $windows_url |                   "url": $windows_url | ||||||
|                 } |                 } | ||||||
| @ -338,31 +345,31 @@ jobs: | |||||||
|             cat last_download.json |             cat last_download.json | ||||||
|  |  | ||||||
|       - name: Authenticate to Google Cloud |       - name: Authenticate to Google Cloud | ||||||
|         uses: 'google-github-actions/auth@v2.1.1' |         uses: 'google-github-actions/auth@v2.0.0' | ||||||
|         with: |         with: | ||||||
|           credentials_json: '${{ secrets.GOOGLE_CLOUD_DL_SA }}' |           credentials_json: '${{ secrets.GOOGLE_CLOUD_DL_SA }}' | ||||||
|  |  | ||||||
|       - name: Set up Google Cloud SDK |       - name: Set up Google Cloud SDK | ||||||
|         uses: google-github-actions/setup-gcloud@v2.1.0 |         uses: google-github-actions/setup-gcloud@v1.1.1 | ||||||
|         with: |         with: | ||||||
|           project_id: kittycadapi |           project_id: kittycadapi | ||||||
|  |  | ||||||
|       - name: Upload release files to public bucket |       - name: Upload release files to public bucket | ||||||
|         uses: google-github-actions/upload-cloud-storage@v2.1.0 |         uses: google-github-actions/upload-cloud-storage@v1.0.3 | ||||||
|         with: |         with: | ||||||
|           path: artifact |           path: artifact | ||||||
|           glob: '*/Zoo*' |           glob: '*/*itty*' | ||||||
|           parent: false |           parent: false | ||||||
|           destination: ${{ env.BUCKET_DIR }}/${{ env.VERSION }} |           destination: ${{ env.BUCKET_DIR }}/${{ env.VERSION }} | ||||||
|  |  | ||||||
|       - name: Upload update endpoint to public bucket |       - name: Upload update endpoint to public bucket | ||||||
|         uses: google-github-actions/upload-cloud-storage@v2.1.0 |         uses: google-github-actions/upload-cloud-storage@v1.0.3 | ||||||
|         with: |         with: | ||||||
|           path: last_update.json |           path: last_update.json | ||||||
|           destination: ${{ env.BUCKET_DIR }} |           destination: ${{ env.BUCKET_DIR }} | ||||||
|  |  | ||||||
|       - name: Upload download endpoint to public bucket |       - name: Upload download endpoint to public bucket | ||||||
|         uses: google-github-actions/upload-cloud-storage@v2.1.0 |         uses: google-github-actions/upload-cloud-storage@v1.0.3 | ||||||
|         with: |         with: | ||||||
|           path: last_download.json |           path: last_download.json | ||||||
|           destination: ${{ env.BUCKET_DIR }} |           destination: ${{ env.BUCKET_DIR }} | ||||||
| @ -371,4 +378,4 @@ jobs: | |||||||
|         if: ${{ github.event_name == 'release' }} |         if: ${{ github.event_name == 'release' }} | ||||||
|         uses: softprops/action-gh-release@v1 |         uses: softprops/action-gh-release@v1 | ||||||
|         with: |         with: | ||||||
|           files: 'artifact/*/Zoo*' |           files: artifact/*/*itty* | ||||||
|  | |||||||
							
								
								
									
										4
									
								
								.github/workflows/playwright.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -14,7 +14,6 @@ jobs: | |||||||
|       with: |       with: | ||||||
|         node-version-file: '.nvmrc' |         node-version-file: '.nvmrc' | ||||||
|         cache: 'yarn' |         cache: 'yarn' | ||||||
|     - uses: KittyCAD/action-install-cli@v0.2.16 |  | ||||||
|     - name: Install dependencies |     - name: Install dependencies | ||||||
|       run: yarn |       run: yarn | ||||||
|     - name: Install Playwright Browsers |     - name: Install Playwright Browsers | ||||||
| @ -34,7 +33,6 @@ jobs: | |||||||
|       env: |       env: | ||||||
|         CI: true |         CI: true | ||||||
|         token: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} |         token: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||||
|         snapshottoken: ${{ secrets.KITTYCAD_API_TOKEN }} |  | ||||||
|     - uses: actions/upload-artifact@v3 |     - uses: actions/upload-artifact@v3 | ||||||
|       if: always() |       if: always() | ||||||
|       with: |       with: | ||||||
| @ -79,7 +77,7 @@ jobs: | |||||||
|  |  | ||||||
|   playwright-macos: |   playwright-macos: | ||||||
|     timeout-minutes: 60 |     timeout-minutes: 60 | ||||||
|     runs-on: macos-14 |     runs-on: macos-latest | ||||||
|     needs: playwright-ubuntu |     needs: playwright-ubuntu | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v4 |     - uses: actions/checkout@v4 | ||||||
|  | |||||||
							
								
								
									
										13
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -37,20 +37,7 @@ src/wasm-lib/lcov.info | |||||||
| e2e/playwright/playwright-secrets.env | e2e/playwright/playwright-secrets.env | ||||||
| e2e/playwright/temp1.png | e2e/playwright/temp1.png | ||||||
| e2e/playwright/temp2.png | e2e/playwright/temp2.png | ||||||
| # exports from snapshot-tests.spec.ts |  | ||||||
| e2e/playwright/export-snapshots/*.ply |  | ||||||
| e2e/playwright/export-snapshots/*.obj |  | ||||||
| e2e/playwright/export-snapshots/*.step |  | ||||||
| e2e/playwright/export-snapshots/*.stl |  | ||||||
| e2e/playwright/export-snapshots/*binary.gltf |  | ||||||
| e2e/playwright/export-snapshots/*embedded.gltf |  | ||||||
|  |  | ||||||
|  |  | ||||||
| /test-results/ | /test-results/ | ||||||
| /playwright-report/ | /playwright-report/ | ||||||
| /blob-report/ | /blob-report/ | ||||||
| /playwright/.cache/ | /playwright/.cache/ | ||||||
|  |  | ||||||
|  |  | ||||||
| ## generated files |  | ||||||
| src/**/*.typegen.ts |  | ||||||
|  | |||||||
| @ -10,4 +10,4 @@ src/wasm-lib/kcl/bindings | |||||||
| e2e/playwright/export-snapshots | e2e/playwright/export-snapshots | ||||||
|  |  | ||||||
| # XState generated files | # XState generated files | ||||||
| src/machines/**.typegen.ts | src/machines/modelingMachine.typegen.ts | ||||||
|  | |||||||
							
								
								
									
										2
									
								
								LICENSE
									
									
									
									
									
								
							
							
						
						| @ -1,6 +1,6 @@ | |||||||
| The MIT License (MIT) | The MIT License (MIT) | ||||||
|  |  | ||||||
| Copyright (c) 2023 The Zoo Authors | Copyright (c) 2023 The KittyCAD Authors | ||||||
|  |  | ||||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | Permission is hereby granted, free of charge, to any person obtaining a copy | ||||||
| of this software and associated documentation files (the "Software"), to deal | of this software and associated documentation files (the "Software"), to deal | ||||||
|  | |||||||
							
								
								
									
										21
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @ -1,17 +1,17 @@ | |||||||
|  |  | ||||||
|  |  | ||||||
| ## Zoo Modeling App | ## KittyCAD Modeling App | ||||||
|  |  | ||||||
| live at [app.zoo.dev](https://app.zoo.dev/) | live at [app.kittycad.io](https://app.kittycad.io/) | ||||||
|  |  | ||||||
| A CAD application from the future, brought to you by the [Zoo team](https://zoo.dev). | A CAD application from the future, brought to you by the [KittyCAD team](https://kittycad.io). | ||||||
|  |  | ||||||
| Modeling App is our take on what a modern modelling experience can be. It is applying several lessons learned in the decades since most major CAD tools came into existence: | The KittyCAD modeling app is our take on what a modern modelling experience can be. It is applying several lessons learned in the decades since most major CAD tools came into existence: | ||||||
|  |  | ||||||
| - All artifacts—including parts and assemblies—should be represented as human-readable code. At the end of the day, your CAD project should be "plain text" | - All artifacts—including parts and assemblies—should be represented as human-readable code. At the end of the day, your CAD project should be "plain text" | ||||||
|   - This makes version control—which is a solved problem in software engineering—trivial for CAD |   - This makes version control—which is a solved problem in software engineering—trivial for CAD | ||||||
| - All GUI (or point-and-click) interactions should be actions performed on this code representation under the hood | - All GUI (or point-and-click) interactions should be actions performed on this code representation under the hood | ||||||
|   - This unlocks a hybrid approach to modeling. Whether you point-and-click as you always have or you write your own KCL code, you are performing the same action in Modeling App |   - This unlocks a hybrid approach to modeling. Whether you point-and-click as you always have or you write your own KCL code, you are performing the same action in KittyCAD Modeling App | ||||||
| - Everything graphics _has_ to be built for the GPU | - Everything graphics _has_ to be built for the GPU | ||||||
|   - Most CAD applications have had to retrofit support for GPUs, but our geometry engine is made for GPUs (primarily Nvidia's Vulkan), getting the order of magnitude rendering performance boost with it |   - Most CAD applications have had to retrofit support for GPUs, but our geometry engine is made for GPUs (primarily Nvidia's Vulkan), getting the order of magnitude rendering performance boost with it | ||||||
| - Make the resource-intensive pieces of an application auto-scaling | - Make the resource-intensive pieces of an application auto-scaling | ||||||
| @ -19,9 +19,9 @@ Modeling App is our take on what a modern modelling experience can be. It is app | |||||||
|  |  | ||||||
| We are excited about what a small team of people could build in a short time with our API. We welcome you to try our API, build your own applications, or contribute to ours! | We are excited about what a small team of people could build in a short time with our API. We welcome you to try our API, build your own applications, or contribute to ours! | ||||||
|  |  | ||||||
| Modeling App is a _hybrid_ user interface for CAD modeling. You can point-and-click to design parts (and soon assemblies), but everything you make is really just [`kcl` code](https://github.com/KittyCAD/kcl-experiments) under the hood. All of your CAD models can be checked into source control such as GitHub and responsibly versioned, rolled back, and more. | KittyCAD Modeling App is a _hybrid_ user interface for CAD modeling. You can point-and-click to design parts (and soon assemblies), but everything you make is really just [`kcl` code](https://github.com/KittyCAD/kcl-experiments) under the hood. All of your CAD models can be checked into source control such as GitHub and responsibly versioned, rolled back, and more. | ||||||
|  |  | ||||||
| The 3D view in Modeling App is just a video stream from our hosted geometry engine. The app sends new modeling commands to the engine via WebSockets, which returns back video frames of the view within the engine. | The 3D view in KittyCAD Modeling App is just a video stream from our hosted geometry engine. The app sends new modeling commands to the engine via WebSockets, which returns back video frames of the view within the engine. | ||||||
|  |  | ||||||
| ## Tools | ## Tools | ||||||
|  |  | ||||||
| @ -94,6 +94,7 @@ For running the rust (not tauri rust though) only, you can | |||||||
| cd src/wasm-lib | cd src/wasm-lib | ||||||
| cargo test | cargo test | ||||||
| ``` | ``` | ||||||
|  | but you will need to have install ffmpeg prior to. | ||||||
|  |  | ||||||
| ## Tauri | ## Tauri | ||||||
|  |  | ||||||
| @ -182,9 +183,9 @@ For more information on fuzzing you can check out | |||||||
| First time running plawright locally, you'll need to add the secrets file | First time running plawright locally, you'll need to add the secrets file | ||||||
| ```bash | ```bash | ||||||
| touch ./e2e/playwright/playwright-secrets.env | touch ./e2e/playwright/playwright-secrets.env | ||||||
| printf 'token="your-token"\nsnapshottoken="your-snapshot-token"' > ./e2e/playwright/playwright-secrets.env | echo 'token="your-token"' > ./e2e/playwright/playwright-secrets.env | ||||||
| ``` | ``` | ||||||
| then replace "your-token" with a dev token from dev.zoo.dev/account/api-tokens | then replace "your-token" with a dev token from dev.kittycad.io/account/api-tokens | ||||||
|  |  | ||||||
| then: | then: | ||||||
| run playwright | run playwright | ||||||
|  | |||||||
							
								
								
									
										
											BIN
										
									
								
								app-icon.png
									
									
									
									
									
								
							
							
						
						| Before Width: | Height: | Size: 120 KiB After Width: | Height: | Size: 207 KiB | 
| @ -1,16 +0,0 @@ | |||||||
| # Known Issues |  | ||||||
|  |  | ||||||
| The following are bugs that are not in modeling-app or kcl itself. These bugs |  | ||||||
| once fixed in engine will just start working here with no language changes. |  | ||||||
|  |  | ||||||
| - **Sketch on Face**: If your sketch is outside the edges of the face (on which you |  | ||||||
|     are sketching) you will get multiple models returned instead of one single |  | ||||||
|     model for that sketch and its underlying 3D object. |  | ||||||
|  |  | ||||||
| - **Patterns**: If you try and pass a pattern to `hole` currently only the first |  | ||||||
|     item in the pattern is being subtracted. This is an engine bug that is being |  | ||||||
|     worked on. |  | ||||||
|    |  | ||||||
| - **Import**: Right now you can import a file, even if that file has brep data |  | ||||||
|     you cannot edit it. You also cannot move or transform the imported objects at |  | ||||||
|    all. In the future, after v1, the engine will account for this. |  | ||||||
							
								
								
									
										31629
									
								
								docs/kcl/std.json
									
									
									
									
									
								
							
							
						
						
							
								
								
									
										5805
									
								
								docs/kcl/std.md
									
									
									
									
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								e2e/playwright/export-snapshots/gltf-binary.gltf
									
									
									
									
									
										Normal file
									
								
							
							
						
						| Before Width: | Height: | Size: 80 KiB | 
							
								
								
									
										3018
									
								
								e2e/playwright/export-snapshots/gltf-embedded.gltf
									
									
									
									
									
										Normal file
									
								
							
							
						
						| Before Width: | Height: | Size: 80 KiB | 
							
								
								
									
										189
									
								
								e2e/playwright/export-snapshots/obj-.obj
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,189 @@ | |||||||
|  | v 0 -4 0 | ||||||
|  | v 0 0 0 | ||||||
|  | v 0 -4 -1 | ||||||
|  | v 0 0 -1 | ||||||
|  | v 3.0950184 -4 -1 | ||||||
|  | v 3.0950184 0 -1 | ||||||
|  | v 5.9513144 -4 -3 | ||||||
|  | v 5.9513144 0 -3 | ||||||
|  | v 9.5 -4 -3 | ||||||
|  | v 9.5 0 -3 | ||||||
|  | v 9.5 -4 -2.5 | ||||||
|  | v 9.5 0 -2.5 | ||||||
|  | v 6.108964 -4 -2.5 | ||||||
|  | v 6.108964 0 -2.5 | ||||||
|  | v 3.4311862 -4 -0.625 | ||||||
|  | v 4.323779 -4 -1.25 | ||||||
|  | v 4.323779 -0 -1.25 | ||||||
|  | v 3.4311862 -0 -0.625 | ||||||
|  | v 2.5385938 0 0 | ||||||
|  | v 2.5385938 -4 0 | ||||||
|  | v 3.342784 -4 0.375 | ||||||
|  | v 4.146974 -4 0.75 | ||||||
|  | v 3.342784 -0 0.375 | ||||||
|  | v 4.146974 -0 0.75 | ||||||
|  | v 5.755354 0 1.5 | ||||||
|  | v 5.755354 -4 1.5 | ||||||
|  | v 9.5 -4 1.5 | ||||||
|  | v 9.5 0 1.5 | ||||||
|  | v 9.5 -4 2 | ||||||
|  | v 9.5 0 2 | ||||||
|  | v 5.644507 -4 2 | ||||||
|  | v 5.644507 0 2 | ||||||
|  | v 3.5 -4 1 | ||||||
|  | v 3.5 0 1 | ||||||
|  | v 0 -4 1 | ||||||
|  | v 0 0 1 | ||||||
|  | vt 0.0127 -0.0508 | ||||||
|  | vt 0.0127 0.0508 | ||||||
|  | vt -0.0127 -0.0508 | ||||||
|  | vt -0.0127 0.0508 | ||||||
|  | vt -0.039306734 0.0508 | ||||||
|  | vt -0.039306734 -0.0508 | ||||||
|  | vt 0.039306734 0.0508 | ||||||
|  | vt 0.039306734 -0.0508 | ||||||
|  | vt -0.04428355 0.0508 | ||||||
|  | vt -0.04428355 -0.0508 | ||||||
|  | vt 0.04428355 0.0508 | ||||||
|  | vt 0.04428355 -0.0508 | ||||||
|  | vt -0.045068305 0.0508 | ||||||
|  | vt -0.045068305 -0.0508 | ||||||
|  | vt 0.045068305 0.0508 | ||||||
|  | vt 0.045068305 -0.0508 | ||||||
|  | vt -0.00635 0.0508 | ||||||
|  | vt -0.00635 -0.0508 | ||||||
|  | vt 0.00635 0.0508 | ||||||
|  | vt 0.00635 -0.0508 | ||||||
|  | vt 0.04306616 -0.0508 | ||||||
|  | vt 0.04306616 0.0508 | ||||||
|  | vt -0.04306616 -0.0508 | ||||||
|  | vt -0.04306616 0.0508 | ||||||
|  | vt -0.027677217 -0.0508 | ||||||
|  | vt 0.000000000000000048572257 -0.0508 | ||||||
|  | vt 0.000000000000000048572257 0.0508 | ||||||
|  | vt 0.055354435 -0.0508 | ||||||
|  | vt 0.055354435 0.0508 | ||||||
|  | vt -0.027677217 0.0508 | ||||||
|  | vt -0.055354435 0.0508 | ||||||
|  | vt -0.055354435 -0.0508 | ||||||
|  | vt -0.02253807 0.0508 | ||||||
|  | vt -0.04507614 0.0508 | ||||||
|  | vt -0.04507614 -0.0508 | ||||||
|  | vt 0.00000000000000005551115 0.0508 | ||||||
|  | vt -0.02253807 -0.0508 | ||||||
|  | vt 0.00000000000000005551115 -0.0508 | ||||||
|  | vt 0.04507614 -0.0508 | ||||||
|  | vt 0.04507614 0.0508 | ||||||
|  | vt -0.047557 0.0508 | ||||||
|  | vt -0.047557 -0.0508 | ||||||
|  | vt 0.047557 0.0508 | ||||||
|  | vt 0.047557 -0.0508 | ||||||
|  | vt 0.04896476 -0.0508 | ||||||
|  | vt 0.04896476 0.0508 | ||||||
|  | vt -0.04896476 -0.0508 | ||||||
|  | vt -0.04896476 0.0508 | ||||||
|  | vt 0.03005076 -0.0508 | ||||||
|  | vt 0.03005076 0.0508 | ||||||
|  | vt -0.03005076 -0.0508 | ||||||
|  | vt -0.03005076 0.0508 | ||||||
|  | vt 0.04445 -0.0508 | ||||||
|  | vt 0.04445 0.0508 | ||||||
|  | vt -0.04445 -0.0508 | ||||||
|  | vt -0.04445 0.0508 | ||||||
|  | vt 0.08490671 0.009525 | ||||||
|  | vt 0.06448028 0 | ||||||
|  | vt 0.0889 0.0254 | ||||||
|  | vt 0.08715213 -0.015875 | ||||||
|  | vt 0.10982399 -0.03175 | ||||||
|  | vt 0.07861347 -0.0254 | ||||||
|  | vt 0.10533314 0.01905 | ||||||
|  | vt 0.15116338 -0.0762 | ||||||
|  | vt 0 -0.0254 | ||||||
|  | vt 0 0 | ||||||
|  | vt 0.2413 -0.0762 | ||||||
|  | vt 0.15516768 -0.0635 | ||||||
|  | vt 0.2413 -0.0635 | ||||||
|  | vt 0.14337048 0.0508 | ||||||
|  | vt 0.146186 0.0381 | ||||||
|  | vt 0.2413 0.0381 | ||||||
|  | vt 0.2413 0.0508 | ||||||
|  | vt 0 0.0254 | ||||||
|  | vn -1 -0 0 | ||||||
|  | vn 0 -0 -1 | ||||||
|  | vn -0.57357645 -0 -0.81915206 | ||||||
|  | vn 1 -0 0 | ||||||
|  | vn 0 -0 1 | ||||||
|  | vn 0.57357645 -0 0.81915206 | ||||||
|  | vn 0.42261827 -0 -0.9063078 | ||||||
|  | vn -0.42261827 -0 0.9063078 | ||||||
|  | vn -0 1 -0 | ||||||
|  | vn 0 -1 0 | ||||||
|  | o Unnamed-0 | ||||||
|  | f 1/1/1 2/2/1 3/3/1 | ||||||
|  | f 3/3/1 2/2/1 4/4/1 | ||||||
|  | f 3/5/2 4/6/2 5/7/2 | ||||||
|  | f 5/7/2 4/6/2 6/8/2 | ||||||
|  | f 5/9/3 6/10/3 7/11/3 | ||||||
|  | f 7/11/3 6/10/3 8/12/3 | ||||||
|  | f 7/13/2 8/14/2 9/15/2 | ||||||
|  | f 9/15/2 8/14/2 10/16/2 | ||||||
|  | f 9/17/4 10/18/4 11/19/4 | ||||||
|  | f 11/19/4 10/18/4 12/20/4 | ||||||
|  | f 11/21/5 12/22/5 13/23/5 | ||||||
|  | f 13/23/5 12/22/5 14/24/5 | ||||||
|  | f 15/25/6 16/26/6 17/27/6 | ||||||
|  | f 16/26/6 13/28/6 14/29/6 | ||||||
|  | f 18/30/6 19/31/6 20/32/6 | ||||||
|  | f 15/25/6 18/30/6 20/32/6 | ||||||
|  | f 16/26/6 14/29/6 17/27/6 | ||||||
|  | f 18/30/6 15/25/6 17/27/6 | ||||||
|  | f 21/33/7 20/34/7 19/35/7 | ||||||
|  | f 22/36/7 21/33/7 23/37/7 | ||||||
|  | f 23/37/7 24/38/7 22/36/7 | ||||||
|  | f 24/38/7 25/39/7 26/40/7 | ||||||
|  | f 21/33/7 19/35/7 23/37/7 | ||||||
|  | f 26/40/7 22/36/7 24/38/7 | ||||||
|  | f 26/41/2 25/42/2 27/43/2 | ||||||
|  | f 27/43/2 25/42/2 28/44/2 | ||||||
|  | f 27/17/4 28/18/4 29/19/4 | ||||||
|  | f 29/19/4 28/18/4 30/20/4 | ||||||
|  | f 29/45/5 30/46/5 31/47/5 | ||||||
|  | f 31/47/5 30/46/5 32/48/5 | ||||||
|  | f 31/49/8 32/50/8 33/51/8 | ||||||
|  | f 33/51/8 32/50/8 34/52/8 | ||||||
|  | f 33/53/5 34/54/5 35/55/5 | ||||||
|  | f 35/55/5 34/54/5 36/56/5 | ||||||
|  | f 35/1/1 36/2/1 1/3/1 | ||||||
|  | f 1/3/1 36/2/1 2/4/1 | ||||||
|  | f 23/57/9 19/58/9 34/59/9 | ||||||
|  | f 18/60/9 17/61/9 6/62/9 | ||||||
|  | f 23/57/9 34/59/9 24/63/9 | ||||||
|  | f 17/61/9 8/64/9 6/62/9 | ||||||
|  | f 4/65/9 19/58/9 6/62/9 | ||||||
|  | f 4/65/9 2/66/9 19/58/9 | ||||||
|  | f 10/67/9 14/68/9 12/69/9 | ||||||
|  | f 10/67/9 8/64/9 14/68/9 | ||||||
|  | f 8/64/9 17/61/9 14/68/9 | ||||||
|  | f 32/70/9 25/71/9 24/63/9 | ||||||
|  | f 6/62/9 19/58/9 18/60/9 | ||||||
|  | f 24/63/9 34/59/9 32/70/9 | ||||||
|  | f 28/72/9 25/71/9 30/73/9 | ||||||
|  | f 25/71/9 32/70/9 30/73/9 | ||||||
|  | f 19/58/9 2/66/9 36/74/9 | ||||||
|  | f 34/59/9 19/58/9 36/74/9 | ||||||
|  | f 21/57/10 33/59/10 20/58/10 | ||||||
|  | f 22/63/10 33/59/10 21/57/10 | ||||||
|  | f 15/60/10 5/62/10 16/61/10 | ||||||
|  | f 22/63/10 26/71/10 31/70/10 | ||||||
|  | f 35/74/10 20/58/10 33/59/10 | ||||||
|  | f 35/74/10 1/66/10 20/58/10 | ||||||
|  | f 31/70/10 26/71/10 29/73/10 | ||||||
|  | f 29/73/10 26/71/10 27/72/10 | ||||||
|  | f 22/63/10 31/70/10 33/59/10 | ||||||
|  | f 20/58/10 5/62/10 15/60/10 | ||||||
|  | f 16/61/10 5/62/10 7/64/10 | ||||||
|  | f 13/68/10 16/61/10 7/64/10 | ||||||
|  | f 11/69/10 13/68/10 9/67/10 | ||||||
|  | f 13/68/10 7/64/10 9/67/10 | ||||||
|  | f 20/58/10 3/65/10 5/62/10 | ||||||
|  | f 3/65/10 20/58/10 1/66/10 | ||||||
| Before Width: | Height: | Size: 56 KiB | 
							
								
								
									
										282
									
								
								e2e/playwright/export-snapshots/ply-ascii.ply
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,282 @@ | |||||||
|  | ply | ||||||
|  | format ascii 1.0 | ||||||
|  | comment Generated by kittycad.io | ||||||
|  | element vertex 204 | ||||||
|  | property float x | ||||||
|  | property float y | ||||||
|  | property float z | ||||||
|  | element face 68 | ||||||
|  | property list uchar uint vertex_indices | ||||||
|  | end_header | ||||||
|  | 0 0 4 | ||||||
|  | 0 0 0 | ||||||
|  | 0 -1 4 | ||||||
|  | 0 -1 4 | ||||||
|  | 0 0 0 | ||||||
|  | 0 -1 0 | ||||||
|  | 0 -1 4 | ||||||
|  | 0 -1 0 | ||||||
|  | 3.0950184 -1 4 | ||||||
|  | 3.0950184 -1 4 | ||||||
|  | 0 -1 0 | ||||||
|  | 3.0950184 -1 0 | ||||||
|  | 3.0950184 -1 4 | ||||||
|  | 3.0950184 -1 0 | ||||||
|  | 5.9513144 -3 4 | ||||||
|  | 5.9513144 -3 4 | ||||||
|  | 3.0950184 -1 0 | ||||||
|  | 5.9513144 -3 0 | ||||||
|  | 5.9513144 -3 4 | ||||||
|  | 5.9513144 -3 0 | ||||||
|  | 9.5 -3 4 | ||||||
|  | 9.5 -3 4 | ||||||
|  | 5.9513144 -3 0 | ||||||
|  | 9.5 -3 0 | ||||||
|  | 9.5 -3 4 | ||||||
|  | 9.5 -3 0 | ||||||
|  | 9.5 -2.5 4 | ||||||
|  | 9.5 -2.5 4 | ||||||
|  | 9.5 -3 0 | ||||||
|  | 9.5 -2.5 0 | ||||||
|  | 9.5 -2.5 4 | ||||||
|  | 9.5 -2.5 0 | ||||||
|  | 6.108964 -2.5 4 | ||||||
|  | 6.108964 -2.5 4 | ||||||
|  | 9.5 -2.5 0 | ||||||
|  | 6.108964 -2.5 0 | ||||||
|  | 3.4311862 -0.625 4 | ||||||
|  | 4.323779 -1.25 4 | ||||||
|  | 4.323779 -1.25 0 | ||||||
|  | 4.323779 -1.25 4 | ||||||
|  | 6.108964 -2.5 4 | ||||||
|  | 6.108964 -2.5 0 | ||||||
|  | 3.4311862 -0.625 0 | ||||||
|  | 2.5385938 0 0 | ||||||
|  | 2.5385938 0 4 | ||||||
|  | 3.4311862 -0.625 4 | ||||||
|  | 3.4311862 -0.625 0 | ||||||
|  | 2.5385938 0 4 | ||||||
|  | 4.323779 -1.25 4 | ||||||
|  | 6.108964 -2.5 0 | ||||||
|  | 4.323779 -1.25 0 | ||||||
|  | 3.4311862 -0.625 0 | ||||||
|  | 3.4311862 -0.625 4 | ||||||
|  | 4.323779 -1.25 0 | ||||||
|  | 3.342784 0.375 4 | ||||||
|  | 2.5385938 0 4 | ||||||
|  | 2.5385938 0 0 | ||||||
|  | 4.146974 0.75 4 | ||||||
|  | 3.342784 0.375 4 | ||||||
|  | 3.342784 0.375 0 | ||||||
|  | 3.342784 0.375 0 | ||||||
|  | 4.146974 0.75 0 | ||||||
|  | 4.146974 0.75 4 | ||||||
|  | 4.146974 0.75 0 | ||||||
|  | 5.755354 1.5 0 | ||||||
|  | 5.755354 1.5 4 | ||||||
|  | 3.342784 0.375 4 | ||||||
|  | 2.5385938 0 0 | ||||||
|  | 3.342784 0.375 0 | ||||||
|  | 5.755354 1.5 4 | ||||||
|  | 4.146974 0.75 4 | ||||||
|  | 4.146974 0.75 0 | ||||||
|  | 5.755354 1.5 4 | ||||||
|  | 5.755354 1.5 0 | ||||||
|  | 9.5 1.5 4 | ||||||
|  | 9.5 1.5 4 | ||||||
|  | 5.755354 1.5 0 | ||||||
|  | 9.5 1.5 0 | ||||||
|  | 9.5 1.5 4 | ||||||
|  | 9.5 1.5 0 | ||||||
|  | 9.5 2 4 | ||||||
|  | 9.5 2 4 | ||||||
|  | 9.5 1.5 0 | ||||||
|  | 9.5 2 0 | ||||||
|  | 9.5 2 4 | ||||||
|  | 9.5 2 0 | ||||||
|  | 5.644507 2 4 | ||||||
|  | 5.644507 2 4 | ||||||
|  | 9.5 2 0 | ||||||
|  | 5.644507 2 0 | ||||||
|  | 5.644507 2 4 | ||||||
|  | 5.644507 2 0 | ||||||
|  | 3.5 1 4 | ||||||
|  | 3.5 1 4 | ||||||
|  | 5.644507 2 0 | ||||||
|  | 3.5 1 0 | ||||||
|  | 3.5 1 4 | ||||||
|  | 3.5 1 0 | ||||||
|  | 0 1 4 | ||||||
|  | 0 1 4 | ||||||
|  | 3.5 1 0 | ||||||
|  | 0 1 0 | ||||||
|  | 0 1 4 | ||||||
|  | 0 1 0 | ||||||
|  | 0 0 4 | ||||||
|  | 0 0 4 | ||||||
|  | 0 1 0 | ||||||
|  | 0 0 0 | ||||||
|  | 3.342784 0.375 0 | ||||||
|  | 2.5385938 0 0 | ||||||
|  | 3.5 1 0 | ||||||
|  | 3.4311862 -0.625 0 | ||||||
|  | 4.323779 -1.25 0 | ||||||
|  | 3.0950184 -1 0 | ||||||
|  | 3.342784 0.375 0 | ||||||
|  | 3.5 1 0 | ||||||
|  | 4.146974 0.75 0 | ||||||
|  | 4.323779 -1.25 0 | ||||||
|  | 5.9513144 -3 0 | ||||||
|  | 3.0950184 -1 0 | ||||||
|  | 0 -1 0 | ||||||
|  | 2.5385938 0 0 | ||||||
|  | 3.0950184 -1 0 | ||||||
|  | 0 -1 0 | ||||||
|  | 0 0 0 | ||||||
|  | 2.5385938 0 0 | ||||||
|  | 9.5 -3 0 | ||||||
|  | 6.108964 -2.5 0 | ||||||
|  | 9.5 -2.5 0 | ||||||
|  | 9.5 -3 0 | ||||||
|  | 5.9513144 -3 0 | ||||||
|  | 6.108964 -2.5 0 | ||||||
|  | 5.9513144 -3 0 | ||||||
|  | 4.323779 -1.25 0 | ||||||
|  | 6.108964 -2.5 0 | ||||||
|  | 5.644507 2 0 | ||||||
|  | 5.755354 1.5 0 | ||||||
|  | 4.146974 0.75 0 | ||||||
|  | 3.0950184 -1 0 | ||||||
|  | 2.5385938 0 0 | ||||||
|  | 3.4311862 -0.625 0 | ||||||
|  | 4.146974 0.75 0 | ||||||
|  | 3.5 1 0 | ||||||
|  | 5.644507 2 0 | ||||||
|  | 9.5 1.5 0 | ||||||
|  | 5.755354 1.5 0 | ||||||
|  | 9.5 2 0 | ||||||
|  | 5.755354 1.5 0 | ||||||
|  | 5.644507 2 0 | ||||||
|  | 9.5 2 0 | ||||||
|  | 2.5385938 0 0 | ||||||
|  | 0 0 0 | ||||||
|  | 0 1 0 | ||||||
|  | 3.5 1 0 | ||||||
|  | 2.5385938 0 0 | ||||||
|  | 0 1 0 | ||||||
|  | 3.342784 0.375 4 | ||||||
|  | 3.5 1 4 | ||||||
|  | 2.5385938 0 4 | ||||||
|  | 4.146974 0.75 4 | ||||||
|  | 3.5 1 4 | ||||||
|  | 3.342784 0.375 4 | ||||||
|  | 3.4311862 -0.625 4 | ||||||
|  | 3.0950184 -1 4 | ||||||
|  | 4.323779 -1.25 4 | ||||||
|  | 4.146974 0.75 4 | ||||||
|  | 5.755354 1.5 4 | ||||||
|  | 5.644507 2 4 | ||||||
|  | 0 1 4 | ||||||
|  | 2.5385938 0 4 | ||||||
|  | 3.5 1 4 | ||||||
|  | 0 1 4 | ||||||
|  | 0 0 4 | ||||||
|  | 2.5385938 0 4 | ||||||
|  | 5.644507 2 4 | ||||||
|  | 5.755354 1.5 4 | ||||||
|  | 9.5 2 4 | ||||||
|  | 9.5 2 4 | ||||||
|  | 5.755354 1.5 4 | ||||||
|  | 9.5 1.5 4 | ||||||
|  | 4.146974 0.75 4 | ||||||
|  | 5.644507 2 4 | ||||||
|  | 3.5 1 4 | ||||||
|  | 2.5385938 0 4 | ||||||
|  | 3.0950184 -1 4 | ||||||
|  | 3.4311862 -0.625 4 | ||||||
|  | 4.323779 -1.25 4 | ||||||
|  | 3.0950184 -1 4 | ||||||
|  | 5.9513144 -3 4 | ||||||
|  | 6.108964 -2.5 4 | ||||||
|  | 4.323779 -1.25 4 | ||||||
|  | 5.9513144 -3 4 | ||||||
|  | 9.5 -2.5 4 | ||||||
|  | 6.108964 -2.5 4 | ||||||
|  | 9.5 -3 4 | ||||||
|  | 6.108964 -2.5 4 | ||||||
|  | 5.9513144 -3 4 | ||||||
|  | 9.5 -3 4 | ||||||
|  | 2.5385938 0 4 | ||||||
|  | 0 -1 4 | ||||||
|  | 3.0950184 -1 4 | ||||||
|  | 0 -1 4 | ||||||
|  | 2.5385938 0 4 | ||||||
|  | 0 0 4 | ||||||
|  | 3 0 1 2 | ||||||
|  | 3 3 4 5 | ||||||
|  | 3 6 7 8 | ||||||
|  | 3 9 10 11 | ||||||
|  | 3 12 13 14 | ||||||
|  | 3 15 16 17 | ||||||
|  | 3 18 19 20 | ||||||
|  | 3 21 22 23 | ||||||
|  | 3 24 25 26 | ||||||
|  | 3 27 28 29 | ||||||
|  | 3 30 31 32 | ||||||
|  | 3 33 34 35 | ||||||
|  | 3 36 37 38 | ||||||
|  | 3 39 40 41 | ||||||
|  | 3 42 43 44 | ||||||
|  | 3 45 46 47 | ||||||
|  | 3 48 49 50 | ||||||
|  | 3 51 52 53 | ||||||
|  | 3 54 55 56 | ||||||
|  | 3 57 58 59 | ||||||
|  | 3 60 61 62 | ||||||
|  | 3 63 64 65 | ||||||
|  | 3 66 67 68 | ||||||
|  | 3 69 70 71 | ||||||
|  | 3 72 73 74 | ||||||
|  | 3 75 76 77 | ||||||
|  | 3 78 79 80 | ||||||
|  | 3 81 82 83 | ||||||
|  | 3 84 85 86 | ||||||
|  | 3 87 88 89 | ||||||
|  | 3 90 91 92 | ||||||
|  | 3 93 94 95 | ||||||
|  | 3 96 97 98 | ||||||
|  | 3 99 100 101 | ||||||
|  | 3 102 103 104 | ||||||
|  | 3 105 106 107 | ||||||
|  | 3 108 109 110 | ||||||
|  | 3 111 112 113 | ||||||
|  | 3 114 115 116 | ||||||
|  | 3 117 118 119 | ||||||
|  | 3 120 121 122 | ||||||
|  | 3 123 124 125 | ||||||
|  | 3 126 127 128 | ||||||
|  | 3 129 130 131 | ||||||
|  | 3 132 133 134 | ||||||
|  | 3 135 136 137 | ||||||
|  | 3 138 139 140 | ||||||
|  | 3 141 142 143 | ||||||
|  | 3 144 145 146 | ||||||
|  | 3 147 148 149 | ||||||
|  | 3 150 151 152 | ||||||
|  | 3 153 154 155 | ||||||
|  | 3 156 157 158 | ||||||
|  | 3 159 160 161 | ||||||
|  | 3 162 163 164 | ||||||
|  | 3 165 166 167 | ||||||
|  | 3 168 169 170 | ||||||
|  | 3 171 172 173 | ||||||
|  | 3 174 175 176 | ||||||
|  | 3 177 178 179 | ||||||
|  | 3 180 181 182 | ||||||
|  | 3 183 184 185 | ||||||
|  | 3 186 187 188 | ||||||
|  | 3 189 190 191 | ||||||
|  | 3 192 193 194 | ||||||
|  | 3 195 196 197 | ||||||
|  | 3 198 199 200 | ||||||
|  | 3 201 202 203 | ||||||
| Before Width: | Height: | Size: 56 KiB | 
							
								
								
									
										
											BIN
										
									
								
								e2e/playwright/export-snapshots/ply-binary_big_endian.ply
									
									
									
									
									
										Normal file
									
								
							
							
						
						| Before Width: | Height: | Size: 56 KiB | 
							
								
								
									
										
											BIN
										
									
								
								e2e/playwright/export-snapshots/ply-binary_little_endian.ply
									
									
									
									
									
										Normal file
									
								
							
							
						
						| Before Width: | Height: | Size: 56 KiB | 
| Before Width: | Height: | Size: 80 KiB | 
| @ -1,7 +1,7 @@ | |||||||
| ISO-10303-21; | ISO-10303-21; | ||||||
| HEADER; | HEADER; | ||||||
| FILE_DESCRIPTION((('zoo.dev export')), '2;1'); | FILE_DESCRIPTION((('kittycad.io export')), '2;1'); | ||||||
| FILE_NAME('dump.step', '1970-01-01T00:00:00.0+00:00', ('Author unknown'), ('Organization unknown'), 'zoo.dev beta', 'zoo.dev', 'Authorization unknown'); | FILE_NAME('dump.step', '1970-01-01T00:00:00.0+00:00', ('Author unknown'), ('Organization unknown'), 'kittycad.io beta', 'kittycad.io', 'Authorization unknown'); | ||||||
| FILE_SCHEMA(('AP203_CONFIGURATION_CONTROLLED_3D_DESIGN_OF_MECHANICAL_PARTS_AND_ASSEMBLIES_MIM_LF')); | FILE_SCHEMA(('AP203_CONFIGURATION_CONTROLLED_3D_DESIGN_OF_MECHANICAL_PARTS_AND_ASSEMBLIES_MIM_LF')); | ||||||
| ENDSEC; | ENDSEC; | ||||||
| DATA; | DATA; | ||||||
| @ -74,171 +74,171 @@ DATA; | |||||||
| #58 = CARTESIAN_POINT('NONE', (0, 0.0254, 0.1016)); | #58 = CARTESIAN_POINT('NONE', (0, 0.0254, 0.1016)); | ||||||
| #59 = VERTEX_POINT('NONE', #58); | #59 = VERTEX_POINT('NONE', #58); | ||||||
| #60 = DIRECTION('NONE', (0, -1, 0)); | #60 = DIRECTION('NONE', (0, -1, 0)); | ||||||
| #61 = VECTOR('NONE', #60, 1); | #61 = VECTOR('NONE', #60, 0.0254); | ||||||
| #62 = CARTESIAN_POINT('NONE', (0, 0, -0)); | #62 = CARTESIAN_POINT('NONE', (0, 0, -0)); | ||||||
| #63 = LINE('NONE', #62, #61); | #63 = LINE('NONE', #62, #61); | ||||||
| #64 = DIRECTION('NONE', (0, 0, 1)); | #64 = DIRECTION('NONE', (0, 0, 1)); | ||||||
| #65 = VECTOR('NONE', #64, 1); | #65 = VECTOR('NONE', #64, 0.1016); | ||||||
| #66 = CARTESIAN_POINT('NONE', (0, -0.0254, -0)); | #66 = CARTESIAN_POINT('NONE', (0, -0.0254, -0)); | ||||||
| #67 = LINE('NONE', #66, #65); | #67 = LINE('NONE', #66, #65); | ||||||
| #68 = DIRECTION('NONE', (0, -1, 0)); | #68 = DIRECTION('NONE', (0, -1, 0)); | ||||||
| #69 = VECTOR('NONE', #68, 1); | #69 = VECTOR('NONE', #68, 0.0254); | ||||||
| #70 = CARTESIAN_POINT('NONE', (0, 0, 0.1016)); | #70 = CARTESIAN_POINT('NONE', (0, 0, 0.1016)); | ||||||
| #71 = LINE('NONE', #70, #69); | #71 = LINE('NONE', #70, #69); | ||||||
| #72 = DIRECTION('NONE', (0, 0, 1)); | #72 = DIRECTION('NONE', (0, 0, 1)); | ||||||
| #73 = VECTOR('NONE', #72, 1); | #73 = VECTOR('NONE', #72, 0.1016); | ||||||
| #74 = CARTESIAN_POINT('NONE', (0, 0, -0)); | #74 = CARTESIAN_POINT('NONE', (0, 0, -0)); | ||||||
| #75 = LINE('NONE', #74, #73); | #75 = LINE('NONE', #74, #73); | ||||||
| #76 = DIRECTION('NONE', (1, 0, 0)); | #76 = DIRECTION('NONE', (1, 0, 0)); | ||||||
| #77 = VECTOR('NONE', #76, 1); | #77 = VECTOR('NONE', #76, 0.07861346939195568); | ||||||
| #78 = CARTESIAN_POINT('NONE', (0, -0.0254, -0)); | #78 = CARTESIAN_POINT('NONE', (0, -0.0254, -0)); | ||||||
| #79 = LINE('NONE', #78, #77); | #79 = LINE('NONE', #78, #77); | ||||||
| #80 = DIRECTION('NONE', (0, 0, 1)); | #80 = DIRECTION('NONE', (0, 0, 1)); | ||||||
| #81 = VECTOR('NONE', #80, 1); | #81 = VECTOR('NONE', #80, 0.1016); | ||||||
| #82 = CARTESIAN_POINT('NONE', (0.07861346939195568, -0.0254, -0)); | #82 = CARTESIAN_POINT('NONE', (0.07861346939195568, -0.0254, -0)); | ||||||
| #83 = LINE('NONE', #82, #81); | #83 = LINE('NONE', #82, #81); | ||||||
| #84 = DIRECTION('NONE', (1, 0, 0)); | #84 = DIRECTION('NONE', (1, 0, 0)); | ||||||
| #85 = VECTOR('NONE', #84, 1); | #85 = VECTOR('NONE', #84, 0.07861346939195568); | ||||||
| #86 = CARTESIAN_POINT('NONE', (0, -0.0254, 0.1016)); | #86 = CARTESIAN_POINT('NONE', (0, -0.0254, 0.1016)); | ||||||
| #87 = LINE('NONE', #86, #85); | #87 = LINE('NONE', #86, #85); | ||||||
| #88 = DIRECTION('NONE', (0.8191520442889919, -0.5735764363510459, 0)); | #88 = DIRECTION('NONE', (0.8191520442889919, -0.5735764363510459, 0)); | ||||||
| #89 = VECTOR('NONE', #88, 1); | #89 = VECTOR('NONE', #88, 0.08856709721755177); | ||||||
| #90 = CARTESIAN_POINT('NONE', (0.07861346939195568, -0.0254, -0)); | #90 = CARTESIAN_POINT('NONE', (0.07861346939195568, -0.0254, -0)); | ||||||
| #91 = LINE('NONE', #90, #89); | #91 = LINE('NONE', #90, #89); | ||||||
| #92 = DIRECTION('NONE', (0, 0, 1)); | #92 = DIRECTION('NONE', (0, 0, 1)); | ||||||
| #93 = VECTOR('NONE', #92, 1); | #93 = VECTOR('NONE', #92, 0.1016); | ||||||
| #94 = CARTESIAN_POINT('NONE', (0.1511633881344551, -0.07619999999999998, -0)); | #94 = CARTESIAN_POINT('NONE', (0.1511633881344551, -0.07619999999999998, -0)); | ||||||
| #95 = LINE('NONE', #94, #93); | #95 = LINE('NONE', #94, #93); | ||||||
| #96 = DIRECTION('NONE', (0.8191520442889919, -0.5735764363510459, 0)); | #96 = DIRECTION('NONE', (0.8191520442889919, -0.5735764363510459, 0)); | ||||||
| #97 = VECTOR('NONE', #96, 1); | #97 = VECTOR('NONE', #96, 0.08856709721755177); | ||||||
| #98 = CARTESIAN_POINT('NONE', (0.07861346939195568, -0.0254, 0.1016)); | #98 = CARTESIAN_POINT('NONE', (0.07861346939195568, -0.0254, 0.1016)); | ||||||
| #99 = LINE('NONE', #98, #97); | #99 = LINE('NONE', #98, #97); | ||||||
| #100 = DIRECTION('NONE', (1, -0.0000000000000003079278779307945, 0)); | #100 = DIRECTION('NONE', (1, -0.0000000000000003079278779307945, 0)); | ||||||
| #101 = VECTOR('NONE', #100, 1); | #101 = VECTOR('NONE', #100, 0.09013661186554489); | ||||||
| #102 = CARTESIAN_POINT('NONE', (0.1511633881344551, -0.07619999999999998, -0)); | #102 = CARTESIAN_POINT('NONE', (0.1511633881344551, -0.07619999999999998, -0)); | ||||||
| #103 = LINE('NONE', #102, #101); | #103 = LINE('NONE', #102, #101); | ||||||
| #104 = DIRECTION('NONE', (0, 0, 1)); | #104 = DIRECTION('NONE', (0, 0, 1)); | ||||||
| #105 = VECTOR('NONE', #104, 1); | #105 = VECTOR('NONE', #104, 0.1016); | ||||||
| #106 = CARTESIAN_POINT('NONE', (0.2413, -0.0762, -0)); | #106 = CARTESIAN_POINT('NONE', (0.2413, -0.0762, -0)); | ||||||
| #107 = LINE('NONE', #106, #105); | #107 = LINE('NONE', #106, #105); | ||||||
| #108 = DIRECTION('NONE', (1, -0.0000000000000003079278779307945, 0)); | #108 = DIRECTION('NONE', (1, -0.0000000000000003079278779307945, 0)); | ||||||
| #109 = VECTOR('NONE', #108, 1); | #109 = VECTOR('NONE', #108, 0.09013661186554489); | ||||||
| #110 = CARTESIAN_POINT('NONE', (0.1511633881344551, -0.07619999999999998, 0.1016)); | #110 = CARTESIAN_POINT('NONE', (0.1511633881344551, -0.07619999999999998, 0.1016)); | ||||||
| #111 = LINE('NONE', #110, #109); | #111 = LINE('NONE', #110, #109); | ||||||
| #112 = DIRECTION('NONE', (0, 1, 0)); | #112 = DIRECTION('NONE', (0, 1, 0)); | ||||||
| #113 = VECTOR('NONE', #112, 1); | #113 = VECTOR('NONE', #112, 0.012700000000000003); | ||||||
| #114 = CARTESIAN_POINT('NONE', (0.2413, -0.0762, -0)); | #114 = CARTESIAN_POINT('NONE', (0.2413, -0.0762, -0)); | ||||||
| #115 = LINE('NONE', #114, #113); | #115 = LINE('NONE', #114, #113); | ||||||
| #116 = DIRECTION('NONE', (0, 0, 1)); | #116 = DIRECTION('NONE', (0, 0, 1)); | ||||||
| #117 = VECTOR('NONE', #116, 1); | #117 = VECTOR('NONE', #116, 0.1016); | ||||||
| #118 = CARTESIAN_POINT('NONE', (0.2413, -0.0635, -0)); | #118 = CARTESIAN_POINT('NONE', (0.2413, -0.0635, -0)); | ||||||
| #119 = LINE('NONE', #118, #117); | #119 = LINE('NONE', #118, #117); | ||||||
| #120 = DIRECTION('NONE', (0, 1, 0)); | #120 = DIRECTION('NONE', (0, 1, 0)); | ||||||
| #121 = VECTOR('NONE', #120, 1); | #121 = VECTOR('NONE', #120, 0.012700000000000003); | ||||||
| #122 = CARTESIAN_POINT('NONE', (0.2413, -0.0762, 0.1016)); | #122 = CARTESIAN_POINT('NONE', (0.2413, -0.0762, 0.1016)); | ||||||
| #123 = LINE('NONE', #122, #121); | #123 = LINE('NONE', #122, #121); | ||||||
| #124 = DIRECTION('NONE', (-1, 0, 0)); | #124 = DIRECTION('NONE', (-1, 0, 0)); | ||||||
| #125 = VECTOR('NONE', #124, 1); | #125 = VECTOR('NONE', #124, 0.08613231724678178); | ||||||
| #126 = CARTESIAN_POINT('NONE', (0.2413, -0.0635, -0)); | #126 = CARTESIAN_POINT('NONE', (0.2413, -0.0635, -0)); | ||||||
| #127 = LINE('NONE', #126, #125); | #127 = LINE('NONE', #126, #125); | ||||||
| #128 = DIRECTION('NONE', (0, 0, 1)); | #128 = DIRECTION('NONE', (0, 0, 1)); | ||||||
| #129 = VECTOR('NONE', #128, 1); | #129 = VECTOR('NONE', #128, 0.1016); | ||||||
| #130 = CARTESIAN_POINT('NONE', (0.1551676827532182, -0.0635, -0)); | #130 = CARTESIAN_POINT('NONE', (0.1551676827532182, -0.0635, -0)); | ||||||
| #131 = LINE('NONE', #130, #129); | #131 = LINE('NONE', #130, #129); | ||||||
| #132 = DIRECTION('NONE', (-1, 0, 0)); | #132 = DIRECTION('NONE', (-1, 0, 0)); | ||||||
| #133 = VECTOR('NONE', #132, 1); | #133 = VECTOR('NONE', #132, 0.08613231724678178); | ||||||
| #134 = CARTESIAN_POINT('NONE', (0.2413, -0.0635, 0.1016)); | #134 = CARTESIAN_POINT('NONE', (0.2413, -0.0635, 0.1016)); | ||||||
| #135 = LINE('NONE', #134, #133); | #135 = LINE('NONE', #134, #133); | ||||||
| #136 = DIRECTION('NONE', (-0.8191520442889919, 0.573576436351046, 0)); | #136 = DIRECTION('NONE', (-0.8191520442889918, 0.573576436351046, 0)); | ||||||
| #137 = VECTOR('NONE', #136, 1); | #137 = VECTOR('NONE', #136, 0.11070887152193974); | ||||||
| #138 = CARTESIAN_POINT('NONE', (0.1551676827532182, -0.0635, -0)); | #138 = CARTESIAN_POINT('NONE', (0.1551676827532182, -0.0635, -0)); | ||||||
| #139 = LINE('NONE', #138, #137); | #139 = LINE('NONE', #138, #137); | ||||||
| #140 = DIRECTION('NONE', (0, 0, 1)); | #140 = DIRECTION('NONE', (0, 0, 1)); | ||||||
| #141 = VECTOR('NONE', #140, 1); | #141 = VECTOR('NONE', #140, 0.1016); | ||||||
| #142 = CARTESIAN_POINT('NONE', (0.06448028432509392, 0, -0)); | #142 = CARTESIAN_POINT('NONE', (0.06448028432509392, 0, -0)); | ||||||
| #143 = LINE('NONE', #142, #141); | #143 = LINE('NONE', #142, #141); | ||||||
| #144 = DIRECTION('NONE', (-0.8191520442889919, 0.573576436351046, 0)); | #144 = DIRECTION('NONE', (-0.8191520442889918, 0.573576436351046, 0)); | ||||||
| #145 = VECTOR('NONE', #144, 1); | #145 = VECTOR('NONE', #144, 0.11070887152193974); | ||||||
| #146 = CARTESIAN_POINT('NONE', (0.1551676827532182, -0.0635, 0.1016)); | #146 = CARTESIAN_POINT('NONE', (0.1551676827532182, -0.0635, 0.1016)); | ||||||
| #147 = LINE('NONE', #146, #145); | #147 = LINE('NONE', #146, #145); | ||||||
| #148 = DIRECTION('NONE', (0.90630778703665, 0.4226182617406993, 0)); | #148 = DIRECTION('NONE', (0.90630778703665, 0.4226182617406993, 0)); | ||||||
| #149 = VECTOR('NONE', #148, 1); | #149 = VECTOR('NONE', #148, 0.09015228031811025); | ||||||
| #150 = CARTESIAN_POINT('NONE', (0.06448028432509392, 0, -0)); | #150 = CARTESIAN_POINT('NONE', (0.06448028432509392, 0, -0)); | ||||||
| #151 = LINE('NONE', #150, #149); | #151 = LINE('NONE', #150, #149); | ||||||
| #152 = DIRECTION('NONE', (0, 0, 1)); | #152 = DIRECTION('NONE', (0, 0, 1)); | ||||||
| #153 = VECTOR('NONE', #152, 1); | #153 = VECTOR('NONE', #152, 0.1016); | ||||||
| #154 = CARTESIAN_POINT('NONE', (0.14618599799650817, 0.03810000000000001, -0)); | #154 = CARTESIAN_POINT('NONE', (0.14618599799650817, 0.03810000000000001, -0)); | ||||||
| #155 = LINE('NONE', #154, #153); | #155 = LINE('NONE', #154, #153); | ||||||
| #156 = DIRECTION('NONE', (0.90630778703665, 0.4226182617406993, 0)); | #156 = DIRECTION('NONE', (0.90630778703665, 0.4226182617406993, 0)); | ||||||
| #157 = VECTOR('NONE', #156, 1); | #157 = VECTOR('NONE', #156, 0.09015228031811025); | ||||||
| #158 = CARTESIAN_POINT('NONE', (0.06448028432509392, 0, 0.1016)); | #158 = CARTESIAN_POINT('NONE', (0.06448028432509392, 0, 0.1016)); | ||||||
| #159 = LINE('NONE', #158, #157); | #159 = LINE('NONE', #158, #157); | ||||||
| #160 = DIRECTION('NONE', (1, -0.00000000000000007295344279228718, 0)); | #160 = DIRECTION('NONE', (1, -0.00000000000000007295344279228718, 0)); | ||||||
| #161 = VECTOR('NONE', #160, 1); | #161 = VECTOR('NONE', #160, 0.09511400200349182); | ||||||
| #162 = CARTESIAN_POINT('NONE', (0.14618599799650817, 0.03810000000000001, -0)); | #162 = CARTESIAN_POINT('NONE', (0.14618599799650817, 0.03810000000000001, -0)); | ||||||
| #163 = LINE('NONE', #162, #161); | #163 = LINE('NONE', #162, #161); | ||||||
| #164 = DIRECTION('NONE', (0, 0, 1)); | #164 = DIRECTION('NONE', (0, 0, 1)); | ||||||
| #165 = VECTOR('NONE', #164, 1); | #165 = VECTOR('NONE', #164, 0.1016); | ||||||
| #166 = CARTESIAN_POINT('NONE', (0.2413, 0.0381, -0)); | #166 = CARTESIAN_POINT('NONE', (0.2413, 0.0381, -0)); | ||||||
| #167 = LINE('NONE', #166, #165); | #167 = LINE('NONE', #166, #165); | ||||||
| #168 = DIRECTION('NONE', (1, -0.00000000000000007295344279228718, 0)); | #168 = DIRECTION('NONE', (1, -0.00000000000000007295344279228718, 0)); | ||||||
| #169 = VECTOR('NONE', #168, 1); | #169 = VECTOR('NONE', #168, 0.09511400200349182); | ||||||
| #170 = CARTESIAN_POINT('NONE', (0.14618599799650817, 0.03810000000000001, 0.1016)); | #170 = CARTESIAN_POINT('NONE', (0.14618599799650817, 0.03810000000000001, 0.1016)); | ||||||
| #171 = LINE('NONE', #170, #169); | #171 = LINE('NONE', #170, #169); | ||||||
| #172 = DIRECTION('NONE', (0, 1, 0)); | #172 = DIRECTION('NONE', (0, 1, 0)); | ||||||
| #173 = VECTOR('NONE', #172, 1); | #173 = VECTOR('NONE', #172, 0.012699999999999996); | ||||||
| #174 = CARTESIAN_POINT('NONE', (0.2413, 0.0381, -0)); | #174 = CARTESIAN_POINT('NONE', (0.2413, 0.0381, -0)); | ||||||
| #175 = LINE('NONE', #174, #173); | #175 = LINE('NONE', #174, #173); | ||||||
| #176 = DIRECTION('NONE', (0, 0, 1)); | #176 = DIRECTION('NONE', (0, 0, 1)); | ||||||
| #177 = VECTOR('NONE', #176, 1); | #177 = VECTOR('NONE', #176, 0.1016); | ||||||
| #178 = CARTESIAN_POINT('NONE', (0.2413, 0.0508, -0)); | #178 = CARTESIAN_POINT('NONE', (0.2413, 0.0508, -0)); | ||||||
| #179 = LINE('NONE', #178, #177); | #179 = LINE('NONE', #178, #177); | ||||||
| #180 = DIRECTION('NONE', (0, 1, 0)); | #180 = DIRECTION('NONE', (0, 1, 0)); | ||||||
| #181 = VECTOR('NONE', #180, 1); | #181 = VECTOR('NONE', #180, 0.012699999999999996); | ||||||
| #182 = CARTESIAN_POINT('NONE', (0.2413, 0.0381, 0.1016)); | #182 = CARTESIAN_POINT('NONE', (0.2413, 0.0381, 0.1016)); | ||||||
| #183 = LINE('NONE', #182, #181); | #183 = LINE('NONE', #182, #181); | ||||||
| #184 = DIRECTION('NONE', (-1, 0, 0)); | #184 = DIRECTION('NONE', (-1, 0, 0)); | ||||||
| #185 = VECTOR('NONE', #184, 1); | #185 = VECTOR('NONE', #184, 0.0979295242190572); | ||||||
| #186 = CARTESIAN_POINT('NONE', (0.2413, 0.0508, -0)); | #186 = CARTESIAN_POINT('NONE', (0.2413, 0.0508, -0)); | ||||||
| #187 = LINE('NONE', #186, #185); | #187 = LINE('NONE', #186, #185); | ||||||
| #188 = DIRECTION('NONE', (0, 0, 1)); | #188 = DIRECTION('NONE', (0, 0, 1)); | ||||||
| #189 = VECTOR('NONE', #188, 1); | #189 = VECTOR('NONE', #188, 0.1016); | ||||||
| #190 = CARTESIAN_POINT('NONE', (0.14337047578094278, 0.0508, -0)); | #190 = CARTESIAN_POINT('NONE', (0.14337047578094278, 0.0508, -0)); | ||||||
| #191 = LINE('NONE', #190, #189); | #191 = LINE('NONE', #190, #189); | ||||||
| #192 = DIRECTION('NONE', (-1, 0, 0)); | #192 = DIRECTION('NONE', (-1, 0, 0)); | ||||||
| #193 = VECTOR('NONE', #192, 1); | #193 = VECTOR('NONE', #192, 0.0979295242190572); | ||||||
| #194 = CARTESIAN_POINT('NONE', (0.2413, 0.0508, 0.1016)); | #194 = CARTESIAN_POINT('NONE', (0.2413, 0.0508, 0.1016)); | ||||||
| #195 = LINE('NONE', #194, #193); | #195 = LINE('NONE', #194, #193); | ||||||
| #196 = DIRECTION('NONE', (-0.90630778703665, -0.42261826174069944, 0)); | #196 = DIRECTION('NONE', (-0.9063077870366499, -0.42261826174069944, 0)); | ||||||
| #197 = VECTOR('NONE', #196, 1); | #197 = VECTOR('NONE', #196, 0.06010152021207346); | ||||||
| #198 = CARTESIAN_POINT('NONE', (0.14337047578094278, 0.0508, -0)); | #198 = CARTESIAN_POINT('NONE', (0.14337047578094278, 0.0508, -0)); | ||||||
| #199 = LINE('NONE', #198, #197); | #199 = LINE('NONE', #198, #197); | ||||||
| #200 = DIRECTION('NONE', (0, 0, 1)); | #200 = DIRECTION('NONE', (0, 0, 1)); | ||||||
| #201 = VECTOR('NONE', #200, 1); | #201 = VECTOR('NONE', #200, 0.1016); | ||||||
| #202 = CARTESIAN_POINT('NONE', (0.08889999999999999, 0.0254, -0)); | #202 = CARTESIAN_POINT('NONE', (0.08889999999999999, 0.0254, -0)); | ||||||
| #203 = LINE('NONE', #202, #201); | #203 = LINE('NONE', #202, #201); | ||||||
| #204 = DIRECTION('NONE', (-0.90630778703665, -0.42261826174069944, 0)); | #204 = DIRECTION('NONE', (-0.9063077870366499, -0.42261826174069944, 0)); | ||||||
| #205 = VECTOR('NONE', #204, 1); | #205 = VECTOR('NONE', #204, 0.06010152021207346); | ||||||
| #206 = CARTESIAN_POINT('NONE', (0.14337047578094278, 0.0508, 0.1016)); | #206 = CARTESIAN_POINT('NONE', (0.14337047578094278, 0.0508, 0.1016)); | ||||||
| #207 = LINE('NONE', #206, #205); | #207 = LINE('NONE', #206, #205); | ||||||
| #208 = DIRECTION('NONE', (-1, 0, 0)); | #208 = DIRECTION('NONE', (-1, 0, 0)); | ||||||
| #209 = VECTOR('NONE', #208, 1); | #209 = VECTOR('NONE', #208, 0.08889999999999999); | ||||||
| #210 = CARTESIAN_POINT('NONE', (0.08889999999999999, 0.0254, -0)); | #210 = CARTESIAN_POINT('NONE', (0.08889999999999999, 0.0254, -0)); | ||||||
| #211 = LINE('NONE', #210, #209); | #211 = LINE('NONE', #210, #209); | ||||||
| #212 = DIRECTION('NONE', (0, 0, 1)); | #212 = DIRECTION('NONE', (0, 0, 1)); | ||||||
| #213 = VECTOR('NONE', #212, 1); | #213 = VECTOR('NONE', #212, 0.1016); | ||||||
| #214 = CARTESIAN_POINT('NONE', (0, 0.0254, -0)); | #214 = CARTESIAN_POINT('NONE', (0, 0.0254, -0)); | ||||||
| #215 = LINE('NONE', #214, #213); | #215 = LINE('NONE', #214, #213); | ||||||
| #216 = DIRECTION('NONE', (-1, 0, 0)); | #216 = DIRECTION('NONE', (-1, 0, 0)); | ||||||
| #217 = VECTOR('NONE', #216, 1); | #217 = VECTOR('NONE', #216, 0.08889999999999999); | ||||||
| #218 = CARTESIAN_POINT('NONE', (0.08889999999999999, 0.0254, 0.1016)); | #218 = CARTESIAN_POINT('NONE', (0.08889999999999999, 0.0254, 0.1016)); | ||||||
| #219 = LINE('NONE', #218, #217); | #219 = LINE('NONE', #218, #217); | ||||||
| #220 = DIRECTION('NONE', (0, -1, 0)); | #220 = DIRECTION('NONE', (0, -1, 0)); | ||||||
| #221 = VECTOR('NONE', #220, 1); | #221 = VECTOR('NONE', #220, 0.0254); | ||||||
| #222 = CARTESIAN_POINT('NONE', (0, 0.0254, -0)); | #222 = CARTESIAN_POINT('NONE', (0, 0.0254, -0)); | ||||||
| #223 = LINE('NONE', #222, #221); | #223 = LINE('NONE', #222, #221); | ||||||
| #224 = DIRECTION('NONE', (0, -1, 0)); | #224 = DIRECTION('NONE', (0, -1, 0)); | ||||||
| #225 = VECTOR('NONE', #224, 1); | #225 = VECTOR('NONE', #224, 0.0254); | ||||||
| #226 = CARTESIAN_POINT('NONE', (0, 0.0254, 0.1016)); | #226 = CARTESIAN_POINT('NONE', (0, 0.0254, 0.1016)); | ||||||
| #227 = LINE('NONE', #226, #225); | #227 = LINE('NONE', #226, #225); | ||||||
| #228 = EDGE_CURVE('NONE', #5, #7, #63, .T.); | #228 = EDGE_CURVE('NONE', #5, #7, #63, .T.); | ||||||
| @ -384,23 +384,23 @@ DATA; | |||||||
| #368 = ORIENTED_EDGE('NONE', *, *, #269, .T.); | #368 = ORIENTED_EDGE('NONE', *, *, #269, .T.); | ||||||
| #369 = EDGE_LOOP('NONE', (#355, #356, #357, #358, #359, #360, #361, #362, #363, #364, #365, #366, #367, #368)); | #369 = EDGE_LOOP('NONE', (#355, #356, #357, #358, #359, #360, #361, #362, #363, #364, #365, #366, #367, #368)); | ||||||
| #370 = CARTESIAN_POINT('NONE', (0, -0.0127, 0.0508)); | #370 = CARTESIAN_POINT('NONE', (0, -0.0127, 0.0508)); | ||||||
| #371 = DIRECTION('NONE', (-1, -0, 0)); | #371 = DIRECTION('NONE', (-1, 0, -0)); | ||||||
| #372 = AXIS2_PLACEMENT_3D('NONE', #370, #371, $); | #372 = AXIS2_PLACEMENT_3D('NONE', #370, #371, $); | ||||||
| #373 = PLANE('NONE', #372); | #373 = PLANE('NONE', #372); | ||||||
| #374 = CARTESIAN_POINT('NONE', (0.039306734695977924, -0.025399999999999995, 0.0508)); | #374 = CARTESIAN_POINT('NONE', (0.039306734695977924, -0.025399999999999995, 0.0508)); | ||||||
| #375 = DIRECTION('NONE', (0, -1, 0)); | #375 = DIRECTION('NONE', (0, -1, -0)); | ||||||
| #376 = AXIS2_PLACEMENT_3D('NONE', #374, #375, $); | #376 = AXIS2_PLACEMENT_3D('NONE', #374, #375, $); | ||||||
| #377 = PLANE('NONE', #376); | #377 = PLANE('NONE', #376); | ||||||
| #378 = CARTESIAN_POINT('NONE', (0.11488842876320533, -0.05079999999999996, 0.05079999999999999)); | #378 = CARTESIAN_POINT('NONE', (0.11488842876320533, -0.05079999999999996, 0.05079999999999999)); | ||||||
| #379 = DIRECTION('NONE', (-0.5735764363510459, -0.8191520442889919, 0)); | #379 = DIRECTION('NONE', (-0.5735764363510459, -0.819152044288992, 0)); | ||||||
| #380 = AXIS2_PLACEMENT_3D('NONE', #378, #379, $); | #380 = AXIS2_PLACEMENT_3D('NONE', #378, #379, $); | ||||||
| #381 = PLANE('NONE', #380); | #381 = PLANE('NONE', #380); | ||||||
| #382 = CARTESIAN_POINT('NONE', (0.19623169406722757, -0.07619999999999999, 0.0508)); | #382 = CARTESIAN_POINT('NONE', (0.19623169406722757, -0.07619999999999999, 0.0508)); | ||||||
| #383 = DIRECTION('NONE', (0, -1, 0)); | #383 = DIRECTION('NONE', (0, -1, -0)); | ||||||
| #384 = AXIS2_PLACEMENT_3D('NONE', #382, #383, $); | #384 = AXIS2_PLACEMENT_3D('NONE', #382, #383, $); | ||||||
| #385 = PLANE('NONE', #384); | #385 = PLANE('NONE', #384); | ||||||
| #386 = CARTESIAN_POINT('NONE', (0.2413, -0.06985, 0.0508)); | #386 = CARTESIAN_POINT('NONE', (0.2413, -0.06985, 0.0508)); | ||||||
| #387 = DIRECTION('NONE', (1, -0, 0)); | #387 = DIRECTION('NONE', (1, 0, -0)); | ||||||
| #388 = AXIS2_PLACEMENT_3D('NONE', #386, #387, $); | #388 = AXIS2_PLACEMENT_3D('NONE', #386, #387, $); | ||||||
| #389 = PLANE('NONE', #388); | #389 = PLANE('NONE', #388); | ||||||
| #390 = CARTESIAN_POINT('NONE', (0.19823384137660915, -0.0635, 0.0508)); | #390 = CARTESIAN_POINT('NONE', (0.19823384137660915, -0.0635, 0.0508)); | ||||||
| @ -408,19 +408,19 @@ DATA; | |||||||
| #392 = AXIS2_PLACEMENT_3D('NONE', #390, #391, $); | #392 = AXIS2_PLACEMENT_3D('NONE', #390, #391, $); | ||||||
| #393 = PLANE('NONE', #392); | #393 = PLANE('NONE', #392); | ||||||
| #394 = CARTESIAN_POINT('NONE', (0.10982398353915601, -0.03174999999999997, 0.0508)); | #394 = CARTESIAN_POINT('NONE', (0.10982398353915601, -0.03174999999999997, 0.0508)); | ||||||
| #395 = DIRECTION('NONE', (0.5735764363510459, 0.8191520442889917, -0)); | #395 = DIRECTION('NONE', (0.573576436351046, 0.8191520442889918, -0)); | ||||||
| #396 = AXIS2_PLACEMENT_3D('NONE', #394, #395, $); | #396 = AXIS2_PLACEMENT_3D('NONE', #394, #395, $); | ||||||
| #397 = PLANE('NONE', #396); | #397 = PLANE('NONE', #396); | ||||||
| #398 = CARTESIAN_POINT('NONE', (0.105333141160801, 0.019049999999999987, 0.0508)); | #398 = CARTESIAN_POINT('NONE', (0.105333141160801, 0.019049999999999987, 0.0508)); | ||||||
| #399 = DIRECTION('NONE', (0.4226182617406993, -0.90630778703665, 0)); | #399 = DIRECTION('NONE', (0.4226182617406993, -0.90630778703665, -0)); | ||||||
| #400 = AXIS2_PLACEMENT_3D('NONE', #398, #399, $); | #400 = AXIS2_PLACEMENT_3D('NONE', #398, #399, $); | ||||||
| #401 = PLANE('NONE', #400); | #401 = PLANE('NONE', #400); | ||||||
| #402 = CARTESIAN_POINT('NONE', (0.19374299899825406, 0.0381, 0.0508)); | #402 = CARTESIAN_POINT('NONE', (0.19374299899825406, 0.0381, 0.0508)); | ||||||
| #403 = DIRECTION('NONE', (0, -1, 0)); | #403 = DIRECTION('NONE', (0, -1, -0)); | ||||||
| #404 = AXIS2_PLACEMENT_3D('NONE', #402, #403, $); | #404 = AXIS2_PLACEMENT_3D('NONE', #402, #403, $); | ||||||
| #405 = PLANE('NONE', #404); | #405 = PLANE('NONE', #404); | ||||||
| #406 = CARTESIAN_POINT('NONE', (0.2413, 0.044449999999999996, 0.0508)); | #406 = CARTESIAN_POINT('NONE', (0.2413, 0.044449999999999996, 0.0508)); | ||||||
| #407 = DIRECTION('NONE', (1, -0, 0)); | #407 = DIRECTION('NONE', (1, 0, -0)); | ||||||
| #408 = AXIS2_PLACEMENT_3D('NONE', #406, #407, $); | #408 = AXIS2_PLACEMENT_3D('NONE', #406, #407, $); | ||||||
| #409 = PLANE('NONE', #408); | #409 = PLANE('NONE', #408); | ||||||
| #410 = CARTESIAN_POINT('NONE', (0.19233523789047138, 0.0508, 0.0508)); | #410 = CARTESIAN_POINT('NONE', (0.19233523789047138, 0.0508, 0.0508)); | ||||||
| @ -436,7 +436,7 @@ DATA; | |||||||
| #420 = AXIS2_PLACEMENT_3D('NONE', #418, #419, $); | #420 = AXIS2_PLACEMENT_3D('NONE', #418, #419, $); | ||||||
| #421 = PLANE('NONE', #420); | #421 = PLANE('NONE', #420); | ||||||
| #422 = CARTESIAN_POINT('NONE', (0, 0.0127, 0.0508)); | #422 = CARTESIAN_POINT('NONE', (0, 0.0127, 0.0508)); | ||||||
| #423 = DIRECTION('NONE', (-1, -0, 0)); | #423 = DIRECTION('NONE', (-1, 0, -0)); | ||||||
| #424 = AXIS2_PLACEMENT_3D('NONE', #422, #423, $); | #424 = AXIS2_PLACEMENT_3D('NONE', #422, #423, $); | ||||||
| #425 = PLANE('NONE', #424); | #425 = PLANE('NONE', #424); | ||||||
| #426 = CARTESIAN_POINT('NONE', (0, 0, -0)); | #426 = CARTESIAN_POINT('NONE', (0, 0, -0)); | ||||||
| @ -475,7 +475,7 @@ DATA; | |||||||
| #459 = ADVANCED_FACE('NONE', (#458), #421, .T.); | #459 = ADVANCED_FACE('NONE', (#458), #421, .T.); | ||||||
| #460 = FACE_OUTER_BOUND('NONE', #339, .T.); | #460 = FACE_OUTER_BOUND('NONE', #339, .T.); | ||||||
| #461 = ADVANCED_FACE('NONE', (#460), #425, .T.); | #461 = ADVANCED_FACE('NONE', (#460), #425, .T.); | ||||||
| #462 = FACE_OUTER_BOUND('NONE', #354, .F.); | #462 = FACE_OUTER_BOUND('NONE', #354, .T.); | ||||||
| #463 = ADVANCED_FACE('NONE', (#462), #429, .F.); | #463 = ADVANCED_FACE('NONE', (#462), #429, .F.); | ||||||
| #464 = FACE_OUTER_BOUND('NONE', #369, .T.); | #464 = FACE_OUTER_BOUND('NONE', #369, .T.); | ||||||
| #465 = ADVANCED_FACE('NONE', (#464), #433, .T.); | #465 = ADVANCED_FACE('NONE', (#464), #433, .T.); | ||||||
|  | |||||||
| Before Width: | Height: | Size: 57 KiB | 
| @ -100,15 +100,15 @@ endfacet | |||||||
| facet normal 0.57357645 0 0.819152 | facet normal 0.57357645 0 0.819152 | ||||||
|     outer loop |     outer loop | ||||||
|         vertex 3.4311862 -0 -0.625 |         vertex 3.4311862 -0 -0.625 | ||||||
|  |         vertex 2.5385938 -0 0 | ||||||
|         vertex 2.5385938 -4 0 |         vertex 2.5385938 -4 0 | ||||||
|         vertex 3.4311862 -4 -0.625 |  | ||||||
|     endloop |     endloop | ||||||
| endfacet | endfacet | ||||||
| facet normal 0.5735763 0 0.8191522 | facet normal 0.57357645 -0 0.819152 | ||||||
|     outer loop |     outer loop | ||||||
|         vertex 4.323779 -0 -1.25 |  | ||||||
|         vertex 3.4311862 -0 -0.625 |  | ||||||
|         vertex 3.4311862 -4 -0.625 |         vertex 3.4311862 -4 -0.625 | ||||||
|  |         vertex 3.4311862 -0 -0.625 | ||||||
|  |         vertex 2.5385938 -4 0 | ||||||
|     endloop |     endloop | ||||||
| endfacet | endfacet | ||||||
| facet normal 0.57357645 -0 0.819152 | facet normal 0.57357645 -0 0.819152 | ||||||
| @ -118,11 +118,11 @@ facet normal 0.57357645 -0 0.819152 | |||||||
|         vertex 4.323779 -0 -1.25 |         vertex 4.323779 -0 -1.25 | ||||||
|     endloop |     endloop | ||||||
| endfacet | endfacet | ||||||
| facet normal 0.57357645 -0 0.819152 | facet normal 0.5735763 0 0.8191522 | ||||||
|     outer loop |     outer loop | ||||||
|         vertex 2.5385938 -4 0 |  | ||||||
|         vertex 3.4311862 -0 -0.625 |         vertex 3.4311862 -0 -0.625 | ||||||
|         vertex 2.5385938 -0 0 |         vertex 3.4311862 -4 -0.625 | ||||||
|  |         vertex 4.323779 -0 -1.25 | ||||||
|     endloop |     endloop | ||||||
| endfacet | endfacet | ||||||
| facet normal 0.42261824 0 -0.9063078 | facet normal 0.42261824 0 -0.9063078 | ||||||
| @ -142,29 +142,29 @@ endfacet | |||||||
| facet normal 0.42261824 0 -0.9063078 | facet normal 0.42261824 0 -0.9063078 | ||||||
|     outer loop |     outer loop | ||||||
|         vertex 3.342784 -0 0.375 |         vertex 3.342784 -0 0.375 | ||||||
|         vertex 3.342784 -4 0.375 |         vertex 4.146974 -0 0.75 | ||||||
|         vertex 2.5385938 -0 0 |         vertex 4.146974 -4 0.75 | ||||||
|     endloop |     endloop | ||||||
| endfacet | endfacet | ||||||
| facet normal 0.42261833 0 -0.90630776 | facet normal 0.42261833 0 -0.90630776 | ||||||
|     outer loop |     outer loop | ||||||
|         vertex 4.146974 -0 0.75 |  | ||||||
|         vertex 5.755354 -4 1.5 |  | ||||||
|         vertex 4.146974 -4 0.75 |  | ||||||
|     endloop |  | ||||||
| endfacet |  | ||||||
| facet normal 0.42261824 0 -0.9063078 |  | ||||||
|     outer loop |  | ||||||
|         vertex 4.146974 -4 0.75 |  | ||||||
|         vertex 3.342784 -0 0.375 |  | ||||||
|         vertex 4.146974 -0 0.75 |  | ||||||
|     endloop |  | ||||||
| endfacet |  | ||||||
| facet normal 0.42261833 0 -0.90630776 |  | ||||||
|     outer loop |  | ||||||
|         vertex 5.755354 -4 1.5 |  | ||||||
|         vertex 4.146974 -0 0.75 |         vertex 4.146974 -0 0.75 | ||||||
|         vertex 5.755354 -0 1.5 |         vertex 5.755354 -0 1.5 | ||||||
|  |         vertex 5.755354 -4 1.5 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0.42261824 0 -0.9063078 | ||||||
|  |     outer loop | ||||||
|  |         vertex 3.342784 -4 0.375 | ||||||
|  |         vertex 2.5385938 -0 0 | ||||||
|  |         vertex 3.342784 -0 0.375 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0.42261833 0 -0.90630776 | ||||||
|  |     outer loop | ||||||
|  |         vertex 5.755354 -4 1.5 | ||||||
|  |         vertex 4.146974 -4 0.75 | ||||||
|  |         vertex 4.146974 -0 0.75 | ||||||
|     endloop |     endloop | ||||||
| endfacet | endfacet | ||||||
| facet normal 0 0 -1 | facet normal 0 0 -1 | ||||||
| @ -258,13 +258,6 @@ facet normal 0 1 -0 | |||||||
|         vertex 3.5 -0 1 |         vertex 3.5 -0 1 | ||||||
|     endloop |     endloop | ||||||
| endfacet | endfacet | ||||||
| facet normal 0 1 0 |  | ||||||
|     outer loop |  | ||||||
|         vertex 4.146974 -0 0.75 |  | ||||||
|         vertex 3.342784 -0 0.375 |  | ||||||
|         vertex 3.5 -0 1 |  | ||||||
|     endloop |  | ||||||
| endfacet |  | ||||||
| facet normal 0 1 0 | facet normal 0 1 0 | ||||||
|     outer loop |     outer loop | ||||||
|         vertex 3.4311862 -0 -0.625 |         vertex 3.4311862 -0 -0.625 | ||||||
| @ -272,39 +265,67 @@ facet normal 0 1 0 | |||||||
|         vertex 3.0950184 -0 -1 |         vertex 3.0950184 -0 -1 | ||||||
|     endloop |     endloop | ||||||
| endfacet | endfacet | ||||||
|  | facet normal 0 1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 3.342784 -0 0.375 | ||||||
|  |         vertex 3.5 -0 1 | ||||||
|  |         vertex 4.146974 -0 0.75 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
| facet normal 0 0.99999994 0 | facet normal 0 0.99999994 0 | ||||||
|     outer loop |     outer loop | ||||||
|         vertex 4.146974 -0 0.75 |         vertex 4.323779 -0 -1.25 | ||||||
|         vertex 5.644507 -0 2 |         vertex 5.9513144 -0 -3 | ||||||
|         vertex 5.755354 -0 1.5 |         vertex 3.0950184 -0 -1 | ||||||
|     endloop |  | ||||||
| endfacet |  | ||||||
| facet normal -0 1 0 |  | ||||||
|     outer loop |  | ||||||
|         vertex 0 -0 1 |  | ||||||
|         vertex 3.5 -0 1 |  | ||||||
|         vertex 2.5385938 -0 0 |  | ||||||
|     endloop |     endloop | ||||||
| endfacet | endfacet | ||||||
| facet normal 0 1 0 | facet normal 0 1 0 | ||||||
|     outer loop |     outer loop | ||||||
|         vertex 0 -0 1 |         vertex 0 -0 -1 | ||||||
|         vertex 2.5385938 -0 0 |         vertex 2.5385938 -0 0 | ||||||
|  |         vertex 3.0950184 -0 -1 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 0 -0 -1 | ||||||
|         vertex 0 -0 0 |         vertex 0 -0 0 | ||||||
|  |         vertex 2.5385938 -0 0 | ||||||
|     endloop |     endloop | ||||||
| endfacet | endfacet | ||||||
| facet normal -0 1 0 | facet normal 0 0.99999994 -0 | ||||||
|     outer loop |     outer loop | ||||||
|         vertex 5.644507 -0 2 |         vertex 9.5 -0 -3 | ||||||
|         vertex 9.5 -0 2 |         vertex 6.108964 -0 -2.5 | ||||||
|         vertex 5.755354 -0 1.5 |         vertex 9.5 -0 -2.5 | ||||||
|     endloop |     endloop | ||||||
| endfacet | endfacet | ||||||
| facet normal 0 1 0 | facet normal 0 1 0 | ||||||
|     outer loop |     outer loop | ||||||
|         vertex 9.5 -0 2 |         vertex 9.5 -0 -3 | ||||||
|         vertex 9.5 -0 1.5 |         vertex 5.9513144 -0 -3 | ||||||
|  |         vertex 6.108964 -0 -2.5 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 1 -0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 5.9513144 -0 -3 | ||||||
|  |         vertex 4.323779 -0 -1.25 | ||||||
|  |         vertex 6.108964 -0 -2.5 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 5.644507 -0 2 | ||||||
|         vertex 5.755354 -0 1.5 |         vertex 5.755354 -0 1.5 | ||||||
|  |         vertex 4.146974 -0 0.75 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 0.99999994 -0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 3.0950184 -0 -1 | ||||||
|  |         vertex 2.5385938 -0 0 | ||||||
|  |         vertex 3.4311862 -0 -0.625 | ||||||
|     endloop |     endloop | ||||||
| endfacet | endfacet | ||||||
| facet normal 0 1 -0 | facet normal 0 1 -0 | ||||||
| @ -314,53 +335,32 @@ facet normal 0 1 -0 | |||||||
|         vertex 5.644507 -0 2 |         vertex 5.644507 -0 2 | ||||||
|     endloop |     endloop | ||||||
| endfacet | endfacet | ||||||
| facet normal 0 0.99999994 0 | facet normal 0 1 -0 | ||||||
|     outer loop |     outer loop | ||||||
|         vertex 2.5385938 -0 0 |         vertex 9.5 -0 1.5 | ||||||
|         vertex 3.4311862 -0 -0.625 |         vertex 5.755354 -0 1.5 | ||||||
|         vertex 3.0950184 -0 -1 |         vertex 9.5 -0 2 | ||||||
|     endloop |     endloop | ||||||
| endfacet | endfacet | ||||||
| facet normal 0 0.99999994 0 | facet normal 0 1 -0 | ||||||
|     outer loop |     outer loop | ||||||
|         vertex 4.323779 -0 -1.25 |         vertex 5.755354 -0 1.5 | ||||||
|         vertex 5.9513144 -0 -3 |         vertex 5.644507 -0 2 | ||||||
|         vertex 3.0950184 -0 -1 |         vertex 9.5 -0 2 | ||||||
|     endloop |  | ||||||
| endfacet |  | ||||||
| facet normal 0 1 0 |  | ||||||
|     outer loop |  | ||||||
|         vertex 6.108964 -0 -2.5 |  | ||||||
|         vertex 5.9513144 -0 -3 |  | ||||||
|         vertex 4.323779 -0 -1.25 |  | ||||||
|     endloop |  | ||||||
| endfacet |  | ||||||
| facet normal 0 0.99999994 0 |  | ||||||
|     outer loop |  | ||||||
|         vertex 9.5 -0 -2.5 |  | ||||||
|         vertex 9.5 -0 -3 |  | ||||||
|         vertex 6.108964 -0 -2.5 |  | ||||||
|     endloop |  | ||||||
| endfacet |  | ||||||
| facet normal 0 1 0 |  | ||||||
|     outer loop |  | ||||||
|         vertex 6.108964 -0 -2.5 |  | ||||||
|         vertex 9.5 -0 -3 |  | ||||||
|         vertex 5.9513144 -0 -3 |  | ||||||
|     endloop |     endloop | ||||||
| endfacet | endfacet | ||||||
| facet normal 0 1 0 | facet normal 0 1 0 | ||||||
|     outer loop |     outer loop | ||||||
|         vertex 2.5385938 -0 0 |         vertex 2.5385938 -0 0 | ||||||
|         vertex 3.0950184 -0 -1 |  | ||||||
|         vertex 0 -0 -1 |  | ||||||
|     endloop |  | ||||||
| endfacet |  | ||||||
| facet normal 0 1 0 |  | ||||||
|     outer loop |  | ||||||
|         vertex 0 -0 -1 |  | ||||||
|         vertex 0 -0 0 |         vertex 0 -0 0 | ||||||
|  |         vertex 0 -0 1 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 3.5 -0 1 | ||||||
|         vertex 2.5385938 -0 0 |         vertex 2.5385938 -0 0 | ||||||
|  |         vertex 0 -0 1 | ||||||
|     endloop |     endloop | ||||||
| endfacet | endfacet | ||||||
| facet normal -0 -1 0 | facet normal -0 -1 0 | ||||||
|  | |||||||
| Before Width: | Height: | Size: 57 KiB | 
| @ -1,5 +1,7 @@ | |||||||
| import { test, expect } from '@playwright/test' | import { test, expect } from '@playwright/test' | ||||||
| import { secrets } from './secrets' | import { secrets } from './secrets' | ||||||
|  | import { EngineCommand } from '../../src/lang/std/engineConnection' | ||||||
|  | import { v4 as uuidv4 } from 'uuid' | ||||||
| import { getUtils } from './test-utils' | import { getUtils } from './test-utils' | ||||||
| import waitOn from 'wait-on' | import waitOn from 'wait-on' | ||||||
| import { Themes } from '../../src/lib/theme' | import { Themes } from '../../src/lib/theme' | ||||||
| @ -51,38 +53,40 @@ test('Basic sketch', async ({ page }) => { | |||||||
|   await page.goto('/') |   await page.goto('/') | ||||||
|   await u.waitForAuthSkipAppStart() |   await u.waitForAuthSkipAppStart() | ||||||
|   await u.openDebugPanel() |   await u.openDebugPanel() | ||||||
|  |   await u.waitForDefaultPlanesVisibilityChange() | ||||||
|  |  | ||||||
|   await expect(page.getByRole('button', { name: 'Start Sketch' })).toBeVisible() |   await expect(page.getByRole('button', { name: 'Start Sketch' })).toBeVisible() | ||||||
|  |  | ||||||
|   // click on "Start Sketch" button |   // click on "Start Sketch" button | ||||||
|   await u.clearCommandLogs() |   await u.clearCommandLogs() | ||||||
|   await u.doAndWaitForImageDiff( |   await Promise.all([ | ||||||
|     () => page.getByRole('button', { name: 'Start Sketch' }).click(), |     u.doAndWaitForImageDiff( | ||||||
|     200 |       () => page.getByRole('button', { name: 'Start Sketch' }).click(), | ||||||
|   ) |       200 | ||||||
|  |     ), | ||||||
|  |     u.waitForDefaultPlanesVisibilityChange(), | ||||||
|  |   ]) | ||||||
|  |  | ||||||
|   // select a plane |   // select a plane | ||||||
|   await page.mouse.click(700, 200) |   await u.doAndWaitForCmd(() => page.mouse.click(700, 200), 'edit_mode_enter') | ||||||
|  |   await u.waitForCmdReceive('set_tool') | ||||||
|  |  | ||||||
|   await expect(page.locator('.cm-content')).toHaveText( |   await u.doAndWaitForCmd( | ||||||
|     `const part001 = startSketchOn('-XZ')` |     () => page.getByRole('button', { name: 'Line' }).click(), | ||||||
|  |     'set_tool' | ||||||
|   ) |   ) | ||||||
|  |  | ||||||
|   await page.waitForTimeout(300) // TODO detect animation ending, or disable animation |  | ||||||
|  |  | ||||||
|   const startXPx = 600 |   const startXPx = 600 | ||||||
|   await page.mouse.click(startXPx + PUR * 10, 500 - PUR * 10) |   await u.doAndWaitForCmd( | ||||||
|   const startAt = '[23.89, -32.23]' |     () => page.mouse.click(startXPx + PUR * 10, 500 - PUR * 10), | ||||||
|   await expect(page.locator('.cm-content')) |     'mouse_click', | ||||||
|     .toHaveText(`const part001 = startSketchOn('-XZ') |     false | ||||||
|   |> startProfileAt(${startAt}, %)`) |   ) | ||||||
|   await page.waitForTimeout(100) |  | ||||||
|  |  | ||||||
|   await u.closeDebugPanel() |  | ||||||
|   await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 10) |   await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 10) | ||||||
|   await page.waitForTimeout(100) |  | ||||||
|  |  | ||||||
|   const num = 24.11 |   const startAt = '[18.26, -24.63]' | ||||||
|  |   const num = '18.43' | ||||||
|   await expect(page.locator('.cm-content')) |   await expect(page.locator('.cm-content')) | ||||||
|     .toHaveText(`const part001 = startSketchOn('-XZ') |     .toHaveText(`const part001 = startSketchOn('-XZ') | ||||||
|   |> startProfileAt(${startAt}, %) |   |> startProfileAt(${startAt}, %) | ||||||
| @ -93,22 +97,27 @@ test('Basic sketch', async ({ page }) => { | |||||||
|     .toHaveText(`const part001 = startSketchOn('-XZ') |     .toHaveText(`const part001 = startSketchOn('-XZ') | ||||||
|   |> startProfileAt(${startAt}, %) |   |> startProfileAt(${startAt}, %) | ||||||
|   |> line([${num}, 0], %) |   |> line([${num}, 0], %) | ||||||
|   |> line([0, ${num + 0.01}], %)`) |   |> line([0, ${num}], %)`) | ||||||
|   await page.mouse.click(startXPx, 500 - PUR * 20) |   await page.mouse.click(startXPx, 500 - PUR * 20) | ||||||
|   await expect(page.locator('.cm-content')) |   await expect(page.locator('.cm-content')) | ||||||
|     .toHaveText(`const part001 = startSketchOn('-XZ') |     .toHaveText(`const part001 = startSketchOn('-XZ') | ||||||
|   |> startProfileAt(${startAt}, %) |   |> startProfileAt(${startAt}, %) | ||||||
|   |> line([${num}, 0], %) |   |> line([${num}, 0], %) | ||||||
|   |> line([0, ${num + 0.01}], %) |   |> line([0, ${num}], %) | ||||||
|   |> line([-48, 0], %)`) |   |> line([-36.69, 0], %)`) | ||||||
|  |  | ||||||
|   // deselect line tool |   // deselect line tool | ||||||
|   await page.getByRole('button', { name: 'Line' }).click() |   await u.doAndWaitForCmd( | ||||||
|   await page.waitForTimeout(100) |     () => page.getByRole('button', { name: 'Line' }).click(), | ||||||
|  |     'set_tool' | ||||||
|  |   ) | ||||||
|  |  | ||||||
|   // click between first two clicks to get center of the line |   // click between first two clicks to get center of the line | ||||||
|   await page.mouse.click(startXPx + PUR * 15, 500 - PUR * 10) |   await u.doAndWaitForCmd( | ||||||
|   await page.waitForTimeout(100) |     () => page.mouse.click(startXPx + PUR * 15, 500 - PUR * 10), | ||||||
|  |     'select_with_point' | ||||||
|  |   ) | ||||||
|  |   await u.closeDebugPanel() | ||||||
|  |  | ||||||
|   // hold down shift |   // hold down shift | ||||||
|   await page.keyboard.down('Shift') |   await page.keyboard.down('Shift') | ||||||
| @ -124,7 +133,7 @@ test('Basic sketch', async ({ page }) => { | |||||||
|     .toHaveText(`const part001 = startSketchOn('-XZ') |     .toHaveText(`const part001 = startSketchOn('-XZ') | ||||||
|   |> startProfileAt(${startAt}, %) |   |> startProfileAt(${startAt}, %) | ||||||
|   |> line({ to: [${num}, 0], tag: 'seg01' }, %) |   |> line({ to: [${num}, 0], tag: 'seg01' }, %) | ||||||
|   |> line([0, ${num + 0.01}], %) |   |> line([0, ${num}], %) | ||||||
|   |> angledLine([180, segLen('seg01', %)], %)`) |   |> angledLine([180, segLen('seg01', %)], %)`) | ||||||
| }) | }) | ||||||
|  |  | ||||||
| @ -197,7 +206,7 @@ test('if you write invalid kcl you get inlined errors', async ({ page }) => { | |||||||
|  |  | ||||||
| test('executes on load', async ({ page, context }) => { | test('executes on load', async ({ page, context }) => { | ||||||
|   const u = getUtils(page) |   const u = getUtils(page) | ||||||
|   await context.addInitScript(async () => { |   await context.addInitScript(async (token) => { | ||||||
|     localStorage.setItem( |     localStorage.setItem( | ||||||
|       'persistCode', |       'persistCode', | ||||||
|       `const part001 = startSketchOn('-XZ') |       `const part001 = startSketchOn('-XZ') | ||||||
| @ -262,41 +271,59 @@ test('Can create sketches on all planes and their back sides', async ({ | |||||||
|   await page.goto('/') |   await page.goto('/') | ||||||
|   await u.waitForAuthSkipAppStart() |   await u.waitForAuthSkipAppStart() | ||||||
|   await u.openDebugPanel() |   await u.openDebugPanel() | ||||||
|  |   await u.waitForDefaultPlanesVisibilityChange() | ||||||
|  |  | ||||||
|   const camPos: [number, number, number] = [100, 100, 100] |   const camCmd: EngineCommand = { | ||||||
|  |     type: 'modeling_cmd_req', | ||||||
|  |     cmd_id: uuidv4(), | ||||||
|  |     cmd: { | ||||||
|  |       type: 'default_camera_look_at', | ||||||
|  |       center: { x: 15, y: 0, z: 0 }, | ||||||
|  |       up: { x: 0, y: 0, z: 1 }, | ||||||
|  |       vantage: { x: 30, y: 30, z: 30 }, | ||||||
|  |     }, | ||||||
|  |   } | ||||||
|  |  | ||||||
|   const TestSinglePlane = async ({ |   const TestSinglePlane = async ({ | ||||||
|     viewCmd, |     viewCmd, | ||||||
|     expectedCode, |     expectedCode, | ||||||
|     clickCoords, |     clickCoords, | ||||||
|   }: { |   }: { | ||||||
|     viewCmd: [number, number, number] |     viewCmd: EngineCommand | ||||||
|     expectedCode: string |     expectedCode: string | ||||||
|     clickCoords: { x: number; y: number } |     clickCoords: { x: number; y: number } | ||||||
|   }) => { |   }) => { | ||||||
|     await u.openDebugPanel() |     await u.openDebugPanel() | ||||||
|  |     await u.sendCustomCmd(viewCmd) | ||||||
|     await u.updateCamPosition(viewCmd) |  | ||||||
|  |  | ||||||
|     await u.clearCommandLogs() |     await u.clearCommandLogs() | ||||||
|  |     // await page.waitForTimeout(200) | ||||||
|     await page.getByRole('button', { name: 'Start Sketch' }).click() |     await page.getByRole('button', { name: 'Start Sketch' }).click() | ||||||
|  |     await u.waitForDefaultPlanesVisibilityChange() | ||||||
|  |  | ||||||
|     await u.closeDebugPanel() |     await u.closeDebugPanel() | ||||||
|     await page.mouse.click(clickCoords.x, clickCoords.y) |     await page.mouse.click(clickCoords.x, clickCoords.y) | ||||||
|     await page.waitForTimeout(300) // wait for animation |     await u.openDebugPanel() | ||||||
|  |  | ||||||
|     await expect(page.getByRole('button', { name: 'Line' })).toBeVisible() |     await expect(page.getByRole('button', { name: 'Line' })).toBeVisible() | ||||||
|  |  | ||||||
|     // draw a line |     // draw a line | ||||||
|     const startXPx = 600 |     const startXPx = 600 | ||||||
|  |     await u.clearCommandLogs() | ||||||
|  |     await page.getByRole('button', { name: 'Line' }).click() | ||||||
|  |     await u.waitForCmdReceive('set_tool') | ||||||
|  |     await u.clearCommandLogs() | ||||||
|     await u.closeDebugPanel() |     await u.closeDebugPanel() | ||||||
|     await page.mouse.click(startXPx + PUR * 10, 500 - PUR * 10) |     await page.mouse.click(startXPx + PUR * 10, 500 - PUR * 10) | ||||||
|  |     await u.openDebugPanel() | ||||||
|  |     await u.waitForCmdReceive('mouse_click') | ||||||
|  |     await u.closeDebugPanel() | ||||||
|  |     await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 10) | ||||||
|  |     await u.openDebugPanel() | ||||||
|  |  | ||||||
|     await expect(page.locator('.cm-content')).toHaveText(expectedCode) |     await expect(page.locator('.cm-content')).toHaveText(expectedCode) | ||||||
|  |  | ||||||
|     await page.getByRole('button', { name: 'Line' }).click() |     await page.getByRole('button', { name: 'Line' }).click() | ||||||
|     await u.openAndClearDebugPanel() |     await u.clearCommandLogs() | ||||||
|     await page.getByRole('button', { name: 'Exit Sketch' }).click() |     await page.getByRole('button', { name: 'Exit Sketch' }).click() | ||||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') |     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||||
|  |  | ||||||
| @ -306,41 +333,51 @@ test('Can create sketches on all planes and their back sides', async ({ | |||||||
|  |  | ||||||
|   const codeTemplate = ( |   const codeTemplate = ( | ||||||
|     plane = 'XY', |     plane = 'XY', | ||||||
|     rounded = false, |     sign = '' | ||||||
|     otherThing = '1' |  | ||||||
|   ) => `const part001 = startSketchOn('${plane}') |   ) => `const part001 = startSketchOn('${plane}') | ||||||
|   |> startProfileAt([28.9${otherThing}, -39${rounded ? '' : '.01'}], %)` |   |> startProfileAt([${sign}6.88, -9.29], %) | ||||||
|  |   |> line([${sign}6.95, 0], %)` | ||||||
|   await TestSinglePlane({ |   await TestSinglePlane({ | ||||||
|     viewCmd: camPos, |     viewCmd: camCmd, | ||||||
|     expectedCode: codeTemplate('XY'), |     expectedCode: codeTemplate('XY'), | ||||||
|     clickCoords: { x: 600, y: 388 }, // red plane |     clickCoords: { x: 700, y: 350 }, // red plane | ||||||
|     // clickCoords: { x: 600, y: 400 }, // red plane // clicks grid helper and that causes problems, should fix so that these coords work too. |  | ||||||
|   }) |   }) | ||||||
|   await TestSinglePlane({ |   await TestSinglePlane({ | ||||||
|     viewCmd: camPos, |     viewCmd: camCmd, | ||||||
|     expectedCode: codeTemplate('YZ', true), |     expectedCode: codeTemplate('YZ'), | ||||||
|     clickCoords: { x: 700, y: 300 }, // green plane |     clickCoords: { x: 1000, y: 200 }, // green plane | ||||||
|   }) |   }) | ||||||
|   await TestSinglePlane({ |   await TestSinglePlane({ | ||||||
|     viewCmd: camPos, |     viewCmd: camCmd, | ||||||
|     expectedCode: codeTemplate('XZ'), |     expectedCode: codeTemplate('XZ', '-'), | ||||||
|     clickCoords: { x: 700, y: 80 }, // blue plane |     clickCoords: { x: 630, y: 130 }, // blue plane | ||||||
|   }) |   }) | ||||||
|   const camCmdBackSide: [number, number, number] = [-100, -100, -100] |  | ||||||
|  |   // new camera angle to click the back side of all three planes | ||||||
|  |   const camCmdBackSide: EngineCommand = { | ||||||
|  |     type: 'modeling_cmd_req', | ||||||
|  |     cmd_id: uuidv4(), | ||||||
|  |     cmd: { | ||||||
|  |       type: 'default_camera_look_at', | ||||||
|  |       center: { x: -15, y: 0, z: 0 }, | ||||||
|  |       up: { x: 0, y: 0, z: 1 }, | ||||||
|  |       vantage: { x: -30, y: -30, z: -30 }, | ||||||
|  |     }, | ||||||
|  |   } | ||||||
|   await TestSinglePlane({ |   await TestSinglePlane({ | ||||||
|     viewCmd: camCmdBackSide, |     viewCmd: camCmdBackSide, | ||||||
|     expectedCode: codeTemplate('-XY', false, '3'), |     expectedCode: codeTemplate('-XY', '-'), | ||||||
|     clickCoords: { x: 601, y: 118 }, // back of red plane |     clickCoords: { x: 705, y: 136 }, // back of red plane | ||||||
|   }) |   }) | ||||||
|   await TestSinglePlane({ |   await TestSinglePlane({ | ||||||
|     viewCmd: camCmdBackSide, |     viewCmd: camCmdBackSide, | ||||||
|     expectedCode: codeTemplate('-YZ'), |     expectedCode: codeTemplate('-YZ', '-'), | ||||||
|     clickCoords: { x: 730, y: 219 }, // back of green plane |     clickCoords: { x: 1000, y: 350 }, // back of green plane | ||||||
|   }) |   }) | ||||||
|   await TestSinglePlane({ |   await TestSinglePlane({ | ||||||
|     viewCmd: camCmdBackSide, |     viewCmd: camCmdBackSide, | ||||||
|     expectedCode: codeTemplate('-XZ', true), |     expectedCode: codeTemplate('-XZ'), | ||||||
|     clickCoords: { x: 680, y: 427 }, // back of blue plane |     clickCoords: { x: 600, y: 400 }, // back of blue plane | ||||||
|   }) |   }) | ||||||
| }) | }) | ||||||
|  |  | ||||||
| @ -350,6 +387,7 @@ test('Auto complete works', async ({ page }) => { | |||||||
|   await page.setViewportSize({ width: 1200, height: 500 }) |   await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|   await page.goto('/') |   await page.goto('/') | ||||||
|   await u.waitForAuthSkipAppStart() |   await u.waitForAuthSkipAppStart() | ||||||
|  |   await u.waitForDefaultPlanesVisibilityChange() | ||||||
|  |  | ||||||
|   // 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. | ||||||
| @ -367,7 +405,6 @@ test('Auto complete works', async ({ page }) => { | |||||||
|   await page.keyboard.type('  |> startProfi') |   await page.keyboard.type('  |> startProfi') | ||||||
|   // expect there be a single auto complete option that we can just hit enter on |   // expect there be a single auto complete option that we can just hit enter on | ||||||
|   await expect(page.locator('.cm-completionLabel')).toBeVisible() |   await expect(page.locator('.cm-completionLabel')).toBeVisible() | ||||||
|   await page.waitForTimeout(100) |  | ||||||
|   await page.keyboard.press('Enter') // accepting the auto complete, not a new line |   await page.keyboard.press('Enter') // accepting the auto complete, not a new line | ||||||
|  |  | ||||||
|   await page.keyboard.type('([0,0], %)') |   await page.keyboard.type('([0,0], %)') | ||||||
| @ -441,36 +478,38 @@ test('Selections work on fresh and edited sketch', async ({ page }) => { | |||||||
|   await page.goto('/') |   await page.goto('/') | ||||||
|   await u.waitForAuthSkipAppStart() |   await u.waitForAuthSkipAppStart() | ||||||
|   await u.openDebugPanel() |   await u.openDebugPanel() | ||||||
|  |   await u.waitForDefaultPlanesVisibilityChange() | ||||||
|  |  | ||||||
|   const xAxisClick = () => |   const xAxisClick = () => page.mouse.click(700, 250) | ||||||
|     page.mouse.click(700, 250).then(() => page.waitForTimeout(100)) |   const emptySpaceClick = () => page.mouse.click(700, 300) | ||||||
|   const emptySpaceClick = () => |   const topHorzSegmentClick = () => page.mouse.click(700, 285) | ||||||
|     page.mouse.click(728, 343).then(() => page.waitForTimeout(100)) |   const bottomHorzSegmentClick = () => page.mouse.click(750, 393) | ||||||
|   const topHorzSegmentClick = () => |  | ||||||
|     page.mouse.click(709, 289).then(() => page.waitForTimeout(100)) |  | ||||||
|   const bottomHorzSegmentClick = () => |  | ||||||
|     page.mouse.click(767, 396).then(() => page.waitForTimeout(100)) |  | ||||||
|  |  | ||||||
|   await u.clearCommandLogs() |   await u.clearCommandLogs() | ||||||
|   await page.getByRole('button', { name: 'Start Sketch' }).click() |   await page.getByRole('button', { name: 'Start Sketch' }).click() | ||||||
|  |   await u.waitForDefaultPlanesVisibilityChange() | ||||||
|  |  | ||||||
|   // select a plane |   // select a plane | ||||||
|   await page.mouse.click(700, 200) |   await u.doAndWaitForCmd(() => page.mouse.click(700, 200), 'edit_mode_enter') | ||||||
|   await page.waitForTimeout(700) // wait for animation |   await u.waitForCmdReceive('set_tool') | ||||||
|  |  | ||||||
|  |   await u.doAndWaitForCmd( | ||||||
|  |     () => page.getByRole('button', { name: 'Line' }).click(), | ||||||
|  |     'set_tool' | ||||||
|  |   ) | ||||||
|  |  | ||||||
|   const startXPx = 600 |   const startXPx = 600 | ||||||
|   await page.mouse.click(startXPx + PUR * 10, 500 - PUR * 10) |   await u.doAndWaitForCmd( | ||||||
|   const startAt = '[23.89, -32.23]' |     () => page.mouse.click(startXPx + PUR * 10, 500 - PUR * 10), | ||||||
|   await expect(page.locator('.cm-content')) |     'mouse_click', | ||||||
|     .toHaveText(`const part001 = startSketchOn('-XZ') |     false | ||||||
|   |> startProfileAt(${startAt}, %)`) |   ) | ||||||
|  |  | ||||||
|   await u.closeDebugPanel() |  | ||||||
|  |  | ||||||
|   await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 10) |   await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 10) | ||||||
|  |  | ||||||
|   const num = 24.11 |   const startAt = '[18.26, -24.63]' | ||||||
|   const num2 = '48' |   const num = '18.43' | ||||||
|  |   const num2 = '36.69' | ||||||
|   await expect(page.locator('.cm-content')) |   await expect(page.locator('.cm-content')) | ||||||
|     .toHaveText(`const part001 = startSketchOn('-XZ') |     .toHaveText(`const part001 = startSketchOn('-XZ') | ||||||
|   |> startProfileAt(${startAt}, %) |   |> startProfileAt(${startAt}, %) | ||||||
| @ -481,17 +520,20 @@ test('Selections work on fresh and edited sketch', async ({ page }) => { | |||||||
|     .toHaveText(`const part001 = startSketchOn('-XZ') |     .toHaveText(`const part001 = startSketchOn('-XZ') | ||||||
|   |> startProfileAt(${startAt}, %) |   |> startProfileAt(${startAt}, %) | ||||||
|   |> line([${num}, 0], %) |   |> line([${num}, 0], %) | ||||||
|   |> line([0, ${num + 0.01}], %)`) |   |> line([0, ${num}], %)`) | ||||||
|   await page.mouse.click(startXPx, 500 - PUR * 20) |   await page.mouse.click(startXPx, 500 - PUR * 20) | ||||||
|   await expect(page.locator('.cm-content')) |   await expect(page.locator('.cm-content')) | ||||||
|     .toHaveText(`const part001 = startSketchOn('-XZ') |     .toHaveText(`const part001 = startSketchOn('-XZ') | ||||||
|   |> startProfileAt(${startAt}, %) |   |> startProfileAt(${startAt}, %) | ||||||
|   |> line([${num}, 0], %) |   |> line([${num}, 0], %) | ||||||
|   |> line([0, ${num + 0.01}], %) |   |> line([0, ${num}], %) | ||||||
|   |> line([-${num2}, 0], %)`) |   |> line([-${num2}, 0], %)`) | ||||||
|  |  | ||||||
|   // deselect line tool |   // deselect line tool | ||||||
|   await page.getByRole('button', { name: 'Line' }).click() |   await u.doAndWaitForCmd( | ||||||
|  |     () => page.getByRole('button', { name: 'Line' }).click(), | ||||||
|  |     'set_tool' | ||||||
|  |   ) | ||||||
|  |  | ||||||
|   await u.closeDebugPanel() |   await u.closeDebugPanel() | ||||||
|   const selectionSequence = async () => { |   const selectionSequence = async () => { | ||||||
| @ -513,72 +555,79 @@ test('Selections work on fresh and edited sketch', async ({ page }) => { | |||||||
|     // now check clicking works including axis |     // now check clicking works including axis | ||||||
|  |  | ||||||
|     // click a segment hold shift and click an axis, see that a relevant constraint is enabled |     // click a segment hold shift and click an axis, see that a relevant constraint is enabled | ||||||
|     await topHorzSegmentClick() |     await u.doAndWaitForCmd(topHorzSegmentClick, 'select_with_point', false) | ||||||
|     await page.keyboard.down('Shift') |     await page.keyboard.down('Shift') | ||||||
|     const absYButton = page.getByRole('button', { name: 'ABS Y' }) |     const absYButton = page.getByRole('button', { name: 'ABS Y' }) | ||||||
|     await expect(absYButton).toBeDisabled() |     await expect(absYButton).toBeDisabled() | ||||||
|     await xAxisClick() |     await u.doAndWaitForCmd(xAxisClick, 'select_with_point', false) | ||||||
|     await page.keyboard.up('Shift') |     await page.keyboard.up('Shift') | ||||||
|     await absYButton.and(page.locator(':not([disabled])')).waitFor() |     await absYButton.and(page.locator(':not([disabled])')).waitFor() | ||||||
|     await expect(absYButton).not.toBeDisabled() |     await expect(absYButton).not.toBeDisabled() | ||||||
|  |  | ||||||
|     // clear selection by clicking on nothing |     // clear selection by clicking on nothing | ||||||
|     await emptySpaceClick() |     await u.doAndWaitForCmd(emptySpaceClick, 'select_clear', false) | ||||||
|  |  | ||||||
|     // same selection but click the axis first |     // same selection but click the axis first | ||||||
|     await xAxisClick() |     await u.doAndWaitForCmd(xAxisClick, 'select_with_point', false) | ||||||
|     await expect(absYButton).toBeDisabled() |     await expect(absYButton).toBeDisabled() | ||||||
|     await page.keyboard.down('Shift') |     await page.keyboard.down('Shift') | ||||||
|     await topHorzSegmentClick() |     await u.doAndWaitForCmd(topHorzSegmentClick, 'select_with_point', false) | ||||||
|  |  | ||||||
|     await page.keyboard.up('Shift') |     await page.keyboard.up('Shift') | ||||||
|     await expect(absYButton).not.toBeDisabled() |     await expect(absYButton).not.toBeDisabled() | ||||||
|  |  | ||||||
|     // clear selection by clicking on nothing |     // clear selection by clicking on nothing | ||||||
|     await emptySpaceClick() |     await u.doAndWaitForCmd(emptySpaceClick, 'select_clear', false) | ||||||
|  |  | ||||||
|     // check the same selection again by putting cursor in code first then selecting axis |     // check the same selection again by putting cursor in code first then selecting axis | ||||||
|     await page.getByText(`  |> line([-${num2}, 0], %)`).click() |     await u.doAndWaitForCmd( | ||||||
|  |       () => page.getByText(`  |> line([-${num2}, 0], %)`).click(), | ||||||
|  |       'select_clear', | ||||||
|  |       false | ||||||
|  |     ) | ||||||
|     await page.keyboard.down('Shift') |     await page.keyboard.down('Shift') | ||||||
|     await expect(absYButton).toBeDisabled() |     await expect(absYButton).toBeDisabled() | ||||||
|     await xAxisClick() |     await u.doAndWaitForCmd(xAxisClick, 'select_with_point', false) | ||||||
|     await page.keyboard.up('Shift') |     await page.keyboard.up('Shift') | ||||||
|     await expect(absYButton).not.toBeDisabled() |     await expect(absYButton).not.toBeDisabled() | ||||||
|  |  | ||||||
|     // clear selection by clicking on nothing |     // clear selection by clicking on nothing | ||||||
|     await emptySpaceClick() |     await u.doAndWaitForCmd(emptySpaceClick, 'select_clear', false) | ||||||
|  |  | ||||||
|     // select segment in editor than another segment in scene and check there are two cursors |     // select segment in editor than another segment in scene and check there are two cursors | ||||||
|     await page.getByText(`  |> line([-${num2}, 0], %)`).click() |     await u.doAndWaitForCmd( | ||||||
|     await page.waitForTimeout(300) |       () => page.getByText(`  |> line([-${num2}, 0], %)`).click(), | ||||||
|  |       'select_clear', | ||||||
|  |       false | ||||||
|  |     ) | ||||||
|     await page.keyboard.down('Shift') |     await page.keyboard.down('Shift') | ||||||
|     await expect(page.locator('.cm-cursor')).toHaveCount(1) |     await expect(page.locator('.cm-cursor')).toHaveCount(1) | ||||||
|     await bottomHorzSegmentClick() |     await u.doAndWaitForCmd(bottomHorzSegmentClick, 'select_with_point', false) // another segment, bottom one | ||||||
|     await page.keyboard.up('Shift') |     await page.keyboard.up('Shift') | ||||||
|     await expect(page.locator('.cm-cursor')).toHaveCount(2) |     await expect(page.locator('.cm-cursor')).toHaveCount(2) | ||||||
|  |  | ||||||
|     // clear selection by clicking on nothing |     // clear selection by clicking on nothing | ||||||
|     await emptySpaceClick() |     await u.doAndWaitForCmd(emptySpaceClick, 'select_clear', false) | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   await selectionSequence() |   await selectionSequence() | ||||||
|  |  | ||||||
|   // hovering in fresh sketch worked, lets try exiting and re-entering |   // hovering in fresh sketch worked, lets try exiting and re-entering | ||||||
|   await u.openAndClearDebugPanel() |   await u.doAndWaitForCmd( | ||||||
|   await page.getByRole('button', { name: 'Exit Sketch' }).click() |     () => page.getByRole('button', { name: 'Exit Sketch' }).click(), | ||||||
|   await page.waitForTimeout(200) |     'edit_mode_exit' | ||||||
|  |   ) | ||||||
|   // wait for execution done |   // wait for execution done | ||||||
|  |  | ||||||
|   await u.expectCmdLog('[data-message-type="execution-done"]') |   await u.expectCmdLog('[data-message-type="execution-done"]') | ||||||
|   await u.closeDebugPanel() |  | ||||||
|  |  | ||||||
|   // select a line |   // select a line | ||||||
|   await topHorzSegmentClick() |   await u.doAndWaitForCmd(topHorzSegmentClick, 'select_clear', false) | ||||||
|   await page.waitForTimeout(200) |  | ||||||
|  |  | ||||||
|   // enter sketch again |   // enter sketch again | ||||||
|   await page.getByRole('button', { name: 'Edit Sketch' }).click() |   await u.doAndWaitForCmd( | ||||||
|   await page.waitForTimeout(700) // wait for animation |     () => page.getByRole('button', { name: 'Start Sketch' }).click(), | ||||||
|  |     'edit_mode_enter', | ||||||
|  |     false | ||||||
|  |   ) | ||||||
|  |  | ||||||
|   // hover again and check it works |   // hover again and check it works | ||||||
|   await selectionSequence() |   await selectionSequence() | ||||||
| @ -648,8 +697,6 @@ test('Can extrude from the command bar', async ({ page, context }) => { | |||||||
|   await page.setViewportSize({ width: 1200, height: 500 }) |   await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|   await page.goto('/') |   await page.goto('/') | ||||||
|   await u.waitForAuthSkipAppStart() |   await u.waitForAuthSkipAppStart() | ||||||
|   await u.openDebugPanel() |  | ||||||
|   await u.expectCmdLog('[data-message-type="execution-done"]') |  | ||||||
|  |  | ||||||
|   let cmdSearchBar = page.getByPlaceholder('Search commands') |   let cmdSearchBar = page.getByPlaceholder('Search commands') | ||||||
|   await page.keyboard.press('Meta+K') |   await page.keyboard.press('Meta+K') | ||||||
| @ -663,7 +710,10 @@ test('Can extrude from the command bar', async ({ page, context }) => { | |||||||
|   await expect(page.getByRole('button', { name: 'selection' })).toBeDisabled() |   await expect(page.getByRole('button', { name: 'selection' })).toBeDisabled() | ||||||
|  |  | ||||||
|   // Click to select face and set distance |   // Click to select face and set distance | ||||||
|  |   await u.openAndClearDebugPanel() | ||||||
|   await page.getByText('|> startProfileAt([-6.95, 4.98], %)').click() |   await page.getByText('|> startProfileAt([-6.95, 4.98], %)').click() | ||||||
|  |   await u.waitForCmdReceive('select_add') | ||||||
|  |   await u.closeDebugPanel() | ||||||
|   await page.getByRole('button', { name: 'Continue' }).click() |   await page.getByRole('button', { name: 'Continue' }).click() | ||||||
|   await expect(page.getByRole('button', { name: 'distance' })).toBeDisabled() |   await expect(page.getByRole('button', { name: 'distance' })).toBeDisabled() | ||||||
|   await page.keyboard.press('Enter') |   await page.keyboard.press('Enter') | ||||||
| @ -685,266 +735,3 @@ test('Can extrude from the command bar', async ({ page, context }) => { | |||||||
|     |> extrude(5, %)` |     |> extrude(5, %)` | ||||||
|   ) |   ) | ||||||
| }) | }) | ||||||
|  |  | ||||||
| test('Can add multiple sketches', async ({ page }) => { |  | ||||||
|   const u = getUtils(page) |  | ||||||
|   await page.setViewportSize({ width: 1200, height: 500 }) |  | ||||||
|   const PUR = 400 / 37.5 //pixeltoUnitRatio |  | ||||||
|   await page.goto('/') |  | ||||||
|   await u.waitForAuthSkipAppStart() |  | ||||||
|   await u.openDebugPanel() |  | ||||||
|  |  | ||||||
|   await expect(page.getByRole('button', { name: 'Start Sketch' })).toBeVisible() |  | ||||||
|  |  | ||||||
|   // click on "Start Sketch" button |  | ||||||
|   await u.clearCommandLogs() |  | ||||||
|   await u.doAndWaitForImageDiff( |  | ||||||
|     () => page.getByRole('button', { name: 'Start Sketch' }).click(), |  | ||||||
|     200 |  | ||||||
|   ) |  | ||||||
|  |  | ||||||
|   // select a plane |  | ||||||
|   await page.mouse.click(700, 200) |  | ||||||
|  |  | ||||||
|   await expect(page.locator('.cm-content')).toHaveText( |  | ||||||
|     `const part001 = startSketchOn('-XZ')` |  | ||||||
|   ) |  | ||||||
|  |  | ||||||
|   await page.waitForTimeout(500) // TODO detect animation ending, or disable animation |  | ||||||
|  |  | ||||||
|   const startXPx = 600 |  | ||||||
|   await page.mouse.click(startXPx + PUR * 10, 500 - PUR * 10) |  | ||||||
|   const startAt = '[23.89, -32.23]' |  | ||||||
|   await expect(page.locator('.cm-content')) |  | ||||||
|     .toHaveText(`const part001 = startSketchOn('-XZ') |  | ||||||
|   |> startProfileAt(${startAt}, %)`) |  | ||||||
|   await page.waitForTimeout(100) |  | ||||||
|  |  | ||||||
|   await u.closeDebugPanel() |  | ||||||
|   await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 10) |  | ||||||
|   await page.waitForTimeout(100) |  | ||||||
|  |  | ||||||
|   const num = 24.11 |  | ||||||
|   await expect(page.locator('.cm-content')) |  | ||||||
|     .toHaveText(`const part001 = startSketchOn('-XZ') |  | ||||||
|   |> startProfileAt(${startAt}, %) |  | ||||||
|   |> line([${num}, 0], %)`) |  | ||||||
|  |  | ||||||
|   await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 20) |  | ||||||
|   await expect(page.locator('.cm-content')) |  | ||||||
|     .toHaveText(`const part001 = startSketchOn('-XZ') |  | ||||||
|   |> startProfileAt(${startAt}, %) |  | ||||||
|   |> line([${num}, 0], %) |  | ||||||
|   |> line([0, ${num + 0.01}], %)`) |  | ||||||
|   await page.mouse.click(startXPx, 500 - PUR * 20) |  | ||||||
|   const finalCodeFirstSketch = `const part001 = startSketchOn('-XZ') |  | ||||||
|   |> startProfileAt(${startAt}, %) |  | ||||||
|   |> line([${num}, 0], %) |  | ||||||
|   |> line([0, ${num + 0.01}], %) |  | ||||||
|   |> line([-48, 0], %)` |  | ||||||
|   await expect(page.locator('.cm-content')).toHaveText(finalCodeFirstSketch) |  | ||||||
|  |  | ||||||
|   // exit the sketch |  | ||||||
|  |  | ||||||
|   await u.openAndClearDebugPanel() |  | ||||||
|   await page.getByRole('button', { name: 'Exit Sketch' }).click() |  | ||||||
|  |  | ||||||
|   await u.expectCmdLog('[data-message-type="execution-done"]') |  | ||||||
|  |  | ||||||
|   await u.updateCamPosition([0, 100, 100]) |  | ||||||
|  |  | ||||||
|   // start a new sketch |  | ||||||
|   await u.clearCommandLogs() |  | ||||||
|   await page.getByRole('button', { name: 'Start Sketch' }).click() |  | ||||||
|   await page.waitForTimeout(100) |  | ||||||
|   await page.mouse.click(673, 384) |  | ||||||
|  |  | ||||||
|   await page.waitForTimeout(500) // TODO detect animation ending, or disable animation |  | ||||||
|   await u.clearAndCloseDebugPanel() |  | ||||||
|  |  | ||||||
|   await page.mouse.click(startXPx + PUR * 10, 500 - PUR * 10) |  | ||||||
|   const startAt2 = '[23.61, -31.85]' |  | ||||||
|   await expect( |  | ||||||
|     (await page.locator('.cm-content').innerText()).replace(/\s/g, '') |  | ||||||
|   ).toBe( |  | ||||||
|     `${finalCodeFirstSketch} |  | ||||||
| const part002 = startSketchOn('XY') |  | ||||||
|   |> startProfileAt(${startAt2}, %)`.replace(/\s/g, '') |  | ||||||
|   ) |  | ||||||
|   await page.waitForTimeout(100) |  | ||||||
|  |  | ||||||
|   await u.closeDebugPanel() |  | ||||||
|   await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 10) |  | ||||||
|   await page.waitForTimeout(100) |  | ||||||
|  |  | ||||||
|   const num2 = 23.83 |  | ||||||
|   await expect( |  | ||||||
|     (await page.locator('.cm-content').innerText()).replace(/\s/g, '') |  | ||||||
|   ).toBe( |  | ||||||
|     `${finalCodeFirstSketch} |  | ||||||
| const part002 = startSketchOn('XY') |  | ||||||
|   |> startProfileAt(${startAt2}, %) |  | ||||||
|   |> line([${num2}, 0], %)`.replace(/\s/g, '') |  | ||||||
|   ) |  | ||||||
|  |  | ||||||
|   await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 20) |  | ||||||
|   await expect( |  | ||||||
|     (await page.locator('.cm-content').innerText()).replace(/\s/g, '') |  | ||||||
|   ).toBe( |  | ||||||
|     `${finalCodeFirstSketch} |  | ||||||
| const part002 = startSketchOn('XY') |  | ||||||
|   |> startProfileAt(${startAt2}, %) |  | ||||||
|   |> line([${num2}, 0], %) |  | ||||||
|   |> line([0, ${num2}], %)`.replace(/\s/g, '') |  | ||||||
|   ) |  | ||||||
|   await page.mouse.click(startXPx, 500 - PUR * 20) |  | ||||||
|   await expect( |  | ||||||
|     (await page.locator('.cm-content').innerText()).replace(/\s/g, '') |  | ||||||
|   ).toBe( |  | ||||||
|     `${finalCodeFirstSketch} |  | ||||||
| const part002 = startSketchOn('XY') |  | ||||||
|   |> startProfileAt(${startAt2}, %) |  | ||||||
|   |> line([${num2}, 0], %) |  | ||||||
|   |> line([0, ${num2}], %) |  | ||||||
|   |> line([-47.44, 0], %)`.replace(/\s/g, '') |  | ||||||
|   ) |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| test('ProgramMemory can be serialised', async ({ page, context }) => { |  | ||||||
|   const u = getUtils(page) |  | ||||||
|   await context.addInitScript(async () => { |  | ||||||
|     localStorage.setItem( |  | ||||||
|       'persistCode', |  | ||||||
|       `const part = startSketchOn('XY') |  | ||||||
|   |> startProfileAt([0, 0], %) |  | ||||||
|   |> line([0, 1], %) |  | ||||||
|   |> line([1, 0], %) |  | ||||||
|   |> line([0, -1], %) |  | ||||||
|   |> close(%) |  | ||||||
|   |> extrude(1, %) |  | ||||||
|   |> patternLinear({ |  | ||||||
|         axis: [1, 0, 1], |  | ||||||
|         repetitions: 3, |  | ||||||
|         distance: 6 |  | ||||||
|       }, %)` |  | ||||||
|     ) |  | ||||||
|   }) |  | ||||||
|   await page.setViewportSize({ width: 1000, height: 500 }) |  | ||||||
|   await page.goto('/') |  | ||||||
|   const messages: string[] = [] |  | ||||||
|  |  | ||||||
|   // Listen for all console events and push the message text to an array |  | ||||||
|   page.on('console', (message) => messages.push(message.text())) |  | ||||||
|   await u.waitForAuthSkipAppStart() |  | ||||||
|  |  | ||||||
|   // wait for execution done |  | ||||||
|   await u.openDebugPanel() |  | ||||||
|   await u.expectCmdLog('[data-message-type="execution-done"]') |  | ||||||
|  |  | ||||||
|   const forbiddenMessages = ['cannot serialize tagged newtype variant'] |  | ||||||
|   forbiddenMessages.forEach((forbiddenMessage) => { |  | ||||||
|     messages.forEach((message) => { |  | ||||||
|       expect(message).not.toContain(forbiddenMessage) |  | ||||||
|     }) |  | ||||||
|   }) |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| test('edit selections', async ({ page, context }) => { |  | ||||||
|   const u = getUtils(page) |  | ||||||
|   const selectionsSnippets = { |  | ||||||
|     extrudeAndEditBlocked: '|> startProfileAt([10.81, 32.99], %)', |  | ||||||
|     extrudeAndEditBlockedInFunction: '|> startProfileAt(pos, %)', |  | ||||||
|     extrudeAndEditAllowed: '|> startProfileAt([15.72, 4.7], %)', |  | ||||||
|     editOnly: '|> startProfileAt([15.79, -14.6], %)', |  | ||||||
|   } |  | ||||||
|   await context.addInitScript( |  | ||||||
|     async ({ |  | ||||||
|       extrudeAndEditBlocked, |  | ||||||
|       extrudeAndEditBlockedInFunction, |  | ||||||
|       extrudeAndEditAllowed, |  | ||||||
|       editOnly, |  | ||||||
|     }: any) => { |  | ||||||
|       localStorage.setItem( |  | ||||||
|         'persistCode', |  | ||||||
|         `const part001 = startSketchOn('-XZ') |  | ||||||
|   ${extrudeAndEditBlocked} |  | ||||||
|   |> line([25.96, 2.93], %) |  | ||||||
|   |> line([5.25, -5.72], %) |  | ||||||
|   |> line([-2.01, -10.35], %) |  | ||||||
|   |> line([-27.65, -2.78], %) |  | ||||||
|   |> close(%) |  | ||||||
|   |> extrude(5, %) |  | ||||||
| const part002 = startSketchOn('-XZ') |  | ||||||
|   ${extrudeAndEditAllowed} |  | ||||||
|   |> line([10.32, 6.47], %) |  | ||||||
|   |> line([9.71, -6.16], %) |  | ||||||
|   |> line([-3.08, -9.86], %) |  | ||||||
|   |> line([-12.02, -1.54], %) |  | ||||||
|   |> close(%) |  | ||||||
| const part003 = startSketchOn('-XZ') |  | ||||||
|   ${editOnly} |  | ||||||
|   |> line([27.55, -1.65], %) |  | ||||||
|   |> line([4.95, -8], %) |  | ||||||
|   |> line([-20.38, -10.12], %) |  | ||||||
|   |> line([-15.79, 17.08], %) |  | ||||||
|  |  | ||||||
| fn yohey = (pos) => { |  | ||||||
|   const part004 = startSketchOn('-XZ') |  | ||||||
|   ${extrudeAndEditBlockedInFunction} |  | ||||||
|   |> line([27.55, -1.65], %) |  | ||||||
|   |> line([4.95, -10.53], %) |  | ||||||
|   |> line([-20.38, -8], %) |  | ||||||
|   |> line([-15.79, 17.08], %) |  | ||||||
|   return '' |  | ||||||
| } |  | ||||||
|      |  | ||||||
|     yohey([15.79, -34.6]) |  | ||||||
| ` |  | ||||||
|       ) |  | ||||||
|     }, |  | ||||||
|     selectionsSnippets |  | ||||||
|   ) |  | ||||||
|   await page.setViewportSize({ width: 1200, height: 500 }) |  | ||||||
|   await page.goto('/') |  | ||||||
|   await u.waitForAuthSkipAppStart() |  | ||||||
|  |  | ||||||
|   // wait for execution done |  | ||||||
|   await u.openDebugPanel() |  | ||||||
|   await u.expectCmdLog('[data-message-type="execution-done"]') |  | ||||||
|   await u.closeDebugPanel() |  | ||||||
|  |  | ||||||
|   await page.getByText(selectionsSnippets.extrudeAndEditBlocked).click() |  | ||||||
|   await expect(page.getByRole('button', { name: 'Extrude' })).toBeDisabled() |  | ||||||
|   await expect( |  | ||||||
|     page.getByRole('button', { name: 'Edit Sketch' }) |  | ||||||
|   ).not.toBeVisible() |  | ||||||
|  |  | ||||||
|   await page.getByText(selectionsSnippets.extrudeAndEditAllowed).click() |  | ||||||
|   await expect(page.getByRole('button', { name: 'Extrude' })).not.toBeDisabled() |  | ||||||
|   await expect( |  | ||||||
|     page.getByRole('button', { name: 'Edit Sketch' }) |  | ||||||
|   ).not.toBeDisabled() |  | ||||||
|  |  | ||||||
|   await page.getByText(selectionsSnippets.editOnly).click() |  | ||||||
|   await expect(page.getByRole('button', { name: 'Extrude' })).toBeDisabled() |  | ||||||
|   await expect( |  | ||||||
|     page.getByRole('button', { name: 'Edit Sketch' }) |  | ||||||
|   ).not.toBeDisabled() |  | ||||||
|  |  | ||||||
|   await page |  | ||||||
|     .getByText(selectionsSnippets.extrudeAndEditBlockedInFunction) |  | ||||||
|     .click() |  | ||||||
|   await expect(page.getByRole('button', { name: 'Extrude' })).toBeDisabled() |  | ||||||
|   await expect( |  | ||||||
|     page.getByRole('button', { name: 'Edit Sketch' }) |  | ||||||
|   ).not.toBeVisible() |  | ||||||
|  |  | ||||||
|   // selecting an editable sketch but clicking "start sktech" should start a new sketch and not edit the existing one |  | ||||||
|   await page.getByText(selectionsSnippets.extrudeAndEditAllowed).click() |  | ||||||
|   await page.getByRole('button', { name: 'Start Sketch' }).click() |  | ||||||
|   await page.mouse.click(700, 200) |  | ||||||
|   // expect main content to contain `part005` i.e. started a new sketch |  | ||||||
|   await expect(page.locator('.cm-content')).toHaveText( |  | ||||||
|     /part005 = startSketchOn\('-XZ'\)/ |  | ||||||
|   ) |  | ||||||
| }) |  | ||||||
|  | |||||||
| @ -14,7 +14,6 @@ try { | |||||||
| } catch (err) { | } catch (err) { | ||||||
|   // probably running in CI |   // probably running in CI | ||||||
|   secrets.token = process.env.token || '' |   secrets.token = process.env.token || '' | ||||||
|   secrets.snapshottoken = process.env.snapshottoken || '' |  | ||||||
|   // add more env vars here to make them available in CI |   // add more env vars here to make them available in CI | ||||||
| } | } | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,10 +1,10 @@ | |||||||
| import { test, expect } from '@playwright/test' | import { test, expect } from '@playwright/test' | ||||||
| import { secrets } from './secrets' | import { secrets } from './secrets' | ||||||
|  | import { EngineCommand } from '../../src/lang/std/engineConnection' | ||||||
|  | import { v4 as uuidv4 } from 'uuid' | ||||||
| import { getUtils } from './test-utils' | import { getUtils } from './test-utils' | ||||||
| import { Models } from '@kittycad/lib' | import { Models } from '@kittycad/lib' | ||||||
| import fsp from 'fs/promises' | import fsp from 'fs/promises' | ||||||
| import { spawn } from 'child_process' |  | ||||||
| import { APP_NAME } from 'lib/constants' |  | ||||||
|  |  | ||||||
| test.beforeEach(async ({ context, page }) => { | test.beforeEach(async ({ context, page }) => { | ||||||
|   await context.addInitScript(async (token) => { |   await context.addInitScript(async (token) => { | ||||||
| @ -38,8 +38,19 @@ test('change camera, show planes', async ({ page, context }) => { | |||||||
|   await u.waitForAuthSkipAppStart() |   await u.waitForAuthSkipAppStart() | ||||||
|   await u.openAndClearDebugPanel() |   await u.openAndClearDebugPanel() | ||||||
|  |  | ||||||
|   const camPos: [number, number, number] = [0, 85, 85] |   const camCmd: EngineCommand = { | ||||||
|   await u.updateCamPosition(camPos) |     type: 'modeling_cmd_req', | ||||||
|  |     cmd_id: uuidv4(), | ||||||
|  |     cmd: { | ||||||
|  |       type: 'default_camera_look_at', | ||||||
|  |       center: { x: 0, y: 0, z: 0 }, | ||||||
|  |       up: { x: 0, y: 0, z: 1 }, | ||||||
|  |       vantage: { x: 0, y: 85, z: 85 }, | ||||||
|  |     }, | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   await u.sendCustomCmd(camCmd) | ||||||
|  |   await u.waitForCmdReceive('default_camera_look_at') | ||||||
|  |  | ||||||
|   // rotate |   // rotate | ||||||
|   await u.closeDebugPanel() |   await u.closeDebugPanel() | ||||||
| @ -49,11 +60,13 @@ test('change camera, show planes', async ({ page, context }) => { | |||||||
|   await page.mouse.up({ button: 'right' }) |   await page.mouse.up({ button: 'right' }) | ||||||
|  |  | ||||||
|   await u.openDebugPanel() |   await u.openDebugPanel() | ||||||
|  |   await u.waitForCmdReceive('camera_drag_end') | ||||||
|   await page.waitForTimeout(500) |   await page.waitForTimeout(500) | ||||||
|   await u.clearCommandLogs() |   await u.clearCommandLogs() | ||||||
|  |  | ||||||
|   await page.getByRole('button', { name: 'Start Sketch' }).click() |   await page.getByRole('button', { name: 'Start Sketch' }).click() | ||||||
|  |  | ||||||
|  |   await u.waitForDefaultPlanesVisibilityChange() | ||||||
|   await u.closeDebugPanel() |   await u.closeDebugPanel() | ||||||
|  |  | ||||||
|   await expect(page).toHaveScreenshot({ |   await expect(page).toHaveScreenshot({ | ||||||
| @ -62,8 +75,10 @@ test('change camera, show planes', async ({ page, context }) => { | |||||||
|  |  | ||||||
|   await u.openAndClearDebugPanel() |   await u.openAndClearDebugPanel() | ||||||
|   await page.getByRole('button', { name: 'Exit Sketch' }).click() |   await page.getByRole('button', { name: 'Exit Sketch' }).click() | ||||||
|  |   await u.waitForDefaultPlanesVisibilityChange() | ||||||
|  |  | ||||||
|   await u.updateCamPosition(camPos) |   await u.sendCustomCmd(camCmd) | ||||||
|  |   await u.waitForCmdReceive('default_camera_look_at') | ||||||
|  |  | ||||||
|   await u.clearCommandLogs() |   await u.clearCommandLogs() | ||||||
|   await u.closeDebugPanel() |   await u.closeDebugPanel() | ||||||
| @ -76,10 +91,12 @@ test('change camera, show planes', async ({ page, context }) => { | |||||||
|   await page.keyboard.up('Shift') |   await page.keyboard.up('Shift') | ||||||
|  |  | ||||||
|   await u.openDebugPanel() |   await u.openDebugPanel() | ||||||
|  |   await u.waitForCmdReceive('camera_drag_end') | ||||||
|   await page.waitForTimeout(300) |   await page.waitForTimeout(300) | ||||||
|   await u.clearCommandLogs() |   await u.clearCommandLogs() | ||||||
|  |  | ||||||
|   await page.getByRole('button', { name: 'Start Sketch' }).click() |   await page.getByRole('button', { name: 'Start Sketch' }).click() | ||||||
|  |   await u.waitForDefaultPlanesVisibilityChange() | ||||||
|   await u.closeDebugPanel() |   await u.closeDebugPanel() | ||||||
|  |  | ||||||
|   await expect(page).toHaveScreenshot({ |   await expect(page).toHaveScreenshot({ | ||||||
| @ -88,8 +105,10 @@ test('change camera, show planes', async ({ page, context }) => { | |||||||
|  |  | ||||||
|   await u.openAndClearDebugPanel() |   await u.openAndClearDebugPanel() | ||||||
|   await page.getByRole('button', { name: 'Exit Sketch' }).click() |   await page.getByRole('button', { name: 'Exit Sketch' }).click() | ||||||
|  |   await u.waitForDefaultPlanesVisibilityChange() | ||||||
|  |  | ||||||
|   await u.updateCamPosition(camPos) |   await u.sendCustomCmd(camCmd) | ||||||
|  |   await u.waitForCmdReceive('default_camera_look_at') | ||||||
|  |  | ||||||
|   await u.clearCommandLogs() |   await u.clearCommandLogs() | ||||||
|   await u.closeDebugPanel() |   await u.closeDebugPanel() | ||||||
| @ -98,15 +117,17 @@ test('change camera, show planes', async ({ page, context }) => { | |||||||
|   await page.keyboard.down('Control') |   await page.keyboard.down('Control') | ||||||
|   await page.mouse.move(700, 400) |   await page.mouse.move(700, 400) | ||||||
|   await page.mouse.down({ button: 'right' }) |   await page.mouse.down({ button: 'right' }) | ||||||
|   await page.mouse.move(700, 300) |   await page.mouse.move(700, 350) | ||||||
|   await page.mouse.up({ button: 'right' }) |   await page.mouse.up({ button: 'right' }) | ||||||
|   await page.keyboard.up('Control') |   await page.keyboard.up('Control') | ||||||
|  |  | ||||||
|   await u.openDebugPanel() |   await u.openDebugPanel() | ||||||
|  |   await u.waitForCmdReceive('camera_drag_end') | ||||||
|   await page.waitForTimeout(300) |   await page.waitForTimeout(300) | ||||||
|   await u.clearCommandLogs() |   await u.clearCommandLogs() | ||||||
|  |  | ||||||
|   await page.getByRole('button', { name: 'Start Sketch' }).click() |   await page.getByRole('button', { name: 'Start Sketch' }).click() | ||||||
|  |   await u.waitForDefaultPlanesVisibilityChange() | ||||||
|   await u.closeDebugPanel() |   await u.closeDebugPanel() | ||||||
|  |  | ||||||
|   await expect(page).toHaveScreenshot({ |   await expect(page).toHaveScreenshot({ | ||||||
| @ -116,7 +137,6 @@ test('change camera, show planes', async ({ page, context }) => { | |||||||
|  |  | ||||||
| test('exports of each format should work', async ({ page, context }) => { | test('exports of each format should work', async ({ page, context }) => { | ||||||
|   // FYI this test doesn't work with only engine running locally |   // FYI this test doesn't work with only engine running locally | ||||||
|   // And you will need to have the KittyCAD CLI installed |  | ||||||
|   const u = getUtils(page) |   const u = getUtils(page) | ||||||
|   await context.addInitScript(async () => { |   await context.addInitScript(async () => { | ||||||
|     ;(window as any).playwrightSkipFilePicker = true |     ;(window as any).playwrightSkipFilePicker = true | ||||||
| @ -141,11 +161,11 @@ const part001 = startSketchOn('-XZ') | |||||||
|   |> xLineTo({ to: totalLen, tag: 'seg03' }, %) |   |> xLineTo({ to: totalLen, tag: 'seg03' }, %) | ||||||
|   |> yLine({ length: -armThick, tag: 'seg01' }, %) |   |> yLine({ length: -armThick, tag: 'seg01' }, %) | ||||||
|   |> angledLineThatIntersects({ |   |> angledLineThatIntersects({ | ||||||
|         angle: HALF_TURN, |         angle: _180, | ||||||
|         offset: -armThick, |         offset: -armThick, | ||||||
|         intersectTag: 'seg04' |         intersectTag: 'seg04' | ||||||
|       }, %) |       }, %) | ||||||
|   |> angledLineToY([segAng('seg04', %) + 180, ZERO], %) |   |> angledLineToY([segAng('seg04', %) + 180, _0], %) | ||||||
|   |> angledLineToY({ |   |> angledLineToY({ | ||||||
|         angle: -bottomAng, |         angle: -bottomAng, | ||||||
|         to: -totalHeightHalf - armThick, |         to: -totalHeightHalf - armThick, | ||||||
| @ -154,12 +174,12 @@ const part001 = startSketchOn('-XZ') | |||||||
|   |> xLineTo(segEndX('seg03', %) + 0, %) |   |> xLineTo(segEndX('seg03', %) + 0, %) | ||||||
|   |> yLine(-segLen('seg01', %), %) |   |> yLine(-segLen('seg01', %), %) | ||||||
|   |> angledLineThatIntersects({ |   |> angledLineThatIntersects({ | ||||||
|         angle: HALF_TURN, |         angle: _180, | ||||||
|         offset: -armThick, |         offset: -armThick, | ||||||
|         intersectTag: 'seg02' |         intersectTag: 'seg02' | ||||||
|       }, %) |       }, %) | ||||||
|   |> angledLineToY([segAng('seg02', %) + 180, -baseHeight], %) |   |> angledLineToY([segAng('seg02', %) + 180, -baseHeight], %) | ||||||
|   |> xLineTo(ZERO, %) |   |> xLineTo(_0, %) | ||||||
|   |> close(%)  |   |> close(%)  | ||||||
|   |> extrude(4, %)` |   |> extrude(4, %)` | ||||||
|     ) |     ) | ||||||
| @ -168,21 +188,15 @@ const part001 = startSketchOn('-XZ') | |||||||
|   await page.goto('/') |   await page.goto('/') | ||||||
|   await u.waitForAuthSkipAppStart() |   await u.waitForAuthSkipAppStart() | ||||||
|   await u.openDebugPanel() |   await u.openDebugPanel() | ||||||
|  |   await u.waitForDefaultPlanesVisibilityChange() | ||||||
|   await u.expectCmdLog('[data-message-type="execution-done"]') |   await u.expectCmdLog('[data-message-type="execution-done"]') | ||||||
|   await u.waitForCmdReceive('extrude') |   await u.waitForCmdReceive('extrude') | ||||||
|   await page.waitForTimeout(1000) |   await page.waitForTimeout(1000) | ||||||
|   await u.clearAndCloseDebugPanel() |   await u.clearAndCloseDebugPanel() | ||||||
|  |  | ||||||
|   await page.getByRole('button', { name: APP_NAME }).click() |   await page.getByRole('button', { name: 'KittyCAD Modeling App' }).click() | ||||||
|  |  | ||||||
|   interface Paths { |   const doExport = async (output: Models['OutputFormat_type']) => { | ||||||
|     modelPath: string |  | ||||||
|     imagePath: string |  | ||||||
|     outputType: string |  | ||||||
|   } |  | ||||||
|   const doExport = async ( |  | ||||||
|     output: Models['OutputFormat_type'] |  | ||||||
|   ): Promise<Paths> => { |  | ||||||
|     await page.getByRole('button', { name: 'Export Model' }).click() |     await page.getByRole('button', { name: 'Export Model' }).click() | ||||||
|  |  | ||||||
|     const exportSelect = page.getByTestId('export-type') |     const exportSelect = page.getByTestId('export-type') | ||||||
| @ -196,10 +210,10 @@ const part001 = startSketchOn('-XZ') | |||||||
|     const downloadPromise = page.waitForEvent('download') |     const downloadPromise = page.waitForEvent('download') | ||||||
|     await page.getByRole('button', { name: 'Export', exact: true }).click() |     await page.getByRole('button', { name: 'Export', exact: true }).click() | ||||||
|     const download = await downloadPromise |     const download = await downloadPromise | ||||||
|     const downloadLocationer = (extra = '', isImage = false) => |     const downloadLocationer = (extra = '') => | ||||||
|       `./e2e/playwright/export-snapshots/${output.type}-${ |       `./e2e/playwright/export-snapshots/${output.type}-${ | ||||||
|         'storage' in output ? output.storage : '' |         'storage' in output ? output.storage : '' | ||||||
|       }${extra}.${isImage ? 'png' : output.type}` |       }${extra}.${output.type}` | ||||||
|     const downloadLocation = downloadLocationer() |     const downloadLocation = downloadLocationer() | ||||||
|     const downloadLocation2 = downloadLocationer('-2') |     const downloadLocation2 = downloadLocationer('-2') | ||||||
|  |  | ||||||
| @ -235,11 +249,6 @@ const part001 = startSketchOn('-XZ') | |||||||
|       ) |       ) | ||||||
|       await fsp.writeFile(downloadLocation, newFileContents) |       await fsp.writeFile(downloadLocation, newFileContents) | ||||||
|     } |     } | ||||||
|     return { |  | ||||||
|       modelPath: downloadLocation, |  | ||||||
|       imagePath: downloadLocationer('', true), |  | ||||||
|       outputType: output.type, |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
|   const axisDirectionPair: Models['AxisDirectionPair_type'] = { |   const axisDirectionPair: Models['AxisDirectionPair_type'] = { | ||||||
|     axis: 'z', |     axis: 'z', | ||||||
| @ -249,231 +258,67 @@ const part001 = startSketchOn('-XZ') | |||||||
|     forward: axisDirectionPair, |     forward: axisDirectionPair, | ||||||
|     up: axisDirectionPair, |     up: axisDirectionPair, | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   const exportLocations: Paths[] = [] |  | ||||||
|  |  | ||||||
|   // NOTE it was easiest to leverage existing types and have doExport take Models['OutputFormat_type'] as in input |   // NOTE it was easiest to leverage existing types and have doExport take Models['OutputFormat_type'] as in input | ||||||
|   // just note that only `type` and `storage` are used for selecting the drop downs is the app |   // just note that only `type` and `storage` are used for selecting the drop downs is the app | ||||||
|   // the rest are only there to make typescript happy |   // the rest are only there to make typescript happy | ||||||
|   exportLocations.push( |   await doExport({ | ||||||
|     await doExport({ |     type: 'step', | ||||||
|       type: 'step', |     coords: sysType, | ||||||
|       coords: sysType, |   }) | ||||||
|     }) |   await doExport({ | ||||||
|   ) |     type: 'gltf', | ||||||
|   exportLocations.push( |     storage: 'embedded', | ||||||
|     await doExport({ |     presentation: 'pretty', | ||||||
|       type: 'ply', |   }) | ||||||
|       coords: sysType, |   await doExport({ | ||||||
|       selection: { type: 'default_scene' }, |     type: 'gltf', | ||||||
|       storage: 'ascii', |     storage: 'binary', | ||||||
|       units: 'in', |     presentation: 'pretty', | ||||||
|     }) |   }) | ||||||
|   ) |  | ||||||
|   exportLocations.push( |  | ||||||
|     await doExport({ |  | ||||||
|       type: 'ply', |  | ||||||
|       storage: 'binary_little_endian', |  | ||||||
|       coords: sysType, |  | ||||||
|       selection: { type: 'default_scene' }, |  | ||||||
|       units: 'in', |  | ||||||
|     }) |  | ||||||
|   ) |  | ||||||
|   exportLocations.push( |  | ||||||
|     await doExport({ |  | ||||||
|       type: 'ply', |  | ||||||
|       storage: 'binary_big_endian', |  | ||||||
|       coords: sysType, |  | ||||||
|       selection: { type: 'default_scene' }, |  | ||||||
|       units: 'in', |  | ||||||
|     }) |  | ||||||
|   ) |  | ||||||
|   exportLocations.push( |  | ||||||
|     await doExport({ |  | ||||||
|       type: 'stl', |  | ||||||
|       storage: 'ascii', |  | ||||||
|       coords: sysType, |  | ||||||
|       units: 'in', |  | ||||||
|       selection: { type: 'default_scene' }, |  | ||||||
|     }) |  | ||||||
|   ) |  | ||||||
|   exportLocations.push( |  | ||||||
|     await doExport({ |  | ||||||
|       type: 'stl', |  | ||||||
|       storage: 'binary', |  | ||||||
|       coords: sysType, |  | ||||||
|       units: 'in', |  | ||||||
|       selection: { type: 'default_scene' }, |  | ||||||
|     }) |  | ||||||
|   ) |  | ||||||
|   exportLocations.push( |  | ||||||
|     await doExport({ |  | ||||||
|       // obj seems to be a little flaky, times out tests sometimes |  | ||||||
|       type: 'obj', |  | ||||||
|       coords: sysType, |  | ||||||
|       units: 'in', |  | ||||||
|     }) |  | ||||||
|   ) |  | ||||||
|   exportLocations.push( |  | ||||||
|     await doExport({ |  | ||||||
|       type: 'gltf', |  | ||||||
|       storage: 'embedded', |  | ||||||
|       presentation: 'pretty', |  | ||||||
|     }) |  | ||||||
|   ) |  | ||||||
|   exportLocations.push( |  | ||||||
|     await doExport({ |  | ||||||
|       type: 'gltf', |  | ||||||
|       storage: 'binary', |  | ||||||
|       presentation: 'pretty', |  | ||||||
|     }) |  | ||||||
|   ) |  | ||||||
|  |  | ||||||
|   // TODO: gltfs don't seem to work with snap shots. push onto exportLocations once it's figured out |  | ||||||
|   await doExport({ |   await doExport({ | ||||||
|     type: 'gltf', |     type: 'gltf', | ||||||
|     storage: 'standard', |     storage: 'standard', | ||||||
|     presentation: 'pretty', |     presentation: 'pretty', | ||||||
|   }) |   }) | ||||||
|  |   await doExport({ | ||||||
|   // close page to disconnect websocket since we can only have one open atm |     type: 'ply', | ||||||
|   await page.close() |     coords: sysType, | ||||||
|  |     selection: { type: 'default_scene' }, | ||||||
|   // snapshot exports, good compromise to capture that exports are healthy without getting bogged down in "did the formatting change" changes |     storage: 'ascii', | ||||||
|   // context: https://github.com/KittyCAD/modeling-app/issues/1222 |     units: 'in', | ||||||
|   for (const { modelPath, imagePath, outputType } of exportLocations) { |  | ||||||
|     const cliCommand = `export KITTYCAD_TOKEN=${secrets.snapshottoken} && kittycad file snapshot --output-format=png --src-format=${outputType} ${modelPath} ${imagePath}` |  | ||||||
|     const child = spawn(cliCommand, { shell: true }) |  | ||||||
|     await new Promise((resolve, reject) => { |  | ||||||
|       child.on('error', (code: any, msg: any) => { |  | ||||||
|         console.log('error', code, msg) |  | ||||||
|         reject() |  | ||||||
|       }) |  | ||||||
|       child.on('exit', (code, msg) => { |  | ||||||
|         console.log('exit', code, msg) |  | ||||||
|         if (code !== 0) { |  | ||||||
|           reject(`exit code ${code} for model ${modelPath}`) |  | ||||||
|         } else { |  | ||||||
|           resolve(true) |  | ||||||
|         } |  | ||||||
|       }) |  | ||||||
|       child.stderr.on('data', (data) => console.log(`stderr: ${data}`)) |  | ||||||
|       child.stdout.on('data', (data) => console.log(`stdout: ${data}`)) |  | ||||||
|     }) |  | ||||||
|   } |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| test('extrude on each default plane should be stable', async ({ |  | ||||||
|   page, |  | ||||||
|   context, |  | ||||||
| }) => { |  | ||||||
|   const u = getUtils(page) |  | ||||||
|   const makeCode = (plane = 'XY') => `const part001 = startSketchOn('${plane}') |  | ||||||
|   |> startProfileAt([14.06, 8.88], %) |  | ||||||
|   |> line([12.98, -0.15], %) |  | ||||||
|   |> line([5.56, 9.89], %) |  | ||||||
|   |> line([-11.28, 8.96], %) |  | ||||||
|   |> line([-10.81, -7.57], %) |  | ||||||
|   |> close(%) |  | ||||||
|   |> extrude(20, %) |  | ||||||
| ` |  | ||||||
|   await context.addInitScript(async (code) => { |  | ||||||
|     localStorage.setItem('persistCode', code) |  | ||||||
|   }, makeCode('XY')) |  | ||||||
|   await page.setViewportSize({ width: 1200, height: 500 }) |  | ||||||
|   await page.goto('/') |  | ||||||
|   await u.waitForAuthSkipAppStart() |  | ||||||
|  |  | ||||||
|   // wait for execution done |  | ||||||
|   await u.openDebugPanel() |  | ||||||
|   await u.expectCmdLog('[data-message-type="execution-done"]') |  | ||||||
|   await u.clearAndCloseDebugPanel() |  | ||||||
|  |  | ||||||
|   await page.getByText('Code').click() |  | ||||||
|   await expect(page).toHaveScreenshot({ |  | ||||||
|     maxDiffPixels: 100, |  | ||||||
|   }) |   }) | ||||||
|   await page.getByText('Code').click() |   await doExport({ | ||||||
|  |     type: 'ply', | ||||||
|   const runSnapshotsForOtherPlanes = async (plane = 'XY') => { |     storage: 'binary_little_endian', | ||||||
|     // clear code |     coords: sysType, | ||||||
|     await u.removeCurrentCode() |     selection: { type: 'default_scene' }, | ||||||
|     // add makeCode('XZ') |     units: 'in', | ||||||
|     await page.locator('.cm-content').fill(makeCode(plane)) |  | ||||||
|     // wait for execution done |  | ||||||
|     await u.openDebugPanel() |  | ||||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') |  | ||||||
|     await u.clearAndCloseDebugPanel() |  | ||||||
|  |  | ||||||
|     await page.getByText('Code').click() |  | ||||||
|     await expect(page).toHaveScreenshot({ |  | ||||||
|       maxDiffPixels: 100, |  | ||||||
|     }) |  | ||||||
|     await page.getByText('Code').click() |  | ||||||
|   } |  | ||||||
|   await runSnapshotsForOtherPlanes('-XY') |  | ||||||
|  |  | ||||||
|   await runSnapshotsForOtherPlanes('XZ') |  | ||||||
|   await runSnapshotsForOtherPlanes('-XZ') |  | ||||||
|  |  | ||||||
|   await runSnapshotsForOtherPlanes('YZ') |  | ||||||
|   await runSnapshotsForOtherPlanes('-YZ') |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| test('Draft segments should look right', async ({ page }) => { |  | ||||||
|   const u = getUtils(page) |  | ||||||
|   await page.setViewportSize({ width: 1200, height: 500 }) |  | ||||||
|   const PUR = 400 / 37.5 //pixeltoUnitRatio |  | ||||||
|   await page.goto('/') |  | ||||||
|   await u.waitForAuthSkipAppStart() |  | ||||||
|   await u.openDebugPanel() |  | ||||||
|  |  | ||||||
|   await expect(page.getByRole('button', { name: 'Start Sketch' })).toBeVisible() |  | ||||||
|  |  | ||||||
|   // click on "Start Sketch" button |  | ||||||
|   await u.clearCommandLogs() |  | ||||||
|   await u.doAndWaitForImageDiff( |  | ||||||
|     () => page.getByRole('button', { name: 'Start Sketch' }).click(), |  | ||||||
|     200 |  | ||||||
|   ) |  | ||||||
|  |  | ||||||
|   // select a plane |  | ||||||
|   await page.mouse.click(700, 200) |  | ||||||
|  |  | ||||||
|   await expect(page.locator('.cm-content')).toHaveText( |  | ||||||
|     `const part001 = startSketchOn('-XZ')` |  | ||||||
|   ) |  | ||||||
|  |  | ||||||
|   await page.waitForTimeout(300) // TODO detect animation ending, or disable animation |  | ||||||
|  |  | ||||||
|   const startXPx = 600 |  | ||||||
|   await page.mouse.click(startXPx + PUR * 10, 500 - PUR * 10) |  | ||||||
|   const startAt = '[23.89, -32.23]' |  | ||||||
|   await expect(page.locator('.cm-content')) |  | ||||||
|     .toHaveText(`const part001 = startSketchOn('-XZ') |  | ||||||
|   |> startProfileAt(${startAt}, %)`) |  | ||||||
|   await page.waitForTimeout(100) |  | ||||||
|  |  | ||||||
|   await u.closeDebugPanel() |  | ||||||
|   await page.mouse.move(startXPx + PUR * 20, 500 - PUR * 10) |  | ||||||
|   await expect(page).toHaveScreenshot({ |  | ||||||
|     maxDiffPixels: 100, |  | ||||||
|   }) |   }) | ||||||
|  |   await doExport({ | ||||||
|   await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 10) |     type: 'ply', | ||||||
|   await page.waitForTimeout(100) |     storage: 'binary_big_endian', | ||||||
|  |     coords: sysType, | ||||||
|   const num = 24.11 |     selection: { type: 'default_scene' }, | ||||||
|   await expect(page.locator('.cm-content')) |     units: 'in', | ||||||
|     .toHaveText(`const part001 = startSketchOn('-XZ') |   }) | ||||||
|   |> startProfileAt(${startAt}, %) |   await doExport({ | ||||||
|   |> line([${num}, 0], %)`) |     type: 'stl', | ||||||
|  |     storage: 'ascii', | ||||||
|   await page.getByRole('button', { name: 'Tangential Arc' }).click() |     coords: sysType, | ||||||
|  |     units: 'in', | ||||||
|   await page.mouse.move(startXPx + PUR * 30, 500 - PUR * 20, { steps: 10 }) |     selection: { type: 'default_scene' }, | ||||||
|  |   }) | ||||||
|   await expect(page).toHaveScreenshot({ |   await doExport({ | ||||||
|     maxDiffPixels: 100, |     type: 'stl', | ||||||
|  |     storage: 'binary', | ||||||
|  |     coords: sysType, | ||||||
|  |     units: 'in', | ||||||
|  |     selection: { type: 'default_scene' }, | ||||||
|  |   }) | ||||||
|  |   await doExport({ | ||||||
|  |     // obj seems to be a little flaky, times out tests sometimes | ||||||
|  |     type: 'obj', | ||||||
|  |     coords: sysType, | ||||||
|  |     units: 'in', | ||||||
|   }) |   }) | ||||||
| }) | }) | ||||||
|  | |||||||
| Before Width: | Height: | Size: 40 KiB | 
| Before Width: | Height: | Size: 43 KiB | 
| Before Width: | Height: | Size: 95 KiB After Width: | Height: | Size: 113 KiB | 
| Before Width: | Height: | Size: 79 KiB After Width: | Height: | Size: 80 KiB | 
| Before Width: | Height: | Size: 111 KiB After Width: | Height: | Size: 54 KiB | 
| Before Width: | Height: | Size: 49 KiB | 
| Before Width: | Height: | Size: 49 KiB | 
| Before Width: | Height: | Size: 51 KiB | 
| Before Width: | Height: | Size: 52 KiB | 
| Before Width: | Height: | Size: 47 KiB | 
| Before Width: | Height: | Size: 46 KiB | 
| @ -80,21 +80,10 @@ export function getUtils(page: Page) { | |||||||
|     waitForAuthSkipAppStart: () => waitForPageLoad(page), |     waitForAuthSkipAppStart: () => waitForPageLoad(page), | ||||||
|     removeCurrentCode: () => removeCurrentCode(page), |     removeCurrentCode: () => removeCurrentCode(page), | ||||||
|     sendCustomCmd: (cmd: EngineCommand) => sendCustomCmd(page, cmd), |     sendCustomCmd: (cmd: EngineCommand) => sendCustomCmd(page, cmd), | ||||||
|     updateCamPosition: async (xyz: [number, number, number]) => { |  | ||||||
|       const fillInput = async () => { |  | ||||||
|         await page.fill('[data-testid="cam-x-position"]', String(xyz[0])) |  | ||||||
|         await page.fill('[data-testid="cam-y-position"]', String(xyz[1])) |  | ||||||
|         await page.fill('[data-testid="cam-z-position"]', String(xyz[2])) |  | ||||||
|       } |  | ||||||
|       await fillInput() |  | ||||||
|       await page.waitForTimeout(100) |  | ||||||
|       await fillInput() |  | ||||||
|       await page.waitForTimeout(100) |  | ||||||
|       await fillInput() |  | ||||||
|       await page.waitForTimeout(100) |  | ||||||
|     }, |  | ||||||
|     clearCommandLogs: () => clearCommandLogs(page), |     clearCommandLogs: () => clearCommandLogs(page), | ||||||
|     expectCmdLog: (locatorStr: string) => expectCmdLog(page, locatorStr), |     expectCmdLog: (locatorStr: string) => expectCmdLog(page, locatorStr), | ||||||
|  |     waitForDefaultPlanesVisibilityChange: () => | ||||||
|  |       waitForDefaultPlanesToBeVisible(page), | ||||||
|     openDebugPanel: () => openDebugPanel(page), |     openDebugPanel: () => openDebugPanel(page), | ||||||
|     closeDebugPanel: () => closeDebugPanel(page), |     closeDebugPanel: () => closeDebugPanel(page), | ||||||
|     openAndClearDebugPanel: async () => { |     openAndClearDebugPanel: async () => { | ||||||
|  | |||||||
| @ -1,26 +1,19 @@ | |||||||
| import { browser, $, expect } from '@wdio/globals' | import { browser, $, expect } from '@wdio/globals' | ||||||
| import fs from 'fs/promises' | import fs from 'fs/promises' | ||||||
|  |  | ||||||
| const defaultDir = `${process.env.HOME}/Documents/zoo-modeling-app-projects` | describe('KCMA (Tauri, Linux)', () => { | ||||||
| const userCodeDir = '/tmp/kittycad_user_code' |   it('opens the auth page, signs in, and signs out', async () => { | ||||||
|  |     // Clean up previous tests | ||||||
| async function click(element: WebdriverIO.Element): Promise<void> { |  | ||||||
|   // Workaround for .click(), see https://github.com/tauri-apps/tauri/issues/6541 |  | ||||||
|   await element.waitForClickable() |  | ||||||
|   await browser.execute('arguments[0].click();', element) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| describe('ZMA (Tauri, Linux)', () => { |  | ||||||
|   it('opens the auth page and signs in', async () => { |  | ||||||
|     // Clean up filesystem from previous tests |  | ||||||
|     await new Promise((resolve) => setTimeout(resolve, 100)) |     await new Promise((resolve) => setTimeout(resolve, 100)) | ||||||
|     await fs.rm(defaultDir, { force: true, recursive: true }) |     await fs.rm('/tmp/kittycad_user_code', { force: true }) | ||||||
|     await fs.rm(userCodeDir, { force: true }) |     await browser.execute('window.localStorage.clear()') | ||||||
|  |  | ||||||
|     const signInButton = await $('[data-testid="sign-in-button"]') |     const signInButton = await $('[data-testid="sign-in-button"]') | ||||||
|     expect(await signInButton.getText()).toEqual('Sign in') |     expect(await signInButton.getText()).toEqual('Sign in') | ||||||
|  |  | ||||||
|     await click(signInButton) |     // Workaround for .click(), see https://github.com/tauri-apps/tauri/issues/6541 | ||||||
|  |     await signInButton.waitForClickable() | ||||||
|  |     await browser.execute('arguments[0].click();', signInButton) | ||||||
|     await new Promise((resolve) => setTimeout(resolve, 2000)) |     await new Promise((resolve) => setTimeout(resolve, 2000)) | ||||||
|  |  | ||||||
|     // Get from main.rs |     // Get from main.rs | ||||||
| @ -56,49 +49,14 @@ describe('ZMA (Tauri, Linux)', () => { | |||||||
|     // Now should be signed in |     // Now should be signed in | ||||||
|     const newFileButton = await $('[data-testid="home-new-file"]') |     const newFileButton = await $('[data-testid="home-new-file"]') | ||||||
|     expect(await newFileButton.getText()).toEqual('New file') |     expect(await newFileButton.getText()).toEqual('New file') | ||||||
|   }) |  | ||||||
|  |  | ||||||
|   it('opens the settings page, checks filesystem settings, and closes the settings page', async () => { |     // So let's sign out! | ||||||
|     const menuButton = await $('[data-testid="user-sidebar-toggle"]') |     const menuButton = await $('[data-testid="user-sidebar-toggle"]') | ||||||
|     await click(menuButton) |     await menuButton.waitForClickable() | ||||||
|  |     await browser.execute('arguments[0].click();', menuButton) | ||||||
|     const settingsButton = await $('[data-testid="settings-button"]') |  | ||||||
|     await click(settingsButton) |  | ||||||
|  |  | ||||||
|     const defaultDirInput = await $('[data-testid="default-directory-input"]') |  | ||||||
|     expect(await defaultDirInput.getValue()).toEqual(defaultDir) |  | ||||||
|  |  | ||||||
|     const nameInput = await $('[data-testid="name-input"]') |  | ||||||
|     expect(await nameInput.getValue()).toEqual('project-$nnn') |  | ||||||
|  |  | ||||||
|     const closeButton = await $('[data-testid="close-button"]') |  | ||||||
|     await click(closeButton) |  | ||||||
|   }) |  | ||||||
|  |  | ||||||
|   it('checks that no file exists, creates a new file', async () => { |  | ||||||
|     const homeSection = await $('[data-testid="home-section"]') |  | ||||||
|     expect(await homeSection.getText()).toContain('No Projects found') |  | ||||||
|  |  | ||||||
|     const newFileButton = await $('[data-testid="home-new-file"]') |  | ||||||
|     await click(newFileButton) |  | ||||||
|     await new Promise((resolve) => setTimeout(resolve, 1000)) |  | ||||||
|  |  | ||||||
|     expect(await homeSection.getText()).toContain('project-000') |  | ||||||
|   }) |  | ||||||
|  |  | ||||||
|   it('opens the new file and expects a loading stream', async () => { |  | ||||||
|     const projectLink = await $('[data-testid="project-link"]') |  | ||||||
|     await click(projectLink) |  | ||||||
|     const loadingText = await $('[data-testid="loading-stream"]') |  | ||||||
|     expect(await loadingText.getText()).toContain('Loading stream...') |  | ||||||
|     await browser.execute('window.location.href = "tauri://localhost/home"') |  | ||||||
|   }) |  | ||||||
|  |  | ||||||
|   it('signs out', async () => { |  | ||||||
|     const menuButton = await $('[data-testid="user-sidebar-toggle"]') |  | ||||||
|     await click(menuButton) |  | ||||||
|     const signoutButton = await $('[data-testid="user-sidebar-sign-out"]') |     const signoutButton = await $('[data-testid="user-sidebar-sign-out"]') | ||||||
|     await click(signoutButton) |     await signoutButton.waitForClickable() | ||||||
|  |     await browser.execute('arguments[0].click();', signoutButton) | ||||||
|     const newSignInButton = await $('[data-testid="sign-in-button"]') |     const newSignInButton = await $('[data-testid="sign-in-button"]') | ||||||
|     expect(await newSignInButton.getText()).toEqual('Sign in') |     expect(await newSignInButton.getText()).toEqual('Sign in') | ||||||
|   }) |   }) | ||||||
|  | |||||||
							
								
								
									
										11
									
								
								index.html
									
									
									
									
									
								
							
							
						
						| @ -7,17 +7,12 @@ | |||||||
|     <meta name="theme-color" content="#000000" /> |     <meta name="theme-color" content="#000000" /> | ||||||
|     <meta |     <meta | ||||||
|       name="description" |       name="description" | ||||||
|       content="An open-source CAD modeling tool from the future by Zoo." |       content="An open-source CAD modeling tool from the future by KittyCAD." | ||||||
|     /> |     /> | ||||||
|     <link rel="apple-touch-icon" href="/logo192.png" /> |     <link rel="apple-touch-icon" href="/logo192.png" /> | ||||||
|     <link rel="manifest" href="/manifest.json" /> |     <link rel="manifest" href="/manifest.json" /> | ||||||
|     <link rel="stylesheet" href="https://use.typekit.net/zzv8rvm.css" /> |     <script defer data-domain="app.kittycad.io" src="https://plausible.corp.kittycad.io/js/script.js"></script> | ||||||
|     <script |     <title>KittyCAD Modeling App</title> | ||||||
|       defer |  | ||||||
|       data-domain="app.zoo.dev" |  | ||||||
|       src="https://plausible.corp.zoo.dev/js/script.js" |  | ||||||
|     ></script> |  | ||||||
|     <title>Zoo Modeling App</title> |  | ||||||
|   </head> |   </head> | ||||||
|   <body class="body-bg"> |   <body class="body-bg"> | ||||||
|     <noscript>You need to enable JavaScript to run this app.</noscript> |     <noscript>You need to enable JavaScript to run this app.</noscript> | ||||||
|  | |||||||
							
								
								
									
										18
									
								
								package.json
									
									
									
									
									
								
							
							
						
						| @ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "name": "untitled-app", |   "name": "untitled-app", | ||||||
|   "version": "0.15.1", |   "version": "0.13.0", | ||||||
|   "private": true, |   "private": true, | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "@codemirror/autocomplete": "^6.10.2", |     "@codemirror/autocomplete": "^6.10.2", | ||||||
| @ -10,7 +10,7 @@ | |||||||
|     "@fortawesome/react-fontawesome": "^0.2.0", |     "@fortawesome/react-fontawesome": "^0.2.0", | ||||||
|     "@headlessui/react": "^1.7.17", |     "@headlessui/react": "^1.7.17", | ||||||
|     "@headlessui/tailwindcss": "^0.2.0", |     "@headlessui/tailwindcss": "^0.2.0", | ||||||
|     "@kittycad/lib": "^0.0.53", |     "@kittycad/lib": "^0.0.46", | ||||||
|     "@lezer/javascript": "^1.4.9", |     "@lezer/javascript": "^1.4.9", | ||||||
|     "@open-rpc/client-js": "^1.8.1", |     "@open-rpc/client-js": "^1.8.1", | ||||||
|     "@react-hook/resize-observer": "^1.2.6", |     "@react-hook/resize-observer": "^1.2.6", | ||||||
| @ -21,7 +21,6 @@ | |||||||
|     "@testing-library/react": "^14.0.0", |     "@testing-library/react": "^14.0.0", | ||||||
|     "@testing-library/user-event": "^14.5.1", |     "@testing-library/user-event": "^14.5.1", | ||||||
|     "@ts-stack/markdown": "^1.5.0", |     "@ts-stack/markdown": "^1.5.0", | ||||||
|     "@tweenjs/tween.js": "^23.1.1", |  | ||||||
|     "@types/node": "^16.7.13", |     "@types/node": "^16.7.13", | ||||||
|     "@types/react": "^18.2.41", |     "@types/react": "^18.2.41", | ||||||
|     "@types/react-dom": "^18.0.0", |     "@types/react-dom": "^18.0.0", | ||||||
| @ -46,7 +45,6 @@ | |||||||
|     "sketch-helpers": "^0.0.4", |     "sketch-helpers": "^0.0.4", | ||||||
|     "swr": "^2.2.2", |     "swr": "^2.2.2", | ||||||
|     "tauri-plugin-fs-extra-api": "https://github.com/tauri-apps/tauri-plugin-fs-extra#v1", |     "tauri-plugin-fs-extra-api": "https://github.com/tauri-apps/tauri-plugin-fs-extra#v1", | ||||||
|     "three": "^0.160.0", |  | ||||||
|     "toml": "^3.0.0", |     "toml": "^3.0.0", | ||||||
|     "ts-node": "^10.9.1", |     "ts-node": "^10.9.1", | ||||||
|     "typescript": "^5.2.2", |     "typescript": "^5.2.2", | ||||||
| @ -84,9 +82,7 @@ | |||||||
|     "remove-importmeta": "sed -i 's/import.meta.url/window.location.origin/g' \"./src/wasm-lib/pkg/wasm_lib.js\"; sed -i '' 's/import.meta.url/window.location.origin/g' \"./src/wasm-lib/pkg/wasm_lib.js\" || echo \"sed for both mac and linux\"", |     "remove-importmeta": "sed -i 's/import.meta.url/window.location.origin/g' \"./src/wasm-lib/pkg/wasm_lib.js\"; sed -i '' 's/import.meta.url/window.location.origin/g' \"./src/wasm-lib/pkg/wasm_lib.js\" || echo \"sed for both mac and linux\"", | ||||||
|     "wasm-prep": "rm -rf src/wasm-lib/pkg && mkdir src/wasm-lib/pkg && rm -rf src/wasm-lib/kcl/bindings", |     "wasm-prep": "rm -rf src/wasm-lib/pkg && mkdir src/wasm-lib/pkg && rm -rf src/wasm-lib/kcl/bindings", | ||||||
|     "lint": "eslint --fix src", |     "lint": "eslint --fix src", | ||||||
|     "bump-jsons": "echo \"$(jq --arg v \"$VERSION\" '.version=$v' package.json --indent 2)\" > package.json && echo \"$(jq --arg v \"$VERSION\" '.package.version=$v' src-tauri/tauri.conf.json --indent 2)\" > src-tauri/tauri.conf.json", |     "bump-jsons": "echo \"$(jq --arg v \"$VERSION\" '.version=$v' package.json --indent 2)\" > package.json && echo \"$(jq --arg v \"$VERSION\" '.package.version=$v' src-tauri/tauri.conf.json --indent 2)\" > src-tauri/tauri.conf.json" | ||||||
|     "postinstall": "patch-package && yarn xstate:typegen", |  | ||||||
|     "xstate:typegen": "yarn xstate typegen \"src/**/*.ts?(x)\"" |  | ||||||
|   }, |   }, | ||||||
|   "prettier": { |   "prettier": { | ||||||
|     "trailingComma": "es5", |     "trailingComma": "es5", | ||||||
| @ -117,7 +113,6 @@ | |||||||
|     "@types/pixelmatch": "^5.2.6", |     "@types/pixelmatch": "^5.2.6", | ||||||
|     "@types/pngjs": "^6.0.4", |     "@types/pngjs": "^6.0.4", | ||||||
|     "@types/react-modal": "^3.16.3", |     "@types/react-modal": "^3.16.3", | ||||||
|     "@types/three": "^0.160.0", |  | ||||||
|     "@types/uuid": "^9.0.4", |     "@types/uuid": "^9.0.4", | ||||||
|     "@types/wait-on": "^5.3.4", |     "@types/wait-on": "^5.3.4", | ||||||
|     "@types/wicg-file-system-access": "^2020.9.6", |     "@types/wicg-file-system-access": "^2020.9.6", | ||||||
| @ -129,26 +124,21 @@ | |||||||
|     "@wdio/local-runner": "^8.24.3", |     "@wdio/local-runner": "^8.24.3", | ||||||
|     "@wdio/mocha-framework": "^8.24.3", |     "@wdio/mocha-framework": "^8.24.3", | ||||||
|     "@wdio/spec-reporter": "^8.24.2", |     "@wdio/spec-reporter": "^8.24.2", | ||||||
|     "@xstate/cli": "^0.5.17", |  | ||||||
|     "autoprefixer": "^10.4.13", |     "autoprefixer": "^10.4.13", | ||||||
|     "eslint": "^8.53.0", |     "eslint": "^8.53.0", | ||||||
|     "eslint-config-react-app": "^7.0.1", |     "eslint-config-react-app": "^7.0.1", | ||||||
|     "eslint-plugin-css-modules": "^2.12.0", |     "eslint-plugin-css-modules": "^2.12.0", | ||||||
|     "happy-dom": "^10.8.0", |     "happy-dom": "^10.8.0", | ||||||
|     "husky": "^8.0.3", |     "husky": "^8.0.3", | ||||||
|     "patch-package": "^8.0.0", |  | ||||||
|     "pixelmatch": "^5.3.0", |     "pixelmatch": "^5.3.0", | ||||||
|     "pngjs": "^7.0.0", |     "pngjs": "^7.0.0", | ||||||
|     "postcss": "^8.4.31", |     "postcss": "^8.4.31", | ||||||
|     "postinstall-postinstall": "^2.1.0", |  | ||||||
|     "prettier": "^2.8.0", |     "prettier": "^2.8.0", | ||||||
|     "setimmediate": "^1.0.5", |     "setimmediate": "^1.0.5", | ||||||
|     "tailwindcss": "^3.3.6", |     "tailwindcss": "^3.3.6", | ||||||
|     "vite": "^4.5.2", |     "vite": "^4.5.0", | ||||||
|     "vite-plugin-eslint": "^1.8.1", |     "vite-plugin-eslint": "^1.8.1", | ||||||
|     "vite-plugin-package-version": "^1.1.0", |  | ||||||
|     "vite-tsconfig-paths": "^4.2.1", |     "vite-tsconfig-paths": "^4.2.1", | ||||||
|     "vitest-webgl-canvas-mock": "^1.1.0", |  | ||||||
|     "wait-on": "^7.2.0", |     "wait-on": "^7.2.0", | ||||||
|     "yarn": "^1.22.19" |     "yarn": "^1.22.19" | ||||||
|   } |   } | ||||||
|  | |||||||
| @ -1,138 +0,0 @@ | |||||||
| diff --git a/node_modules/three/examples/jsm/controls/OrbitControls.js b/node_modules/three/examples/jsm/controls/OrbitControls.js |  | ||||||
| index f29e7fe..0ef636b 100644 |  | ||||||
| --- a/node_modules/three/examples/jsm/controls/OrbitControls.js |  | ||||||
| +++ b/node_modules/three/examples/jsm/controls/OrbitControls.js |  | ||||||
| @@ -113,6 +113,25 @@ class OrbitControls extends EventDispatcher { |  | ||||||
|  		// public methods |  | ||||||
|  		// |  | ||||||
|   |  | ||||||
| +		this.interactionGuards = { |  | ||||||
| +			pan: { |  | ||||||
| +				description: 'Right click + Shift + drag or middle click + drag', |  | ||||||
| +				callback: (e) => e.button === 2 && !e.ctrlKey, |  | ||||||
| +			}, |  | ||||||
| +			zoom: { |  | ||||||
| +				description: 'Scroll wheel or Right click + Ctrl + drag', |  | ||||||
| +				dragCallback: (e) => e.button === 2 && e.ctrlKey, |  | ||||||
| +				scrollCallback: () => true, |  | ||||||
| +			}, |  | ||||||
| +			rotate: { |  | ||||||
| +				description: 'Right click + drag', |  | ||||||
| +				callback: (e) => e.button === 0, |  | ||||||
| +			}, |  | ||||||
| +		} |  | ||||||
| +		this.setMouseGuards = (interactionGuards) => { |  | ||||||
| +			this.interactionGuards = interactionGuards |  | ||||||
| +		} |  | ||||||
| + |  | ||||||
|  		this.getPolarAngle = function () { |  | ||||||
|   |  | ||||||
|  			return spherical.phi; |  | ||||||
| @@ -1057,92 +1076,21 @@ class OrbitControls extends EventDispatcher { |  | ||||||
|   |  | ||||||
|  		function onMouseDown( event ) { |  | ||||||
|         |  | ||||||
| -			let mouseAction; |  | ||||||
| - |  | ||||||
| -			switch ( event.button ) { |  | ||||||
| - |  | ||||||
| -				case 0: |  | ||||||
| - |  | ||||||
| -					mouseAction = scope.mouseButtons.LEFT; |  | ||||||
| -					break; |  | ||||||
| - |  | ||||||
| -				case 1: |  | ||||||
| - |  | ||||||
| -					mouseAction = scope.mouseButtons.MIDDLE; |  | ||||||
| -					break; |  | ||||||
| - |  | ||||||
| -				case 2: |  | ||||||
| - |  | ||||||
| -					mouseAction = scope.mouseButtons.RIGHT; |  | ||||||
| -					break; |  | ||||||
| - |  | ||||||
| -				default: |  | ||||||
| - |  | ||||||
| -					mouseAction = - 1; |  | ||||||
| - |  | ||||||
| -			} |  | ||||||
| - |  | ||||||
| -			switch ( mouseAction ) { |  | ||||||
| - |  | ||||||
| -				case MOUSE.DOLLY: |  | ||||||
| - |  | ||||||
| -					if ( scope.enableZoom === false ) return; |  | ||||||
| - |  | ||||||
| -					handleMouseDownDolly( event ); |  | ||||||
| - |  | ||||||
| -					state = STATE.DOLLY; |  | ||||||
| - |  | ||||||
| -					break; |  | ||||||
| - |  | ||||||
| -				case MOUSE.ROTATE: |  | ||||||
| - |  | ||||||
| -					if ( event.ctrlKey || event.metaKey || event.shiftKey ) { |  | ||||||
| - |  | ||||||
| -						if ( scope.enablePan === false ) return; |  | ||||||
| - |  | ||||||
| -						handleMouseDownPan( event ); |  | ||||||
| - |  | ||||||
| -						state = STATE.PAN; |  | ||||||
| - |  | ||||||
| -					} else { |  | ||||||
| - |  | ||||||
| -						if ( scope.enableRotate === false ) return; |  | ||||||
| - |  | ||||||
| -						handleMouseDownRotate( event ); |  | ||||||
| - |  | ||||||
| -						state = STATE.ROTATE; |  | ||||||
| - |  | ||||||
| -					} |  | ||||||
| - |  | ||||||
| -					break; |  | ||||||
| - |  | ||||||
| -				case MOUSE.PAN: |  | ||||||
| - |  | ||||||
| -					if ( event.ctrlKey || event.metaKey || event.shiftKey ) { |  | ||||||
| - |  | ||||||
| -						if ( scope.enableRotate === false ) return; |  | ||||||
| - |  | ||||||
| -						handleMouseDownRotate( event ); |  | ||||||
| - |  | ||||||
| -						state = STATE.ROTATE; |  | ||||||
| - |  | ||||||
| -					} else { |  | ||||||
| - |  | ||||||
| -						if ( scope.enablePan === false ) return; |  | ||||||
| - |  | ||||||
| -						handleMouseDownPan( event ); |  | ||||||
| - |  | ||||||
| -						state = STATE.PAN; |  | ||||||
| - |  | ||||||
| -					} |  | ||||||
| - |  | ||||||
| -					break; |  | ||||||
| - |  | ||||||
| -				default: |  | ||||||
| - |  | ||||||
| -					state = STATE.NONE; |  | ||||||
| - |  | ||||||
| -			} |  | ||||||
| +      if (scope.interactionGuards.pan.callback(event)) { |  | ||||||
| +        if (scope.enablePan === false) return |  | ||||||
| +        handleMouseDownPan(event) |  | ||||||
| +        state = STATE.PAN |  | ||||||
| +      } else if (scope.interactionGuards.rotate.callback(event)) { |  | ||||||
| +        if (scope.enableRotate === false) return |  | ||||||
| +        handleMouseDownRotate(event) |  | ||||||
| +        state = STATE.ROTATE |  | ||||||
| +      } else if (scope.interactionGuards.zoom.dragCallback(event)) { |  | ||||||
| +        if (scope.enableZoom === false) return |  | ||||||
| +        handleMouseDownDolly(event) |  | ||||||
| +        state = STATE.DOLLY |  | ||||||
| +      } else { |  | ||||||
| +        return |  | ||||||
| +      } |  | ||||||
|   |  | ||||||
|  			if ( state !== STATE.NONE ) { |  | ||||||
|   |  | ||||||
| @ -1,4 +1,4 @@ | |||||||
| import { defineConfig, devices } from '@playwright/test' | import { defineConfig, devices } from '@playwright/test'; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Read environment variables from file. |  * Read environment variables from file. | ||||||
| @ -78,4 +78,5 @@ export default defineConfig({ | |||||||
|     // url: 'http://127.0.0.1:3000', |     // url: 'http://127.0.0.1:3000', | ||||||
|     reuseExistingServer: !process.env.CI, |     reuseExistingServer: !process.env.CI, | ||||||
|   }, |   }, | ||||||
| }) | }); | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,26 +0,0 @@ | |||||||
| import requests |  | ||||||
| import os |  | ||||||
|  |  | ||||||
| webhook_url = os.getenv('DISCORD_WEBHOOK_URL') |  | ||||||
| release_version = os.getenv('RELEASE_VERSION') |  | ||||||
| release_body = os.getenv('RELEASE_BODY') |  | ||||||
|  |  | ||||||
| # message to send to Discord |  | ||||||
| data = { |  | ||||||
|     "content":  |  | ||||||
|         f''' |  | ||||||
|         **{release_version}** is now available! Check out the latest features and improvements here: https://zoo.dev/modeling-app/download |  | ||||||
|         {release_body} |  | ||||||
|         ''', |  | ||||||
|     "username": "Modeling App Release Updates", |  | ||||||
|     "avatar_url": "https://raw.githubusercontent.com/KittyCAD/modeling-app/main/public/discord-avatar.png" |  | ||||||
| } |  | ||||||
|  |  | ||||||
| # POST request to the Discord webhook |  | ||||||
| response = requests.post(webhook_url, json=data) |  | ||||||
|  |  | ||||||
| # Check for success |  | ||||||
| if response.status_code == 204: |  | ||||||
|     print("Successfully sent the message to Discord.") |  | ||||||
| else: |  | ||||||
|     print("Failed to send the message to Discord.") |  | ||||||
| Before Width: | Height: | Size: 11 KiB | 
| @ -4,9 +4,9 @@ | |||||||
|  |  | ||||||
| First off, thank you so much for your interest in being a part of the closed Alpha program! We are thrilled to have others use our product and see what you build with it (and truthfully, how you break it too). | First off, thank you so much for your interest in being a part of the closed Alpha program! We are thrilled to have others use our product and see what you build with it (and truthfully, how you break it too). | ||||||
|  |  | ||||||
| ### Zoo Modeling App (ZMA) | ### KittyCAD Modeling App (KCMA) | ||||||
|  |  | ||||||
| What we are introducing to you is our Zoo Modeling App (ZMA). ZMA is a CAD application that expresses a hybrid style of traditional CAD interface along with a code-CAD interface. ZMA is a great way for us to test our own APIs as well as inspire others to develop their own applications. | What we are introducing to you is our KittyCAD Modeling App (KCMA). KCMA is a CAD application that expresses a hybrid style of traditional CAD interface along with a code-CAD interface. KCMA is a great way for us to test our own APIs as well as inspire others to develop their own applications. | ||||||
|  |  | ||||||
| ### Why Code? | ### Why Code? | ||||||
|  |  | ||||||
| @ -18,11 +18,11 @@ Plenty of you have professional CAD experience, and may not understand why codin | |||||||
| - Reproducibility | - Reproducibility | ||||||
| - Easier integration with other tools | - Easier integration with other tools | ||||||
|  |  | ||||||
| ### Before You Use ZMA | ### Before You Use KCMA | ||||||
|  |  | ||||||
| Before you dive straight into the app, we wanted to lay some expectations out for you.  | Before you dive straight into the app, we wanted to lay some expectations out for you.  | ||||||
|  |  | ||||||
| - ZMA is in early development. Kurt pitched the idea back in January, and the team has been working hard on it since then. ZMA has really basic CAD features for now, but we have plenty of features on our roadmap. Most of the features that you may be currently used to in your CAD workflow today will be available down the road. | - KCMA is in early development. Kurt pitched the idea back in January, and the team has been working hard on it since then. KCMA has really basic CAD features for now, but we have plenty of features on our roadmap. Most of the features that you may be currently used to in your CAD workflow today will be available down the road. | ||||||
| - For a list of all scripting functions, please reference our [documentation](https://github.com/KittyCAD/modeling-app/blob/main/docs/kcl/std.md). For a basic rundown of our types, please reference [this document](https://github.com/KittyCAD/modeling-app/blob/main/docs/kcl/types.md). | - For a list of all scripting functions, please reference our [documentation](https://github.com/KittyCAD/modeling-app/blob/main/docs/kcl/std.md). For a basic rundown of our types, please reference [this document](https://github.com/KittyCAD/modeling-app/blob/main/docs/kcl/types.md). | ||||||
| - With that being said, we have created an external new features list in [GH Discussions](https://github.com/KittyCAD/modeling-app/discussions). For our current priority list, please click [here](https://github.com/KittyCAD/modeling-app/blob/main/public/roadmap.md). Please upvote any features in the GH Discussions page that you would like to see implemented first. We will prioritize the highest upvoted items or items that are foundational for other features on the list. You can also add your own, but we will review it to make sure it’s not a duplicate or it’s feasible for the current state of the app. | - With that being said, we have created an external new features list in [GH Discussions](https://github.com/KittyCAD/modeling-app/discussions). For our current priority list, please click [here](https://github.com/KittyCAD/modeling-app/blob/main/public/roadmap.md). Please upvote any features in the GH Discussions page that you would like to see implemented first. We will prioritize the highest upvoted items or items that are foundational for other features on the list. You can also add your own, but we will review it to make sure it’s not a duplicate or it’s feasible for the current state of the app. | ||||||
| - Please report any and all bugs/issues you find. Even the smallest bugs are important! You can report them in a GH Issue [here](https://github.com/KittyCAD/modeling-app/issues/new). You are more than welcome to link your GH Issue in the **bugs** section of our Discord, but if you want to discuss the bug further, please keep that in the GH Issue thread. Please include the severity of the bug in your GH Issue ticket (High, Medium, or Low). If you are having trouble deciding what severity the bug is, use this guideline: | - Please report any and all bugs/issues you find. Even the smallest bugs are important! You can report them in a GH Issue [here](https://github.com/KittyCAD/modeling-app/issues/new). You are more than welcome to link your GH Issue in the **bugs** section of our Discord, but if you want to discuss the bug further, please keep that in the GH Issue thread. Please include the severity of the bug in your GH Issue ticket (High, Medium, or Low). If you are having trouble deciding what severity the bug is, use this guideline: | ||||||
|  | |||||||
| @ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "short_name": "ZMA", |   "short_name": "KCMA", | ||||||
|   "name": "Zoo Modeling App", |   "name": "KittyCAD Modeling App", | ||||||
|   "icons": [ |   "icons": [ | ||||||
|     { |     { | ||||||
|       "src": "favicon.ico", |       "src": "favicon.ico", | ||||||
|  | |||||||
| @ -1,4 +1,4 @@ | |||||||
| ## Zoo Modeling App Roadmap | ## KittyCAD Modeling App Roadmap | ||||||
|  |  | ||||||
| This document ties into our [GH Discussions Feature List](https://github.com/KittyCAD/modeling-app/discussions). Please upvote any features that you want to see next, or add ones that are not listed and we will review.  | This document ties into our [GH Discussions Feature List](https://github.com/KittyCAD/modeling-app/discussions). Please upvote any features that you want to see next, or add ones that are not listed and we will review.  | ||||||
|  |  | ||||||
|  | |||||||
| Before Width: | Height: | Size: 13 KiB | 
| Before Width: | Height: | Size: 11 KiB | 
| Before Width: | Height: | Size: 13 KiB | 
| @ -1,7 +0,0 @@ | |||||||
| <svg width="438" height="145" viewBox="0 0 438 145" fill="none" xmlns="http://www.w3.org/2000/svg"> |  | ||||||
| <path d="M88.2136 25.3021V3.12744H0.595269V34.3994H79.827L0.609484 120.312H0.595269V120.326L0.581055 120.34L0.595269 120.355V141.364H20.8936L41.3341 119.189V141.364H128.952V110.092H49.7349L128.952 24.1649V3.12744L108.64 3.15587L88.2136 25.3021Z" fill="white"/> |  | ||||||
| <path d="M167.36 72.4372C167.36 49.7366 185.824 31.2719 208.525 31.2719C216.514 31.2719 223.976 33.5605 230.288 37.5121L251.78 14.3709C239.698 5.34466 224.73 0 208.525 0C168.582 0 136.088 32.4944 136.088 72.4372C136.088 90.5465 142.769 107.135 153.828 119.857L175.32 96.7156C170.316 89.9069 167.36 81.5061 167.36 72.4372Z" fill="white"/> |  | ||||||
| <path d="M241.745 48.1442C246.734 54.9671 249.691 63.3679 249.691 72.4368C249.691 95.1232 231.226 113.588 208.525 113.588C200.537 113.588 193.088 111.299 186.777 107.348L165.271 130.503C177.353 139.515 192.321 144.86 208.525 144.86C248.468 144.86 280.963 112.365 280.963 72.4368C280.963 54.3133 274.282 37.7249 263.223 25.0029L241.745 48.1442Z" fill="white"/> |  | ||||||
| <path d="M419.312 25.0029L397.834 48.1442C402.823 54.9671 405.779 63.3679 405.779 72.4368C405.779 95.1232 387.315 113.588 364.614 113.588C356.626 113.588 349.177 111.299 342.866 107.348L321.359 130.503C333.442 139.515 348.41 144.86 364.614 144.86C404.557 144.86 437.051 112.365 437.051 72.4368C437.051 54.3133 430.371 37.7249 419.312 25.0029Z" fill="white"/> |  | ||||||
| <path d="M323.449 72.4372C323.449 49.7366 341.913 31.2719 364.614 31.2719C372.603 31.2719 380.065 33.5605 386.376 37.5121L407.869 14.3709C395.786 5.34466 380.819 0 364.614 0C324.671 0 292.177 32.4944 292.177 72.4372C292.177 90.5465 298.858 107.135 309.916 119.857L331.409 96.7156C326.405 89.9069 323.449 81.5061 323.449 72.4372Z" fill="white"/> |  | ||||||
| </svg> |  | ||||||
| Before Width: | Height: | Size: 1.7 KiB | 
							
								
								
									
										246
									
								
								src-tauri/Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						| @ -67,9 +67,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "anyhow" | name = "anyhow" | ||||||
| version = "1.0.79" | version = "1.0.75" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "080e9890a082662b09c1ad45f567faeeb47f22b5fb23895fbe1e651e718e25ca" | checksum = "a4668cab20f66d8d020e1fbc0ebe47217433c1b6c8f2040faf858554e394ace6" | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "app" | name = "app" | ||||||
| @ -95,7 +95,7 @@ checksum = "bc00ceb34980c03614e35a3a4e218276a0a824e911d07651cd0d858a51e8c0f0" | |||||||
| dependencies = [ | dependencies = [ | ||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
|  "quote", |  "quote", | ||||||
|  "syn 2.0.48", |  "syn 2.0.33", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -542,17 +542,17 @@ source = "registry+https://github.com/rust-lang/crates.io-index" | |||||||
| checksum = "13b588ba4ac1a99f7f2964d24b3d896ddc6bf847ee3855dbd4366f058cfcd331" | checksum = "13b588ba4ac1a99f7f2964d24b3d896ddc6bf847ee3855dbd4366f058cfcd331" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "quote", |  "quote", | ||||||
|  "syn 2.0.48", |  "syn 2.0.33", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "ctor" | name = "ctor" | ||||||
| version = "0.2.6" | version = "0.1.26" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "30d2b3721e861707777e3195b0158f950ae6dc4a27e4d02ff9f67e3eb3de199e" | checksum = "6d2301688392eb071b0bf1a37be05c469d3cc4dbbd95df672fe28ab021e6a096" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "quote", |  "quote", | ||||||
|  "syn 2.0.48", |  "syn 1.0.109", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -582,7 +582,7 @@ dependencies = [ | |||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
|  "quote", |  "quote", | ||||||
|  "strsim", |  "strsim", | ||||||
|  "syn 2.0.48", |  "syn 2.0.33", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -593,7 +593,7 @@ checksum = "29a358ff9f12ec09c3e61fef9b5a9902623a695a46a917b07f269bff1445611a" | |||||||
| dependencies = [ | dependencies = [ | ||||||
|  "darling_core", |  "darling_core", | ||||||
|  "quote", |  "quote", | ||||||
|  "syn 2.0.48", |  "syn 2.0.33", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -899,7 +899,7 @@ checksum = "89ca545a94061b6365f2c7355b4b32bd20df3ff95f02da9329b34ccc3bd6ee72" | |||||||
| dependencies = [ | dependencies = [ | ||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
|  "quote", |  "quote", | ||||||
|  "syn 2.0.48", |  "syn 2.0.33", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -1242,9 +1242,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "h2" | name = "h2" | ||||||
| version = "0.3.24" | version = "0.3.20" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "bb2c4422095b67ee78da96fbb51a4cc413b3b25883c7717ff7ca1ab31022c9c9" | checksum = "97ec8491ebaf99c8eaa73058b045fe58073cd6be7f596ac993ced0b0a0c01049" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "bytes", |  "bytes", | ||||||
|  "fnv", |  "fnv", | ||||||
| @ -1252,7 +1252,7 @@ dependencies = [ | |||||||
|  "futures-sink", |  "futures-sink", | ||||||
|  "futures-util", |  "futures-util", | ||||||
|  "http", |  "http", | ||||||
|  "indexmap 2.0.0", |  "indexmap 1.9.3", | ||||||
|  "slab", |  "slab", | ||||||
|  "tokio", |  "tokio", | ||||||
|  "tokio-util", |  "tokio-util", | ||||||
| @ -1530,9 +1530,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "infer" | name = "infer" | ||||||
| version = "0.13.0" | version = "0.12.0" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "f551f8c3a39f68f986517db0d1759de85881894fdc7db798bd2a9df9cb04b7fc" | checksum = "a898e4b7951673fce96614ce5751d13c40fc5674bc2d759288e46c3ab62598b3" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "cfb", |  "cfb", | ||||||
| ] | ] | ||||||
| @ -1652,9 +1652,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "json-patch" | name = "json-patch" | ||||||
| version = "1.2.0" | version = "1.0.0" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "55ff1e1486799e3f64129f8ccad108b38290df9cd7015cd31bed17239f0789d6" | checksum = "1f54898088ccb91df1b492cc80029a6fdf1c48ca0db7c6822a8babad69c94658" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "serde", |  "serde", | ||||||
|  "serde_json", |  "serde_json", | ||||||
| @ -1664,9 +1664,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "kittycad" | name = "kittycad" | ||||||
| version = "0.2.53" | version = "0.2.42" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "a086e1a1bbddb3b38959c0f0ce6de6b3a3b7566e38e0b7d5fb101e91911beed4" | checksum = "6aa554d86b6dbbd976a659c912ae25ce817b4378eb12a5684907e263410f0a7b" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "anyhow", |  "anyhow", | ||||||
|  "async-trait", |  "async-trait", | ||||||
| @ -2194,11 +2194,11 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "openssl" | name = "openssl" | ||||||
| version = "0.10.61" | version = "0.10.55" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "6b8419dc8cc6d866deb801274bba2e6f8f6108c1bb7fcc10ee5ab864931dbb45" | checksum = "345df152bc43501c5eb9e4654ff05f794effb78d4efe3d53abc158baddc0703d" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "bitflags 2.4.0", |  "bitflags 1.3.2", | ||||||
|  "cfg-if", |  "cfg-if", | ||||||
|  "foreign-types", |  "foreign-types", | ||||||
|  "libc", |  "libc", | ||||||
| @ -2215,7 +2215,7 @@ checksum = "a948666b637a0f465e8564c73e89d4dde00d72d4d473cc972f390fc3dcee7d9c" | |||||||
| dependencies = [ | dependencies = [ | ||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
|  "quote", |  "quote", | ||||||
|  "syn 2.0.48", |  "syn 2.0.33", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -2226,9 +2226,9 @@ checksum = "ff011a302c396a5197692431fc1948019154afc178baf7d8e37367442a4601cf" | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "openssl-sys" | name = "openssl-sys" | ||||||
| version = "0.9.97" | version = "0.9.90" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "c3eaad34cdd97d81de97964fc7f29e2d104f483840d906ef56daa1912338460b" | checksum = "374533b0e45f3a7ced10fcaeccca020e66656bc03dac384f852e4e5a7a8104a6" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "cc", |  "cc", | ||||||
|  "libc", |  "libc", | ||||||
| @ -2368,7 +2368,7 @@ dependencies = [ | |||||||
|  "regex", |  "regex", | ||||||
|  "regex-syntax 0.7.5", |  "regex-syntax 0.7.5", | ||||||
|  "structmeta", |  "structmeta", | ||||||
|  "syn 2.0.48", |  "syn 2.0.33", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -2400,17 +2400,9 @@ version = "0.10.1" | |||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "fabbf1ead8a5bcbc20f5f8b939ee3f5b0f6f281b6ad3468b84656b658b455259" | checksum = "fabbf1ead8a5bcbc20f5f8b939ee3f5b0f6f281b6ad3468b84656b658b455259" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  |  "phf_macros 0.10.0", | ||||||
|  "phf_shared 0.10.0", |  "phf_shared 0.10.0", | ||||||
| ] |  "proc-macro-hack", | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "phf" |  | ||||||
| version = "0.11.2" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "ade2d8b8f33c7333b51bcf0428d37e217e9f32192ae4772156f65063b8ce03dc" |  | ||||||
| dependencies = [ |  | ||||||
|  "phf_macros 0.11.2", |  | ||||||
|  "phf_shared 0.11.2", |  | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -2453,16 +2445,6 @@ dependencies = [ | |||||||
|  "rand 0.8.5", |  "rand 0.8.5", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "phf_generator" |  | ||||||
| version = "0.11.2" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "48e4cc64c2ad9ebe670cb8fd69dd50ae301650392e81c05f9bfcb2d5bdbc24b0" |  | ||||||
| dependencies = [ |  | ||||||
|  "phf_shared 0.11.2", |  | ||||||
|  "rand 0.8.5", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "phf_macros" | name = "phf_macros" | ||||||
| version = "0.8.0" | version = "0.8.0" | ||||||
| @ -2479,15 +2461,16 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "phf_macros" | name = "phf_macros" | ||||||
| version = "0.11.2" | version = "0.10.0" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "3444646e286606587e49f3bcf1679b8cef1dc2c5ecc29ddacaffc305180d464b" | checksum = "58fdf3184dd560f160dd73922bea2d5cd6e8f064bf4b13110abd81b03697b4e0" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "phf_generator 0.11.2", |  "phf_generator 0.10.0", | ||||||
|  "phf_shared 0.11.2", |  "phf_shared 0.10.0", | ||||||
|  |  "proc-macro-hack", | ||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
|  "quote", |  "quote", | ||||||
|  "syn 2.0.48", |  "syn 1.0.109", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -2508,15 +2491,6 @@ dependencies = [ | |||||||
|  "siphasher", |  "siphasher", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "phf_shared" |  | ||||||
| version = "0.11.2" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "90fcb95eef784c2ac79119d1dd819e162b5da872ce6f3c3abe1e8ca1c082f72b" |  | ||||||
| dependencies = [ |  | ||||||
|  "siphasher", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "phonenumber" | name = "phonenumber" | ||||||
| version = "0.3.3+8.13.9" | version = "0.3.3+8.13.9" | ||||||
| @ -2555,7 +2529,7 @@ checksum = "4359fd9c9171ec6e8c62926d6faaf553a8dc3f64e1507e76da7911b4f6a04405" | |||||||
| dependencies = [ | dependencies = [ | ||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
|  "quote", |  "quote", | ||||||
|  "syn 2.0.48", |  "syn 2.0.33", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -2657,9 +2631,9 @@ checksum = "dc375e1527247fe1a97d8b7156678dfe7c1af2fc075c9a4db3690ecd2a148068" | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "proc-macro2" | name = "proc-macro2" | ||||||
| version = "1.0.78" | version = "1.0.67" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "e2422ad645d89c99f8f3e6b88a9fdeca7fabeac836b1002371c4367c8f984aae" | checksum = "3d433d9f1a3e8c1263d9456598b16fec66f4acc9a74dacffd35c7bb09b3a1328" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "unicode-ident", |  "unicode-ident", | ||||||
| ] | ] | ||||||
| @ -2675,9 +2649,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "quote" | name = "quote" | ||||||
| version = "1.0.35" | version = "1.0.33" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "291ec9ab5efd934aaf503a6466c5d5251535d108ee747472c3977cc5acc868ef" | checksum = "5267fca4496028628a95160fc423a33e8b2e6af8a5302579e322e4b520293cae" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
| ] | ] | ||||||
| @ -3235,9 +3209,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "serde" | name = "serde" | ||||||
| version = "1.0.196" | version = "1.0.193" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "870026e60fa08c69f064aa766c10f10b1d62db9ccd4d0abb206472bee0ce3b32" | checksum = "25dd9975e68d0cb5aa1120c288333fc98731bd1dd12f561e468ea4728c042b89" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "serde_derive", |  "serde_derive", | ||||||
| ] | ] | ||||||
| @ -3253,13 +3227,13 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "serde_derive" | name = "serde_derive" | ||||||
| version = "1.0.196" | version = "1.0.193" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "33c85360c95e7d137454dc81d9a4ed2b8efd8fbe19cee57357b32b9771fccb67" | checksum = "43576ca501357b9b071ac53cdc7da8ef0cbd9493d8df094cd821777ea6e894d3" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
|  "quote", |  "quote", | ||||||
|  "syn 2.0.48", |  "syn 2.0.33", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -3275,9 +3249,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "serde_json" | name = "serde_json" | ||||||
| version = "1.0.113" | version = "1.0.108" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "69801b70b1c3dac963ecb03a364ba0ceda9cf60c71cfe475e99864759c8b8a79" | checksum = "3d1c7e3eac408d115102c4c24ad393e0821bb3a5df4d506a80f85f7a742a526b" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "itoa 1.0.6", |  "itoa 1.0.6", | ||||||
|  "ryu", |  "ryu", | ||||||
| @ -3302,7 +3276,7 @@ checksum = "bcec881020c684085e55a25f7fd888954d56609ef363479dc5a1305eb0d40cab" | |||||||
| dependencies = [ | dependencies = [ | ||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
|  "quote", |  "quote", | ||||||
|  "syn 2.0.48", |  "syn 2.0.33", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -3352,7 +3326,7 @@ dependencies = [ | |||||||
|  "darling", |  "darling", | ||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
|  "quote", |  "quote", | ||||||
|  "syn 2.0.48", |  "syn 2.0.33", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -3556,7 +3530,7 @@ dependencies = [ | |||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
|  "quote", |  "quote", | ||||||
|  "structmeta-derive", |  "structmeta-derive", | ||||||
|  "syn 2.0.48", |  "syn 2.0.33", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -3567,7 +3541,7 @@ checksum = "a60bcaff7397072dca0017d1db428e30d5002e00b6847703e2e42005c95fbe00" | |||||||
| dependencies = [ | dependencies = [ | ||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
|  "quote", |  "quote", | ||||||
|  "syn 2.0.48", |  "syn 2.0.33", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -3605,9 +3579,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "syn" | name = "syn" | ||||||
| version = "2.0.48" | version = "2.0.33" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "0f3531638e407dfc0814761abb7c00a5b54992b849452a0646b7f65c9f770f3f" | checksum = "9caece70c63bfba29ec2fed841a09851b14a235c60010fa4de58089b6c025668" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
|  "quote", |  "quote", | ||||||
| @ -3760,9 +3734,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "tauri" | name = "tauri" | ||||||
| version = "1.5.4" | version = "1.5.2" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "fd27c04b9543776a972c86ccf70660b517ecabbeced9fb58d8b961a13ad129af" | checksum = "9bfe673cf125ef364d6f56b15e8ce7537d9ca7e4dae1cf6fbbdeed2e024db3d9" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "anyhow", |  "anyhow", | ||||||
|  "bytes", |  "bytes", | ||||||
| @ -3812,9 +3786,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "tauri-build" | name = "tauri-build" | ||||||
| version = "1.5.1" | version = "1.5.0" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "e9914a4715e0b75d9f387a285c7e26b5bbfeb1249ad9f842675a82481565c532" | checksum = "defbfc551bd38ab997e5f8e458f87396d2559d05ce32095076ad6c30f7fc5f9c" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "anyhow", |  "anyhow", | ||||||
|  "cargo_toml", |  "cargo_toml", | ||||||
| @ -3831,9 +3805,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "tauri-codegen" | name = "tauri-codegen" | ||||||
| version = "1.4.2" | version = "1.4.1" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "a1554c5857f65dbc377cefb6b97c8ac77b1cb2a90d30d3448114d5d6b48a77fc" | checksum = "7b3475e55acec0b4a50fb96435f19631fb58cbcd31923e1a213de5c382536bbb" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "base64 0.21.2", |  "base64 0.21.2", | ||||||
|  "brotli", |  "brotli", | ||||||
| @ -3857,9 +3831,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "tauri-macros" | name = "tauri-macros" | ||||||
| version = "1.4.3" | version = "1.4.1" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "277abf361a3a6993ec16bcbb179de0d6518009b851090a01adfea12ac89fa875" | checksum = "613740228de92d9196b795ac455091d3a5fbdac2654abb8bb07d010b62ab43af" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "heck 0.4.1", |  "heck 0.4.1", | ||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
| @ -3872,7 +3846,7 @@ dependencies = [ | |||||||
| [[package]] | [[package]] | ||||||
| name = "tauri-plugin-fs-extra" | name = "tauri-plugin-fs-extra" | ||||||
| version = "0.0.0" | version = "0.0.0" | ||||||
| source = "git+https://github.com/tauri-apps/plugins-workspace?branch=v1#01211ff0759d578e0e9ac8c98c31fdf09077eb34" | source = "git+https://github.com/tauri-apps/plugins-workspace?branch=v1#537053d3171a7374a1a86fed422523e7b45a4fb8" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "log", |  "log", | ||||||
|  "serde", |  "serde", | ||||||
| @ -3883,9 +3857,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "tauri-runtime" | name = "tauri-runtime" | ||||||
| version = "0.14.2" | version = "0.14.1" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "cf2d0652aa2891ff3e9caa2401405257ea29ab8372cce01f186a5825f1bd0e76" | checksum = "07f8e9e53e00e9f41212c115749e87d5cd2a9eebccafca77a19722eeecd56d43" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "gtk", |  "gtk", | ||||||
|  "http", |  "http", | ||||||
| @ -3904,9 +3878,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "tauri-runtime-wry" | name = "tauri-runtime-wry" | ||||||
| version = "0.14.3" | version = "0.14.1" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "6cae61fbc731f690a4899681c9052dde6d05b159b44563ace8186fc1bfb7d158" | checksum = "8141d72b6b65f2008911e9ef5b98a68d1e3413b7a1464e8f85eb3673bb19a895" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "cocoa", |  "cocoa", | ||||||
|  "gtk", |  "gtk", | ||||||
| @ -3924,9 +3898,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "tauri-utils" | name = "tauri-utils" | ||||||
| version = "1.5.2" | version = "1.5.0" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "ece74810b1d3d44f29f732a7ae09a63183d63949bbdd59c61f8ed2a1b70150db" | checksum = "34d55e185904a84a419308d523c2c6891d5e2dbcee740c4997eb42e75a7b0f46" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "brotli", |  "brotli", | ||||||
|  "ctor", |  "ctor", | ||||||
| @ -3939,7 +3913,7 @@ dependencies = [ | |||||||
|  "kuchikiki", |  "kuchikiki", | ||||||
|  "log", |  "log", | ||||||
|  "memchr", |  "memchr", | ||||||
|  "phf 0.11.2", |  "phf 0.10.1", | ||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
|  "quote", |  "quote", | ||||||
|  "semver", |  "semver", | ||||||
| @ -3949,7 +3923,7 @@ dependencies = [ | |||||||
|  "thiserror", |  "thiserror", | ||||||
|  "url", |  "url", | ||||||
|  "walkdir", |  "walkdir", | ||||||
|  "windows-version", |  "windows 0.39.0", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -4009,7 +3983,7 @@ checksum = "f9456a42c5b0d803c8cd86e73dd7cc9edd429499f37a3550d286d5e86720569f" | |||||||
| dependencies = [ | dependencies = [ | ||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
|  "quote", |  "quote", | ||||||
|  "syn 2.0.48", |  "syn 2.0.33", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -4051,9 +4025,9 @@ checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20" | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "tokio" | name = "tokio" | ||||||
| version = "1.36.0" | version = "1.34.0" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "61285f6515fa018fb2d1e46eb21223fff441ee8db5d0f1435e8ab4f5cdb80931" | checksum = "d0c014766411e834f7af5b8f4cf46257aab4036ca95e9d2c144a10f59ad6f5b9" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "backtrace", |  "backtrace", | ||||||
|  "bytes", |  "bytes", | ||||||
| @ -4193,7 +4167,7 @@ checksum = "0f57e3ca2a01450b1a921183a9c9cbfda207fd822cef4ccb00a65402cbba7a74" | |||||||
| dependencies = [ | dependencies = [ | ||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
|  "quote", |  "quote", | ||||||
|  "syn 2.0.48", |  "syn 2.0.33", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -4443,7 +4417,7 @@ dependencies = [ | |||||||
|  "once_cell", |  "once_cell", | ||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
|  "quote", |  "quote", | ||||||
|  "syn 2.0.48", |  "syn 2.0.33", | ||||||
|  "wasm-bindgen-shared", |  "wasm-bindgen-shared", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| @ -4477,7 +4451,7 @@ checksum = "e128beba882dd1eb6200e1dc92ae6c5dbaa4311aa7bb211ca035779e5efc39f8" | |||||||
| dependencies = [ | dependencies = [ | ||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
|  "quote", |  "quote", | ||||||
|  "syn 2.0.48", |  "syn 2.0.33", | ||||||
|  "wasm-bindgen-backend", |  "wasm-bindgen-backend", | ||||||
|  "wasm-bindgen-shared", |  "wasm-bindgen-shared", | ||||||
| ] | ] | ||||||
| @ -4773,36 +4747,12 @@ dependencies = [ | |||||||
|  "windows_x86_64_msvc 0.48.0", |  "windows_x86_64_msvc 0.48.0", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "windows-targets" |  | ||||||
| version = "0.52.0" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "8a18201040b24831fbb9e4eb208f8892e1f50a37feb53cc7ff887feb8f50e7cd" |  | ||||||
| dependencies = [ |  | ||||||
|  "windows_aarch64_gnullvm 0.52.0", |  | ||||||
|  "windows_aarch64_msvc 0.52.0", |  | ||||||
|  "windows_i686_gnu 0.52.0", |  | ||||||
|  "windows_i686_msvc 0.52.0", |  | ||||||
|  "windows_x86_64_gnu 0.52.0", |  | ||||||
|  "windows_x86_64_gnullvm 0.52.0", |  | ||||||
|  "windows_x86_64_msvc 0.52.0", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "windows-tokens" | name = "windows-tokens" | ||||||
| version = "0.39.0" | version = "0.39.0" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "f838de2fe15fe6bac988e74b798f26499a8b21a9d97edec321e79b28d1d7f597" | checksum = "f838de2fe15fe6bac988e74b798f26499a8b21a9d97edec321e79b28d1d7f597" | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "windows-version" |  | ||||||
| version = "0.1.0" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "75aa004c988e080ad34aff5739c39d0312f4684699d6d71fc8a198d057b8b9b4" |  | ||||||
| dependencies = [ |  | ||||||
|  "windows-targets 0.52.0", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "windows_aarch64_gnullvm" | name = "windows_aarch64_gnullvm" | ||||||
| version = "0.42.2" | version = "0.42.2" | ||||||
| @ -4815,12 +4765,6 @@ version = "0.48.0" | |||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "91ae572e1b79dba883e0d315474df7305d12f569b400fcf90581b06062f7e1bc" | checksum = "91ae572e1b79dba883e0d315474df7305d12f569b400fcf90581b06062f7e1bc" | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "windows_aarch64_gnullvm" |  | ||||||
| version = "0.52.0" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "cb7764e35d4db8a7921e09562a0304bf2f93e0a51bfccee0bd0bb0b666b015ea" |  | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "windows_aarch64_msvc" | name = "windows_aarch64_msvc" | ||||||
| version = "0.37.0" | version = "0.37.0" | ||||||
| @ -4845,12 +4789,6 @@ version = "0.48.0" | |||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "b2ef27e0d7bdfcfc7b868b317c1d32c641a6fe4629c171b8928c7b08d98d7cf3" | checksum = "b2ef27e0d7bdfcfc7b868b317c1d32c641a6fe4629c171b8928c7b08d98d7cf3" | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "windows_aarch64_msvc" |  | ||||||
| version = "0.52.0" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "bbaa0368d4f1d2aaefc55b6fcfee13f41544ddf36801e793edbbfd7d7df075ef" |  | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "windows_i686_gnu" | name = "windows_i686_gnu" | ||||||
| version = "0.37.0" | version = "0.37.0" | ||||||
| @ -4875,12 +4813,6 @@ version = "0.48.0" | |||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "622a1962a7db830d6fd0a69683c80a18fda201879f0f447f065a3b7467daa241" | checksum = "622a1962a7db830d6fd0a69683c80a18fda201879f0f447f065a3b7467daa241" | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "windows_i686_gnu" |  | ||||||
| version = "0.52.0" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "a28637cb1fa3560a16915793afb20081aba2c92ee8af57b4d5f28e4b3e7df313" |  | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "windows_i686_msvc" | name = "windows_i686_msvc" | ||||||
| version = "0.37.0" | version = "0.37.0" | ||||||
| @ -4905,12 +4837,6 @@ version = "0.48.0" | |||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "4542c6e364ce21bf45d69fdd2a8e455fa38d316158cfd43b3ac1c5b1b19f8e00" | checksum = "4542c6e364ce21bf45d69fdd2a8e455fa38d316158cfd43b3ac1c5b1b19f8e00" | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "windows_i686_msvc" |  | ||||||
| version = "0.52.0" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "ffe5e8e31046ce6230cc7215707b816e339ff4d4d67c65dffa206fd0f7aa7b9a" |  | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "windows_x86_64_gnu" | name = "windows_x86_64_gnu" | ||||||
| version = "0.37.0" | version = "0.37.0" | ||||||
| @ -4935,12 +4861,6 @@ version = "0.48.0" | |||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "ca2b8a661f7628cbd23440e50b05d705db3686f894fc9580820623656af974b1" | checksum = "ca2b8a661f7628cbd23440e50b05d705db3686f894fc9580820623656af974b1" | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "windows_x86_64_gnu" |  | ||||||
| version = "0.52.0" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "3d6fa32db2bc4a2f5abeacf2b69f7992cd09dca97498da74a151a3132c26befd" |  | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "windows_x86_64_gnullvm" | name = "windows_x86_64_gnullvm" | ||||||
| version = "0.42.2" | version = "0.42.2" | ||||||
| @ -4953,12 +4873,6 @@ version = "0.48.0" | |||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "7896dbc1f41e08872e9d5e8f8baa8fdd2677f29468c4e156210174edc7f7b953" | checksum = "7896dbc1f41e08872e9d5e8f8baa8fdd2677f29468c4e156210174edc7f7b953" | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "windows_x86_64_gnullvm" |  | ||||||
| version = "0.52.0" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "1a657e1e9d3f514745a572a6846d3c7aa7dbe1658c056ed9c3344c4109a6949e" |  | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "windows_x86_64_msvc" | name = "windows_x86_64_msvc" | ||||||
| version = "0.37.0" | version = "0.37.0" | ||||||
| @ -4983,12 +4897,6 @@ version = "0.48.0" | |||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "1a515f5799fe4961cb532f983ce2b23082366b898e52ffbce459c86f67c8378a" | checksum = "1a515f5799fe4961cb532f983ce2b23082366b898e52ffbce459c86f67c8378a" | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "windows_x86_64_msvc" |  | ||||||
| version = "0.52.0" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "dff9641d1cd4be8d1a070daf9e3773c5f67e78b4d9d42263020c057706765c04" |  | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "winnow" | name = "winnow" | ||||||
| version = "0.5.15" | version = "0.5.15" | ||||||
| @ -5020,9 +4928,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "wry" | name = "wry" | ||||||
| version = "0.24.6" | version = "0.24.4" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "64a70547e8f9d85da0f5af609143f7bde3ac7457a6e1073104d9b73d6c5ac744" | checksum = "88ef04bdad49eba2e01f06e53688c8413bd6a87b0bc14b72284465cf96e3578e" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "base64 0.13.1", |  "base64 0.13.1", | ||||||
|  "block", |  "block", | ||||||
|  | |||||||
| @ -12,17 +12,17 @@ rust-version = "1.60" | |||||||
| # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html | # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html | ||||||
|  |  | ||||||
| [build-dependencies] | [build-dependencies] | ||||||
| tauri-build = { version = "1.5.1", features = [] } | tauri-build = { version = "1.5.0", features = [] } | ||||||
|  |  | ||||||
| [dependencies] | [dependencies] | ||||||
| anyhow = "1" | anyhow = "1" | ||||||
| kittycad = "0.2.53" | kittycad = "0.2.42" | ||||||
| oauth2 = "4.4.2" | oauth2 = "4.4.2" | ||||||
| serde = { version = "1.0", features = ["derive"] } | serde = { version = "1.0", features = ["derive"] } | ||||||
| serde_json = "1.0" | serde_json = "1.0" | ||||||
| tauri = { version = "1.5.4", features = [ "os-all", "dialog-all", "fs-all", "http-request", "path-all", "shell-open", "shell-open-api", "devtools"] } | tauri = { version = "1.5.2", features = [ "os-all", "dialog-all", "fs-all", "http-request", "path-all", "shell-open", "shell-open-api", "devtools"] } | ||||||
| tauri-plugin-fs-extra = { git = "https://github.com/tauri-apps/plugins-workspace", branch = "v1" } | tauri-plugin-fs-extra = { git = "https://github.com/tauri-apps/plugins-workspace", branch = "v1" } | ||||||
| tokio = { version = "1.36.0", features = ["time"] } | tokio = { version = "1.34.0", features = ["time"] } | ||||||
| toml = "0.8.2" | toml = "0.8.2" | ||||||
|  |  | ||||||
| [features] | [features] | ||||||
|  | |||||||
| Before Width: | Height: | Size: 10 KiB After Width: | Height: | Size: 20 KiB | 
| Before Width: | Height: | Size: 25 KiB After Width: | Height: | Size: 48 KiB | 
| Before Width: | Height: | Size: 1.8 KiB After Width: | Height: | Size: 2.7 KiB | 
| Before Width: | Height: | Size: 8.4 KiB After Width: | Height: | Size: 16 KiB | 
| Before Width: | Height: | Size: 12 KiB After Width: | Height: | Size: 24 KiB | 
| Before Width: | Height: | Size: 13 KiB After Width: | Height: | Size: 25 KiB | 
| Before Width: | Height: | Size: 29 KiB After Width: | Height: | Size: 55 KiB | 
| Before Width: | Height: | Size: 1.6 KiB After Width: | Height: | Size: 2.5 KiB | 
| Before Width: | Height: | Size: 33 KiB After Width: | Height: | Size: 62 KiB | 
| Before Width: | Height: | Size: 2.7 KiB After Width: | Height: | Size: 4.5 KiB | 
| Before Width: | Height: | Size: 5.0 KiB After Width: | Height: | Size: 9.2 KiB | 
| Before Width: | Height: | Size: 6.6 KiB After Width: | Height: | Size: 13 KiB | 
| Before Width: | Height: | Size: 3.1 KiB After Width: | Height: | Size: 5.4 KiB | 
| Before Width: | Height: | Size: 39 KiB After Width: | Height: | Size: 68 KiB | 
| Before Width: | Height: | Size: 69 KiB After Width: | Height: | Size: 122 KiB | 
| @ -7,7 +7,6 @@ use std::io::Read; | |||||||
|  |  | ||||||
| use anyhow::Result; | use anyhow::Result; | ||||||
| use oauth2::TokenResponse; | use oauth2::TokenResponse; | ||||||
| use std::process::Command; |  | ||||||
| use tauri::{InvokeError, Manager}; | use tauri::{InvokeError, Manager}; | ||||||
| const DEFAULT_HOST: &str = "https://api.kittycad.io"; | const DEFAULT_HOST: &str = "https://api.kittycad.io"; | ||||||
|  |  | ||||||
| @ -77,13 +76,16 @@ async fn login(app: tauri::AppHandle, host: &str) -> Result<String, InvokeError> | |||||||
|     // Here we're using an env var to enable the /tmp file (windows not supported for now) |     // Here we're using an env var to enable the /tmp file (windows not supported for now) | ||||||
|     // and bypass the shell::open call as it fails on GitHub Actions. |     // and bypass the shell::open call as it fails on GitHub Actions. | ||||||
|     let e2e_tauri_enabled = env::var("E2E_TAURI_ENABLED").is_ok(); |     let e2e_tauri_enabled = env::var("E2E_TAURI_ENABLED").is_ok(); | ||||||
|     if e2e_tauri_enabled { |     if (e2e_tauri_enabled) { | ||||||
|         println!( |         println!( | ||||||
|             "E2E_TAURI_ENABLED is set, won't open {} externally", |             "E2E_TAURI_ENABLED is set, won't open {} externally", | ||||||
|             auth_uri.secret() |             auth_uri.secret() | ||||||
|         ); |         ); | ||||||
|         fs::write("/tmp/kittycad_user_code", details.user_code().secret()) |         fs::write( | ||||||
|             .expect("Unable to write /tmp/kittycad_user_code file"); |             "/tmp/kittycad_user_code", | ||||||
|  |             details.user_code().secret().to_string(), | ||||||
|  |         ) | ||||||
|  |         .expect("Unable to write /tmp/kittycad_user_code file"); | ||||||
|     } else { |     } else { | ||||||
|         tauri::api::shell::open(&app.shell_scope(), auth_uri.secret(), None) |         tauri::api::shell::open(&app.shell_scope(), auth_uri.secret(), None) | ||||||
|             .map_err(|e| InvokeError::from_anyhow(e.into()))?; |             .map_err(|e| InvokeError::from_anyhow(e.into()))?; | ||||||
|  | |||||||
| @ -6,8 +6,8 @@ | |||||||
|     "distDir": "../build" |     "distDir": "../build" | ||||||
|   }, |   }, | ||||||
|   "package": { |   "package": { | ||||||
|     "productName": "zoo-modeling-app", |     "productName": "kittycad-modeling", | ||||||
|     "version": "0.15.1" |     "version": "0.13.0" | ||||||
|   }, |   }, | ||||||
|   "tauri": { |   "tauri": { | ||||||
|     "allowlist": { |     "allowlist": { | ||||||
| @ -84,7 +84,7 @@ | |||||||
|         "fullscreen": false, |         "fullscreen": false, | ||||||
|         "height": 1200, |         "height": 1200, | ||||||
|         "resizable": true, |         "resizable": true, | ||||||
|         "title": "Zoo Modeling App", |         "title": "KittyCAD Modeling", | ||||||
|         "width": 1800 |         "width": 1800 | ||||||
|       } |       } | ||||||
|     ] |     ] | ||||||
|  | |||||||
| @ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "$schema": "../node_modules/@tauri-apps/cli/schema.json", |   "$schema": "../node_modules/@tauri-apps/cli/schema.json", | ||||||
|   "package": { |   "package": { | ||||||
|     "productName": "Zoo Modeling App" |     "productName": "KittyCAD Modeling" | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  | |||||||
| @ -4,7 +4,7 @@ | |||||||
|     "updater": { |     "updater": { | ||||||
|       "active": true, |       "active": true, | ||||||
|       "endpoints": [ |       "endpoints": [ | ||||||
|         "https://dl.zoo.dev/releases/modeling-app/last_update.json" |         "https://dl.kittycad.io/releases/modeling-app/last_update.json" | ||||||
|       ], |       ], | ||||||
|       "dialog": true, |       "dialog": true, | ||||||
|       "pubkey": "dW50cnVzdGVkIGNvbW1lbnQ6IG1pbmlzaWduIHB1YmxpYyBrZXk6IEUzNzA4MjBEQjFBRTY4NzYKUldSMmFLNnhEWUp3NCtsT21Jd05wQktOaGVkOVp6MUFma0hNTDRDSnI2RkJJTEZOWG1ncFhqcU8K" |       "pubkey": "dW50cnVzdGVkIGNvbW1lbnQ6IG1pbmlzaWduIHB1YmxpYyBrZXk6IEUzNzA4MjBEQjFBRTY4NzYKUldSMmFLNnhEWUp3NCtsT21Jd05wQktOaGVkOVp6MUFma0hNTDRDSnI2RkJJTEZOWG1ncFhqcU8K" | ||||||
|  | |||||||
| @ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "$schema": "../node_modules/@tauri-apps/cli/schema.json", |   "$schema": "../node_modules/@tauri-apps/cli/schema.json", | ||||||
|   "package": { |   "package": { | ||||||
|     "productName": "Zoo Modeling App" |     "productName": "KittyCAD Modeling" | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										73
									
								
								src/App.test.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,73 @@ | |||||||
|  | import { render, screen } from '@testing-library/react' | ||||||
|  | import { App } from './App' | ||||||
|  | import { describe, test, vi } from 'vitest' | ||||||
|  | import { | ||||||
|  |   Route, | ||||||
|  |   RouterProvider, | ||||||
|  |   createMemoryRouter, | ||||||
|  |   createRoutesFromElements, | ||||||
|  | } from 'react-router-dom' | ||||||
|  | import { GlobalStateProvider } from './components/GlobalStateProvider' | ||||||
|  | import CommandBarProvider from 'components/CommandBar/CommandBar' | ||||||
|  | import ModelingMachineProvider from 'components/ModelingMachineProvider' | ||||||
|  | import { BROWSER_FILE_NAME } from 'Router' | ||||||
|  |  | ||||||
|  | let listener: ((rect: any) => void) | undefined = undefined | ||||||
|  | ;(global as any).ResizeObserver = class ResizeObserver { | ||||||
|  |   constructor(ls: ((rect: any) => void) | undefined) { | ||||||
|  |     listener = ls | ||||||
|  |   } | ||||||
|  |   observe() {} | ||||||
|  |   unobserve() {} | ||||||
|  |   disconnect() {} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | describe('App tests', () => { | ||||||
|  |   test('Renders the modeling app screen, including "Variables" pane.', () => { | ||||||
|  |     vi.mock('react-router-dom', async () => { | ||||||
|  |       const actual = (await vi.importActual('react-router-dom')) as Record< | ||||||
|  |         string, | ||||||
|  |         any | ||||||
|  |       > | ||||||
|  |       return { | ||||||
|  |         ...actual, | ||||||
|  |         useParams: () => ({ id: BROWSER_FILE_NAME }), | ||||||
|  |         useLoaderData: () => ({ code: null }), | ||||||
|  |       } | ||||||
|  |     }) | ||||||
|  |     render( | ||||||
|  |       <TestWrap> | ||||||
|  |         <App /> | ||||||
|  |       </TestWrap> | ||||||
|  |     ) | ||||||
|  |     const linkElement = screen.getByText(/Variables/i) | ||||||
|  |     expect(linkElement).toBeInTheDocument() | ||||||
|  |  | ||||||
|  |     vi.restoreAllMocks() | ||||||
|  |   }) | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | function TestWrap({ children }: { children: React.ReactNode }) { | ||||||
|  |   // We have to use a memory router in the testing environment, | ||||||
|  |   // and we have to use the createMemoryRouter function instead of <MemoryRouter /> as of react-router v6.4: | ||||||
|  |   // https://reactrouter.com/en/6.16.0/routers/picking-a-router#using-v64-data-apis | ||||||
|  |   const router = createMemoryRouter( | ||||||
|  |     createRoutesFromElements( | ||||||
|  |       <Route | ||||||
|  |         path="/file/:id" | ||||||
|  |         element={ | ||||||
|  |           <CommandBarProvider> | ||||||
|  |             <GlobalStateProvider> | ||||||
|  |               <ModelingMachineProvider>{children}</ModelingMachineProvider> | ||||||
|  |             </GlobalStateProvider> | ||||||
|  |           </CommandBarProvider> | ||||||
|  |         } | ||||||
|  |       /> | ||||||
|  |     ), | ||||||
|  |     { | ||||||
|  |       initialEntries: ['/file/new'], | ||||||
|  |       initialIndex: 0, | ||||||
|  |     } | ||||||
|  |   ) | ||||||
|  |   return <RouterProvider router={router} /> | ||||||
|  | } | ||||||
							
								
								
									
										84
									
								
								src/App.tsx
									
									
									
									
									
								
							
							
						
						| @ -19,24 +19,21 @@ import { | |||||||
| } from '@fortawesome/free-solid-svg-icons' | } from '@fortawesome/free-solid-svg-icons' | ||||||
| import { useHotkeys } from 'react-hotkeys-hook' | import { useHotkeys } from 'react-hotkeys-hook' | ||||||
| import { getNormalisedCoordinates } from './lib/utils' | import { getNormalisedCoordinates } from './lib/utils' | ||||||
| import { useLoaderData, useNavigate } from 'react-router-dom' | import { useLoaderData } from 'react-router-dom' | ||||||
| import { type IndexLoaderData } from 'lib/types' | import { IndexLoaderData } from './Router' | ||||||
| import { paths } from 'lib/paths' |  | ||||||
| import { useGlobalStateContext } from 'hooks/useGlobalStateContext' | import { useGlobalStateContext } from 'hooks/useGlobalStateContext' | ||||||
| import { onboardingPaths } from 'routes/Onboarding/paths' | import { onboardingPaths } from 'routes/Onboarding' | ||||||
|  | import { cameraMouseDragGuards } from 'lib/cameraControls' | ||||||
|  | import { CameraDragInteractionType_type } from '@kittycad/lib/dist/types/src/models' | ||||||
| import { CodeMenu } from 'components/CodeMenu' | import { CodeMenu } from 'components/CodeMenu' | ||||||
| import { TextEditor } from 'components/TextEditor' | import { TextEditor } from 'components/TextEditor' | ||||||
| import { Themes, getSystemTheme } from 'lib/theme' | import { Themes, getSystemTheme } from 'lib/theme' | ||||||
| import { useEngineConnectionSubscriptions } from 'hooks/useEngineConnectionSubscriptions' | import { useEngineConnectionSubscriptions } from 'hooks/useEngineConnectionSubscriptions' | ||||||
| import { engineCommandManager } from './lang/std/engineConnection' | import { engineCommandManager } from './lang/std/engineConnection' | ||||||
| import { useModelingContext } from 'hooks/useModelingContext' | import { useModelingContext } from 'hooks/useModelingContext' | ||||||
| import { useAbsoluteFilePath } from 'hooks/useAbsoluteFilePath' |  | ||||||
| import { isTauri } from 'lib/isTauri' |  | ||||||
|  |  | ||||||
| export function App() { | export function App() { | ||||||
|   const { project, file } = useLoaderData() as IndexLoaderData |   const { project, file } = useLoaderData() as IndexLoaderData | ||||||
|   const navigate = useNavigate() |  | ||||||
|   const filePath = useAbsoluteFilePath() |  | ||||||
|  |  | ||||||
|   useHotKeyListener() |   useHotKeyListener() | ||||||
|   const { |   const { | ||||||
| @ -54,7 +51,8 @@ export function App() { | |||||||
|   })) |   })) | ||||||
|  |  | ||||||
|   const { settings } = useGlobalStateContext() |   const { settings } = useGlobalStateContext() | ||||||
|   const { showDebugPanel, onboardingStatus, theme } = settings?.context || {} |   const { showDebugPanel, onboardingStatus, cameraControls, theme } = | ||||||
|  |     settings?.context || {} | ||||||
|   const { state, send } = useModelingContext() |   const { state, send } = useModelingContext() | ||||||
|  |  | ||||||
|   const editorTheme = theme === Themes.System ? getSystemTheme() : theme |   const editorTheme = theme === Themes.System ? getSystemTheme() : theme | ||||||
| @ -73,16 +71,6 @@ export function App() { | |||||||
|   useHotkeys('shift + e', () => togglePane('kclErrors')) |   useHotkeys('shift + e', () => togglePane('kclErrors')) | ||||||
|   useHotkeys('shift + d', () => togglePane('debug')) |   useHotkeys('shift + d', () => togglePane('debug')) | ||||||
|   useHotkeys('esc', () => send('Cancel')) |   useHotkeys('esc', () => send('Cancel')) | ||||||
|   useHotkeys('backspace', (e) => { |  | ||||||
|     e.preventDefault() |  | ||||||
|   }) |  | ||||||
|   useHotkeys( |  | ||||||
|     isTauri() ? 'mod + ,' : 'shift + mod + ,', |  | ||||||
|     () => navigate(filePath + paths.SETTINGS), |  | ||||||
|     { |  | ||||||
|       splitKey: '|', |  | ||||||
|     } |  | ||||||
|   ) |  | ||||||
|  |  | ||||||
|   const paneOpacity = [onboardingPaths.CAMERA, onboardingPaths.STREAMING].some( |   const paneOpacity = [onboardingPaths.CAMERA, onboardingPaths.STREAMING].some( | ||||||
|     (p) => p === onboardingStatus |     (p) => p === onboardingStatus | ||||||
| @ -96,11 +84,9 @@ export function App() { | |||||||
|  |  | ||||||
|   const debounceSocketSend = throttle<EngineCommand>((message) => { |   const debounceSocketSend = throttle<EngineCommand>((message) => { | ||||||
|     engineCommandManager.sendSceneCommand(message) |     engineCommandManager.sendSceneCommand(message) | ||||||
|   }, 1000 / 15) |   }, 16) | ||||||
|   const handleMouseMove: MouseEventHandler<HTMLDivElement> = (e) => { |   const handleMouseMove: MouseEventHandler<HTMLDivElement> = (e) => { | ||||||
|     if (state.matches('Sketch')) { |     e.nativeEvent.preventDefault() | ||||||
|       return |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     const { x, y } = getNormalisedCoordinates({ |     const { x, y } = getNormalisedCoordinates({ | ||||||
|       clientX: e.clientX, |       clientX: e.clientX, | ||||||
| @ -111,11 +97,58 @@ export function App() { | |||||||
|  |  | ||||||
|     const newCmdId = uuidv4() |     const newCmdId = uuidv4() | ||||||
|     if (buttonDownInStream === undefined) { |     if (buttonDownInStream === undefined) { | ||||||
|  |       if (state.matches('Sketch.Line Tool')) { | ||||||
|  |         debounceSocketSend({ | ||||||
|  |           type: 'modeling_cmd_req', | ||||||
|  |           cmd_id: newCmdId, | ||||||
|  |           cmd: { | ||||||
|  |             type: 'mouse_move', | ||||||
|  |             window: { x, y }, | ||||||
|  |           }, | ||||||
|  |         }) | ||||||
|  |       } else { | ||||||
|  |         debounceSocketSend({ | ||||||
|  |           type: 'modeling_cmd_req', | ||||||
|  |           cmd: { | ||||||
|  |             type: 'highlight_set_entity', | ||||||
|  |             selected_at_window: { x, y }, | ||||||
|  |           }, | ||||||
|  |           cmd_id: newCmdId, | ||||||
|  |         }) | ||||||
|  |       } | ||||||
|  |     } else { | ||||||
|  |       if (state.matches('Sketch.Move Tool')) { | ||||||
|  |         debounceSocketSend({ | ||||||
|  |           type: 'modeling_cmd_req', | ||||||
|  |           cmd_id: newCmdId, | ||||||
|  |           cmd: { | ||||||
|  |             type: 'handle_mouse_drag_move', | ||||||
|  |             window: { x, y }, | ||||||
|  |           }, | ||||||
|  |         }) | ||||||
|  |         return | ||||||
|  |       } | ||||||
|  |       const interactionGuards = cameraMouseDragGuards[cameraControls] | ||||||
|  |       let interaction: CameraDragInteractionType_type | ||||||
|  |  | ||||||
|  |       const eWithButton = { ...e, button: buttonDownInStream } | ||||||
|  |  | ||||||
|  |       if (interactionGuards.pan.callback(eWithButton)) { | ||||||
|  |         interaction = 'pan' | ||||||
|  |       } else if (interactionGuards.rotate.callback(eWithButton)) { | ||||||
|  |         interaction = 'rotate' | ||||||
|  |       } else if (interactionGuards.zoom.dragCallback(eWithButton)) { | ||||||
|  |         interaction = 'zoom' | ||||||
|  |       } else { | ||||||
|  |         return | ||||||
|  |       } | ||||||
|  |  | ||||||
|       debounceSocketSend({ |       debounceSocketSend({ | ||||||
|         type: 'modeling_cmd_req', |         type: 'modeling_cmd_req', | ||||||
|         cmd: { |         cmd: { | ||||||
|           type: 'highlight_set_entity', |           type: 'camera_drag_move', | ||||||
|           selected_at_window: { x, y }, |           interaction, | ||||||
|  |           window: { x, y }, | ||||||
|         }, |         }, | ||||||
|         cmd_id: newCmdId, |         cmd_id: newCmdId, | ||||||
|       }) |       }) | ||||||
| @ -205,7 +238,6 @@ export function App() { | |||||||
|           open={openPanes.includes('debug')} |           open={openPanes.includes('debug')} | ||||||
|         /> |         /> | ||||||
|       )} |       )} | ||||||
|       {/* <CamToggle /> */} |  | ||||||
|     </div> |     </div> | ||||||
|   ) |   ) | ||||||
| } | } | ||||||
|  | |||||||
| @ -8,7 +8,7 @@ export const Auth = ({ children }: React.PropsWithChildren) => { | |||||||
|  |  | ||||||
|   return isLoggingIn ? ( |   return isLoggingIn ? ( | ||||||
|     <Loading> |     <Loading> | ||||||
|       <span data-testid="initial-load">Loading Modeling App...</span> |       <span data-testid="initial-load">Loading KittyCAD Modeling App...</span> | ||||||
|     </Loading> |     </Loading> | ||||||
|   ) : ( |   ) : ( | ||||||
|     <>{children}</> |     <>{children}</> | ||||||
|  | |||||||
| @ -14,7 +14,10 @@ import { | |||||||
| import { useEffect } from 'react' | import { useEffect } from 'react' | ||||||
| import { ErrorPage } from './components/ErrorPage' | import { ErrorPage } from './components/ErrorPage' | ||||||
| import { Settings } from './routes/Settings' | import { Settings } from './routes/Settings' | ||||||
| import Onboarding, { onboardingRoutes } from './routes/Onboarding' | import Onboarding, { | ||||||
|  |   onboardingRoutes, | ||||||
|  |   onboardingPaths, | ||||||
|  | } from './routes/Onboarding' | ||||||
| import SignIn from './routes/SignIn' | import SignIn from './routes/SignIn' | ||||||
| import { Auth } from './Auth' | import { Auth } from './Auth' | ||||||
| import { isTauri } from './lib/isTauri' | import { isTauri } from './lib/isTauri' | ||||||
| @ -26,7 +29,7 @@ import { | |||||||
|   isProjectDirectory, |   isProjectDirectory, | ||||||
|   PROJECT_ENTRYPOINT, |   PROJECT_ENTRYPOINT, | ||||||
| } from './lib/tauriFS' | } from './lib/tauriFS' | ||||||
| import { metadata } from 'tauri-plugin-fs-extra-api' | import { metadata, type Metadata } from 'tauri-plugin-fs-extra-api' | ||||||
| import DownloadAppBanner from './components/DownloadAppBanner' | import DownloadAppBanner from './components/DownloadAppBanner' | ||||||
| import { WasmErrBanner } from './components/WasmErrBanner' | import { WasmErrBanner } from './components/WasmErrBanner' | ||||||
| import { GlobalStateProvider } from './components/GlobalStateProvider' | import { GlobalStateProvider } from './components/GlobalStateProvider' | ||||||
| @ -39,12 +42,9 @@ import CommandBarProvider from 'components/CommandBar/CommandBar' | |||||||
| import { TEST, VITE_KC_SENTRY_DSN } from './env' | import { TEST, VITE_KC_SENTRY_DSN } from './env' | ||||||
| import * as Sentry from '@sentry/react' | import * as Sentry from '@sentry/react' | ||||||
| import ModelingMachineProvider from 'components/ModelingMachineProvider' | import ModelingMachineProvider from 'components/ModelingMachineProvider' | ||||||
| import { KclContextProvider, kclManager } from 'lang/KclSingleton' | import { KclContextProvider, kclManager } from 'lang/KclSinglton' | ||||||
| import FileMachineProvider from 'components/FileMachineProvider' | import FileMachineProvider from 'components/FileMachineProvider' | ||||||
| import { sep } from '@tauri-apps/api/path' | import { sep } from '@tauri-apps/api/path' | ||||||
| import { paths } from 'lib/paths' |  | ||||||
| import { IndexLoaderData, HomeLoaderData } from 'lib/types' |  | ||||||
| import { fileSystemManager } from 'lang/std/fileSystemManager' |  | ||||||
|  |  | ||||||
| if (VITE_KC_SENTRY_DSN && !TEST) { | if (VITE_KC_SENTRY_DSN && !TEST) { | ||||||
|   Sentry.init({ |   Sentry.init({ | ||||||
| @ -78,8 +78,42 @@ if (VITE_KC_SENTRY_DSN && !TEST) { | |||||||
|   }) |   }) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | const prependRoutes = | ||||||
|  |   (routesObject: Record<string, string>) => (prepend: string) => { | ||||||
|  |     return Object.fromEntries( | ||||||
|  |       Object.entries(routesObject).map(([constName, path]) => [ | ||||||
|  |         constName, | ||||||
|  |         prepend + path, | ||||||
|  |       ]) | ||||||
|  |     ) | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | export const paths = { | ||||||
|  |   INDEX: '/', | ||||||
|  |   HOME: '/home', | ||||||
|  |   FILE: '/file', | ||||||
|  |   SETTINGS: '/settings', | ||||||
|  |   SIGN_IN: '/signin', | ||||||
|  |   ONBOARDING: prependRoutes(onboardingPaths)( | ||||||
|  |     '/onboarding' | ||||||
|  |   ) as typeof onboardingPaths, | ||||||
|  | } | ||||||
|  |  | ||||||
| export const BROWSER_FILE_NAME = 'new' | export const BROWSER_FILE_NAME = 'new' | ||||||
|  |  | ||||||
|  | export type IndexLoaderData = { | ||||||
|  |   code: string | null | ||||||
|  |   project?: ProjectWithEntryPointMetadata | ||||||
|  |   file?: FileEntry | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export type ProjectWithEntryPointMetadata = FileEntry & { | ||||||
|  |   entrypointMetadata: Metadata | ||||||
|  | } | ||||||
|  | export type HomeLoaderData = { | ||||||
|  |   projects: ProjectWithEntryPointMetadata[] | ||||||
|  | } | ||||||
|  |  | ||||||
| type CreateBrowserRouterArg = Parameters<typeof createBrowserRouter>[0] | type CreateBrowserRouterArg = Parameters<typeof createBrowserRouter>[0] | ||||||
|  |  | ||||||
| const addGlobalContextToElements = ( | const addGlobalContextToElements = ( | ||||||
| @ -111,18 +145,18 @@ const router = createBrowserRouter( | |||||||
|     { |     { | ||||||
|       path: paths.FILE + '/:id', |       path: paths.FILE + '/:id', | ||||||
|       element: ( |       element: ( | ||||||
|         <KclContextProvider> |         <Auth> | ||||||
|           <Auth> |           <FileMachineProvider> | ||||||
|             <FileMachineProvider> |             <KclContextProvider> | ||||||
|               <ModelingMachineProvider> |               <ModelingMachineProvider> | ||||||
|                 <Outlet /> |                 <Outlet /> | ||||||
|                 <App /> |                 <App /> | ||||||
|               </ModelingMachineProvider> |               </ModelingMachineProvider> | ||||||
|               <WasmErrBanner /> |               <WasmErrBanner /> | ||||||
|             </FileMachineProvider> |             </KclContextProvider> | ||||||
|             {!isTauri() && import.meta.env.PROD && <DownloadAppBanner />} |           </FileMachineProvider> | ||||||
|           </Auth> |           {!isTauri() && import.meta.env.PROD && <DownloadAppBanner />} | ||||||
|         </KclContextProvider> |         </Auth> | ||||||
|       ), |       ), | ||||||
|       id: paths.FILE, |       id: paths.FILE, | ||||||
|       loader: async ({ |       loader: async ({ | ||||||
| @ -175,10 +209,6 @@ const router = createBrowserRouter( | |||||||
|           const children = await readDir(projectPath, { recursive: true }) |           const children = await readDir(projectPath, { recursive: true }) | ||||||
|           kclManager.setCodeAndExecute(code, false) |           kclManager.setCodeAndExecute(code, false) | ||||||
|  |  | ||||||
|           // Set the file system manager to the project path |  | ||||||
|           // So that WASM gets an updated path for operations |  | ||||||
|           fileSystemManager.dir = projectPath |  | ||||||
|  |  | ||||||
|           return { |           return { | ||||||
|             code, |             code, | ||||||
|             project: { |             project: { | ||||||
| @ -218,7 +248,7 @@ const router = createBrowserRouter( | |||||||
|           <Home /> |           <Home /> | ||||||
|         </Auth> |         </Auth> | ||||||
|       ), |       ), | ||||||
|       loader: async (): Promise<HomeLoaderData | Response> => { |       loader: async () => { | ||||||
|         if (!isTauri()) { |         if (!isTauri()) { | ||||||
|           return redirect(paths.FILE + '/' + BROWSER_FILE_NAME) |           return redirect(paths.FILE + '/' + BROWSER_FILE_NAME) | ||||||
|         } |         } | ||||||
| @ -229,7 +259,6 @@ const router = createBrowserRouter( | |||||||
|         const projectDir = await initializeProjectDirectory( |         const projectDir = await initializeProjectDirectory( | ||||||
|           persistedSettings.defaultDirectory || '' |           persistedSettings.defaultDirectory || '' | ||||||
|         ) |         ) | ||||||
|         let newDefaultDirectory: string | undefined = undefined |  | ||||||
|         if (projectDir !== persistedSettings.defaultDirectory) { |         if (projectDir !== persistedSettings.defaultDirectory) { | ||||||
|           localStorage.setItem( |           localStorage.setItem( | ||||||
|             SETTINGS_PERSIST_KEY, |             SETTINGS_PERSIST_KEY, | ||||||
| @ -238,7 +267,6 @@ const router = createBrowserRouter( | |||||||
|               defaultDirectory: projectDir, |               defaultDirectory: projectDir, | ||||||
|             }) |             }) | ||||||
|           ) |           ) | ||||||
|           newDefaultDirectory = projectDir |  | ||||||
|         } |         } | ||||||
|         const projectsNoMeta = (await readDir(projectDir)).filter( |         const projectsNoMeta = (await readDir(projectDir)).filter( | ||||||
|           isProjectDirectory |           isProjectDirectory | ||||||
| @ -254,7 +282,6 @@ const router = createBrowserRouter( | |||||||
|  |  | ||||||
|         return { |         return { | ||||||
|           projects, |           projects, | ||||||
|           newDefaultDirectory, |  | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       children: [ |       children: [ | ||||||
|  | |||||||
							
								
								
									
										105
									
								
								src/Toolbar.tsx
									
									
									
									
									
								
							
							
						
						| @ -5,8 +5,6 @@ import { useModelingContext } from 'hooks/useModelingContext' | |||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' | import { useCommandsContext } from 'hooks/useCommandsContext' | ||||||
| import { ActionButton } from 'components/ActionButton' | import { ActionButton } from 'components/ActionButton' | ||||||
| import usePlatform from 'hooks/usePlatform' | import usePlatform from 'hooks/usePlatform' | ||||||
| import { isSingleCursorInPipe } from 'lang/queryAst' |  | ||||||
| import { kclManager } from 'lang/KclSingleton' |  | ||||||
|  |  | ||||||
| export const Toolbar = () => { | export const Toolbar = () => { | ||||||
|   const platform = usePlatform() |   const platform = usePlatform() | ||||||
| @ -15,15 +13,14 @@ export const Toolbar = () => { | |||||||
|   const toolbarButtonsRef = useRef<HTMLUListElement>(null) |   const toolbarButtonsRef = useRef<HTMLUListElement>(null) | ||||||
|   const bgClassName = |   const bgClassName = | ||||||
|     'group-enabled:group-hover:bg-energy-10 group-pressed:bg-energy-10 dark:group-enabled:group-hover:bg-chalkboard-80 dark:group-pressed:bg-chalkboard-80' |     'group-enabled:group-hover:bg-energy-10 group-pressed:bg-energy-10 dark:group-enabled:group-hover:bg-chalkboard-80 dark:group-pressed:bg-chalkboard-80' | ||||||
|   const pathId = useMemo(() => { |   const pathId = useMemo( | ||||||
|     if (!isSingleCursorInPipe(context.selectionRanges, kclManager.ast)) { |     () => | ||||||
|       return false |       isCursorInSketchCommandRange( | ||||||
|     } |         engineCommandManager.artifactMap, | ||||||
|     return isCursorInSketchCommandRange( |         context.selectionRanges | ||||||
|       engineCommandManager.artifactMap, |       ), | ||||||
|       context.selectionRanges |     [engineCommandManager.artifactMap, context.selectionRanges] | ||||||
|     ) |   ) | ||||||
|   }, [engineCommandManager.artifactMap, context.selectionRanges]) |  | ||||||
|  |  | ||||||
|   function handleToolbarButtonsWheelEvent(ev: WheelEvent<HTMLSpanElement>) { |   function handleToolbarButtonsWheelEvent(ev: WheelEvent<HTMLSpanElement>) { | ||||||
|     const span = toolbarButtonsRef.current |     const span = toolbarButtonsRef.current | ||||||
| @ -53,9 +50,7 @@ export const Toolbar = () => { | |||||||
|           <li className="contents"> |           <li className="contents"> | ||||||
|             <ActionButton |             <ActionButton | ||||||
|               Element="button" |               Element="button" | ||||||
|               onClick={() => |               onClick={() => send({ type: 'Enter sketch' })} | ||||||
|                 send({ type: 'Enter sketch', data: { forceNewSketch: true } }) |  | ||||||
|               } |  | ||||||
|               icon={{ |               icon={{ | ||||||
|                 icon: 'sketch', |                 icon: 'sketch', | ||||||
|                 bgClassName, |                 bgClassName, | ||||||
| @ -94,48 +89,44 @@ export const Toolbar = () => { | |||||||
|           </li> |           </li> | ||||||
|         )} |         )} | ||||||
|         {state.matches('Sketch') && !state.matches('idle') && ( |         {state.matches('Sketch') && !state.matches('idle') && ( | ||||||
|           <> |           <li className="contents"> | ||||||
|             <li className="contents" key="line-button"> |             <ActionButton | ||||||
|               <ActionButton |               Element="button" | ||||||
|                 Element="button" |               onClick={() => | ||||||
|                 onClick={() => |                 state.matches('Sketch.Line Tool') | ||||||
|                   state?.matches('Sketch.Line tool') |                   ? send('CancelSketch') | ||||||
|                     ? send('CancelSketch') |                   : send('Equip tool') | ||||||
|                     : send('Equip Line tool') |               } | ||||||
|                 } |               aria-pressed={state.matches('Sketch.Line Tool')} | ||||||
|                 aria-pressed={state?.matches('Sketch.Line tool')} |               className="pressed:bg-energy-10/20 dark:pressed:bg-energy-80" | ||||||
|                 className="pressed:bg-energy-10/20 dark:pressed:bg-energy-80" |               icon={{ | ||||||
|                 icon={{ |                 icon: 'line', | ||||||
|                   icon: 'line', |                 bgClassName, | ||||||
|                   bgClassName, |               }} | ||||||
|                 }} |             > | ||||||
|               > |               Line | ||||||
|                 Line |             </ActionButton> | ||||||
|               </ActionButton> |           </li> | ||||||
|             </li> |         )} | ||||||
|             <li className="contents" key="tangential-arc-button"> |         {state.matches('Sketch') && ( | ||||||
|               <ActionButton |           <li className="contents"> | ||||||
|                 Element="button" |             <ActionButton | ||||||
|                 onClick={() => |               Element="button" | ||||||
|                   state.matches('Sketch.Tangential arc to') |               onClick={() => | ||||||
|                     ? send('CancelSketch') |                 state.matches('Sketch.Move Tool') | ||||||
|                     : send('Equip tangential arc to') |                   ? send('CancelSketch') | ||||||
|                 } |                   : send('Equip move tool') | ||||||
|                 aria-pressed={state.matches('Sketch.Tangential arc to')} |               } | ||||||
|                 className="pressed:bg-energy-10/20 dark:pressed:bg-energy-80" |               aria-pressed={state.matches('Sketch.Move Tool')} | ||||||
|                 icon={{ |               className="pressed:bg-energy-10/20 dark:pressed:bg-energy-80" | ||||||
|                   icon: 'line', |               icon={{ | ||||||
|                   bgClassName, |                 icon: 'move', | ||||||
|                 }} |                 bgClassName, | ||||||
|                 disabled={ |               }} | ||||||
|                   !state.can('Equip tangential arc to') && |             > | ||||||
|                   !state.matches('Sketch.Tangential arc to') |               Move | ||||||
|                 } |             </ActionButton> | ||||||
|               > |           </li> | ||||||
|                 Tangential Arc |  | ||||||
|               </ActionButton> |  | ||||||
|             </li> |  | ||||||
|           </> |  | ||||||
|         )} |         )} | ||||||
|         {state.matches('Sketch.SketchIdle') && |         {state.matches('Sketch.SketchIdle') && | ||||||
|           state.nextEvents |           state.nextEvents | ||||||
| @ -160,7 +151,7 @@ export const Toolbar = () => { | |||||||
|               return 0 |               return 0 | ||||||
|             }) |             }) | ||||||
|             .map((eventName) => ( |             .map((eventName) => ( | ||||||
|               <li className="contents" key={eventName}> |               <li className="contents"> | ||||||
|                 <ActionButton |                 <ActionButton | ||||||
|                   Element="button" |                   Element="button" | ||||||
|                   className="text-sm" |                   className="text-sm" | ||||||
|  | |||||||
| @ -1,252 +0,0 @@ | |||||||
| import { useRef, useEffect, useState } from 'react' |  | ||||||
| import { useModelingContext } from 'hooks/useModelingContext' |  | ||||||
|  |  | ||||||
| import { cameraMouseDragGuards } from 'lib/cameraControls' |  | ||||||
| import { useGlobalStateContext } from 'hooks/useGlobalStateContext' |  | ||||||
| import { useStore } from 'useStore' |  | ||||||
| import { |  | ||||||
|   DEBUG_SHOW_BOTH_SCENES, |  | ||||||
|   ReactCameraProperties, |  | ||||||
|   sceneInfra, |  | ||||||
| } from './sceneInfra' |  | ||||||
| import { throttle } from 'lib/utils' |  | ||||||
|  |  | ||||||
| function useShouldHideScene(): { hideClient: boolean; hideServer: boolean } { |  | ||||||
|   const [isCamMoving, setIsCamMoving] = useState(false) |  | ||||||
|   const [isTween, setIsTween] = useState(false) |  | ||||||
|  |  | ||||||
|   const { state } = useModelingContext() |  | ||||||
|  |  | ||||||
|   useEffect(() => { |  | ||||||
|     sceneInfra.setIsCamMovingCallback((isMoving, isTween) => { |  | ||||||
|       setIsCamMoving(isMoving) |  | ||||||
|       setIsTween(isTween) |  | ||||||
|     }) |  | ||||||
|   }, []) |  | ||||||
|  |  | ||||||
|   if (DEBUG_SHOW_BOTH_SCENES || !isCamMoving) |  | ||||||
|     return { hideClient: false, hideServer: false } |  | ||||||
|   let hideServer = state.matches('Sketch') |  | ||||||
|   if (isTween) { |  | ||||||
|     hideServer = false |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   return { hideClient: !hideServer, hideServer } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export const ClientSideScene = ({ |  | ||||||
|   cameraControls, |  | ||||||
| }: { |  | ||||||
|   cameraControls: ReturnType< |  | ||||||
|     typeof useGlobalStateContext |  | ||||||
|   >['settings']['context']['cameraControls'] |  | ||||||
| }) => { |  | ||||||
|   const canvasRef = useRef<HTMLDivElement>(null) |  | ||||||
|   const { state, send } = useModelingContext() |  | ||||||
|   const { hideClient, hideServer } = useShouldHideScene() |  | ||||||
|   const { setHighlightRange } = useStore((s) => ({ |  | ||||||
|     setHighlightRange: s.setHighlightRange, |  | ||||||
|     highlightRange: s.highlightRange, |  | ||||||
|   })) |  | ||||||
|  |  | ||||||
|   // Listen for changes to the camera controls setting |  | ||||||
|   // and update the client-side scene's controls accordingly. |  | ||||||
|   useEffect(() => { |  | ||||||
|     sceneInfra.setInteractionGuards(cameraMouseDragGuards[cameraControls]) |  | ||||||
|   }, [cameraControls]) |  | ||||||
|   useEffect(() => { |  | ||||||
|     sceneInfra.updateOtherSelectionColors( |  | ||||||
|       state?.context?.selectionRanges?.otherSelections || [] |  | ||||||
|     ) |  | ||||||
|   }, [state?.context?.selectionRanges?.otherSelections]) |  | ||||||
|  |  | ||||||
|   useEffect(() => { |  | ||||||
|     if (!canvasRef.current) return |  | ||||||
|     const canvas = canvasRef.current |  | ||||||
|     canvas.appendChild(sceneInfra.renderer.domElement) |  | ||||||
|     sceneInfra.animate() |  | ||||||
|     sceneInfra.setHighlightCallback(setHighlightRange) |  | ||||||
|     canvas.addEventListener('mousemove', sceneInfra.onMouseMove, false) |  | ||||||
|     canvas.addEventListener('mousedown', sceneInfra.onMouseDown, false) |  | ||||||
|     canvas.addEventListener('mouseup', sceneInfra.onMouseUp, false) |  | ||||||
|     sceneInfra.setSend(send) |  | ||||||
|     return () => { |  | ||||||
|       canvas?.removeEventListener('mousemove', sceneInfra.onMouseMove) |  | ||||||
|       canvas?.removeEventListener('mousedown', sceneInfra.onMouseDown) |  | ||||||
|       canvas?.removeEventListener('mouseup', sceneInfra.onMouseUp) |  | ||||||
|     } |  | ||||||
|   }, []) |  | ||||||
|  |  | ||||||
|   return ( |  | ||||||
|     <div |  | ||||||
|       ref={canvasRef} |  | ||||||
|       className={`absolute inset-0 h-full w-full transition-all duration-300 ${ |  | ||||||
|         hideClient ? 'opacity-0' : 'opacity-100' |  | ||||||
|       } ${hideServer ? 'bg-black' : ''} ${ |  | ||||||
|         !hideClient && !hideServer && state.matches('Sketch') |  | ||||||
|           ? 'bg-black/80' |  | ||||||
|           : '' |  | ||||||
|       }`} |  | ||||||
|     ></div> |  | ||||||
|   ) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| const throttled = throttle((a: ReactCameraProperties) => { |  | ||||||
|   if (a.type === 'perspective' && a.fov) { |  | ||||||
|     sceneInfra.dollyZoom(a.fov) |  | ||||||
|   } |  | ||||||
| }, 1000 / 15) |  | ||||||
|  |  | ||||||
| export const CamDebugSettings = () => { |  | ||||||
|   const [camSettings, setCamSettings] = useState<ReactCameraProperties>({ |  | ||||||
|     type: 'perspective', |  | ||||||
|     fov: 12, |  | ||||||
|     position: [0, 0, 0], |  | ||||||
|     quaternion: [0, 0, 0, 1], |  | ||||||
|   }) |  | ||||||
|   const [fov, setFov] = useState(12) |  | ||||||
|  |  | ||||||
|   useEffect(() => { |  | ||||||
|     sceneInfra.setReactCameraPropertiesCallback(setCamSettings) |  | ||||||
|   }, [sceneInfra]) |  | ||||||
|   useEffect(() => { |  | ||||||
|     if (camSettings.type === 'perspective' && camSettings.fov) { |  | ||||||
|       setFov(camSettings.fov) |  | ||||||
|     } |  | ||||||
|   }, [(camSettings as any)?.fov]) |  | ||||||
|  |  | ||||||
|   return ( |  | ||||||
|     <div> |  | ||||||
|       <h3>cam settings</h3> |  | ||||||
|       perspective cam |  | ||||||
|       <input |  | ||||||
|         type="checkbox" |  | ||||||
|         checked={camSettings.type === 'perspective'} |  | ||||||
|         onChange={(e) => { |  | ||||||
|           if (camSettings.type === 'perspective') { |  | ||||||
|             sceneInfra.useOrthographicCamera() |  | ||||||
|           } else { |  | ||||||
|             sceneInfra.usePerspectiveCamera() |  | ||||||
|           } |  | ||||||
|         }} |  | ||||||
|       /> |  | ||||||
|       {camSettings.type === 'perspective' && ( |  | ||||||
|         <input |  | ||||||
|           type="range" |  | ||||||
|           min="4" |  | ||||||
|           max="90" |  | ||||||
|           step={0.5} |  | ||||||
|           value={fov} |  | ||||||
|           onChange={(e) => { |  | ||||||
|             setFov(parseFloat(e.target.value)) |  | ||||||
|  |  | ||||||
|             throttled({ |  | ||||||
|               ...camSettings, |  | ||||||
|               fov: parseFloat(e.target.value), |  | ||||||
|             }) |  | ||||||
|           }} |  | ||||||
|           className="w-full cursor-pointer pointer-events-auto" |  | ||||||
|         /> |  | ||||||
|       )} |  | ||||||
|       {camSettings.type === 'perspective' && ( |  | ||||||
|         <div> |  | ||||||
|           <span>fov</span> |  | ||||||
|           <input |  | ||||||
|             type="number" |  | ||||||
|             value={camSettings.fov} |  | ||||||
|             className="text-black w-16" |  | ||||||
|             onChange={(e) => { |  | ||||||
|               sceneInfra.setCam({ |  | ||||||
|                 ...camSettings, |  | ||||||
|                 fov: parseFloat(e.target.value), |  | ||||||
|               }) |  | ||||||
|             }} |  | ||||||
|           /> |  | ||||||
|         </div> |  | ||||||
|       )} |  | ||||||
|       {camSettings.type === 'orthographic' && ( |  | ||||||
|         <> |  | ||||||
|           <div> |  | ||||||
|             <span>fov</span> |  | ||||||
|             <input |  | ||||||
|               type="number" |  | ||||||
|               value={camSettings.zoom} |  | ||||||
|               className="text-black w-16" |  | ||||||
|               onChange={(e) => { |  | ||||||
|                 sceneInfra.setCam({ |  | ||||||
|                   ...camSettings, |  | ||||||
|                   zoom: parseFloat(e.target.value), |  | ||||||
|                 }) |  | ||||||
|               }} |  | ||||||
|             /> |  | ||||||
|           </div> |  | ||||||
|         </> |  | ||||||
|       )} |  | ||||||
|       <div> |  | ||||||
|         Position |  | ||||||
|         <ul className="flex"> |  | ||||||
|           <li> |  | ||||||
|             <span className="pl-2 pr-1">x:</span> |  | ||||||
|             <input |  | ||||||
|               type="number" |  | ||||||
|               step={5} |  | ||||||
|               data-testid="cam-x-position" |  | ||||||
|               value={camSettings.position[0]} |  | ||||||
|               className="text-black w-16" |  | ||||||
|               onChange={(e) => { |  | ||||||
|                 sceneInfra.setCam({ |  | ||||||
|                   ...camSettings, |  | ||||||
|                   position: [ |  | ||||||
|                     parseFloat(e.target.value), |  | ||||||
|                     camSettings.position[1], |  | ||||||
|                     camSettings.position[2], |  | ||||||
|                   ], |  | ||||||
|                 }) |  | ||||||
|               }} |  | ||||||
|             /> |  | ||||||
|           </li> |  | ||||||
|           <li> |  | ||||||
|             <span className="pl-2 pr-1">y:</span> |  | ||||||
|             <input |  | ||||||
|               type="number" |  | ||||||
|               step={5} |  | ||||||
|               data-testid="cam-y-position" |  | ||||||
|               value={camSettings.position[1]} |  | ||||||
|               className="text-black w-16" |  | ||||||
|               onChange={(e) => { |  | ||||||
|                 sceneInfra.setCam({ |  | ||||||
|                   ...camSettings, |  | ||||||
|                   position: [ |  | ||||||
|                     camSettings.position[0], |  | ||||||
|                     parseFloat(e.target.value), |  | ||||||
|                     camSettings.position[2], |  | ||||||
|                   ], |  | ||||||
|                 }) |  | ||||||
|               }} |  | ||||||
|             /> |  | ||||||
|           </li> |  | ||||||
|           <li> |  | ||||||
|             <span className="pl-2 pr-1">z:</span> |  | ||||||
|             <input |  | ||||||
|               type="number" |  | ||||||
|               step={5} |  | ||||||
|               data-testid="cam-z-position" |  | ||||||
|               value={camSettings.position[2]} |  | ||||||
|               className="text-black w-16" |  | ||||||
|               onChange={(e) => { |  | ||||||
|                 sceneInfra.setCam({ |  | ||||||
|                   ...camSettings, |  | ||||||
|                   position: [ |  | ||||||
|                     camSettings.position[0], |  | ||||||
|                     camSettings.position[1], |  | ||||||
|                     parseFloat(e.target.value), |  | ||||||
|                   ], |  | ||||||
|                 }) |  | ||||||
|               }} |  | ||||||
|             /> |  | ||||||
|           </li> |  | ||||||
|         </ul> |  | ||||||
|       </div> |  | ||||||
|     </div> |  | ||||||
|   ) |  | ||||||
| } |  | ||||||
| @ -1,33 +0,0 @@ | |||||||
| import { |  | ||||||
|   GridHelper, |  | ||||||
|   LineBasicMaterial, |  | ||||||
|   OrthographicCamera, |  | ||||||
|   PerspectiveCamera, |  | ||||||
|   Group, |  | ||||||
|   Mesh, |  | ||||||
| } from 'three' |  | ||||||
|  |  | ||||||
| export function createGridHelper({ |  | ||||||
|   size, |  | ||||||
|   divisions, |  | ||||||
| }: { |  | ||||||
|   size: number |  | ||||||
|   divisions: number |  | ||||||
| }) { |  | ||||||
|   const gridHelperMaterial = new LineBasicMaterial({ |  | ||||||
|     color: 0xaaaaaa, |  | ||||||
|     transparent: true, |  | ||||||
|     opacity: 0.5, |  | ||||||
|     depthTest: false, |  | ||||||
|   }) |  | ||||||
|   const gridHelper = new GridHelper(size, divisions, 0x0000ff, 0xffffff) |  | ||||||
|   gridHelper.material = gridHelperMaterial |  | ||||||
|   gridHelper.rotation.x = Math.PI / 2 |  | ||||||
|   return gridHelper |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export const orthoScale = (cam: OrthographicCamera | PerspectiveCamera) => |  | ||||||
|   0.55 / cam.zoom |  | ||||||
|  |  | ||||||
| export const perspScale = (cam: PerspectiveCamera, group: Group | Mesh) => |  | ||||||
|   (group.position.distanceTo(cam.position) * cam.fov) / 4000 |  | ||||||
| @ -1,28 +0,0 @@ | |||||||
| import { Quaternion } from 'three' |  | ||||||
| import { isQuaternionVertical } from './sceneInfra' |  | ||||||
|  |  | ||||||
| describe('isQuaternionVertical', () => { |  | ||||||
|   it('should identify vertical quaternions', () => { |  | ||||||
|     const verticalQuaternions = [ |  | ||||||
|       new Quaternion(1, 0, 0, 0).normalize(), // bottom |  | ||||||
|       new Quaternion(-0.7, 0.7, 0, 0).normalize(), // bottom 2 |  | ||||||
|       new Quaternion(0, 1, 0, 0).normalize(), // bottom 3 |  | ||||||
|       new Quaternion(0, 0, 0, 1).normalize(), // look from top |  | ||||||
|     ] |  | ||||||
|     verticalQuaternions.forEach((quaternion) => { |  | ||||||
|       expect(isQuaternionVertical(quaternion)).toBe(true) |  | ||||||
|     }) |  | ||||||
|   }) |  | ||||||
|  |  | ||||||
|   it('should identify non-vertical quaternions', () => { |  | ||||||
|     const nonVerticalQuaternions = [ |  | ||||||
|       new Quaternion(0.7, 0, 0, 0.7).normalize(), // front |  | ||||||
|       new Quaternion(0, 0.7, 0.7, 0).normalize(), // back |  | ||||||
|       new Quaternion(-0.5, 0.5, 0.5, -0.5).normalize(), // left side |  | ||||||
|       new Quaternion(0.5, 0.5, 0.5, 0.5).normalize(), // right side |  | ||||||
|     ] |  | ||||||
|     nonVerticalQuaternions.forEach((quaternion) => { |  | ||||||
|       expect(isQuaternionVertical(quaternion)).toBe(false) |  | ||||||
|     }) |  | ||||||
|   }) |  | ||||||
| }) |  | ||||||
