Compare commits
	
		
			35 Commits
		
	
	
		
			achalmers/
			...
			pierremtb/
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 7c935741e4 | |||
| 87e299e0bb | |||
| 465e71c12f | |||
| df86c93a04 | |||
| 824669a1c2 | |||
| ba8f8a1722 | |||
| f4a4e6c5be | |||
| 0d148e80aa | |||
| 3300993ac8 | |||
| 033eaed32e | |||
| 8aabac0be7 | |||
| 138728a95d | |||
| 9a92e7d642 | |||
| efedc8de58 | |||
| f7ee248a26 | |||
| 336f4f27ba | |||
| e1f128d64a | |||
| f858a611f1 | |||
| 6ce270c0d0 | |||
| 30ac0e4f48 | |||
| 8f90c352fe | |||
| bc6f0fceca | |||
| 5c830a4ed4 | |||
| 8a9d50226f | |||
| 8397405998 | |||
| f5b8298735 | |||
| 25ad603502 | |||
| 86349375d0 | |||
| 56d861f2cc | |||
| 3e8ee3ffc4 | |||
| a44516bc7e | |||
| ce62fe67cf | |||
| 763a1b6628 | |||
| 3281e62e6b | |||
| f1a458f124 | 
| @ -2,8 +2,8 @@ NODE_ENV=development | ||||
| DEV=true | ||||
| VITE_KC_API_WS_MODELING_URL=wss://api.dev.zoo.dev/ws/modeling/commands | ||||
| VITE_KC_API_BASE_URL=https://api.dev.zoo.dev | ||||
| BASE_URL=https://api.dev.zoo.dev | ||||
| VITE_KC_SITE_BASE_URL=https://dev.zoo.dev | ||||
| VITE_KC_SITE_APP_URL=https://app.dev.zoo.dev | ||||
| VITE_KC_SKIP_AUTH=false | ||||
| VITE_KC_CONNECTION_TIMEOUT_MS=5000 | ||||
| # ONLY add your token in .env.development.local if you want to skip auth, otherwise this token takes precedence! | ||||
|  | ||||
| @ -1,5 +1,8 @@ | ||||
| NODE_ENV=production | ||||
| DEV=false | ||||
| VITE_KC_API_WS_MODELING_URL=wss://api.zoo.dev/ws/modeling/commands | ||||
| VITE_KC_API_BASE_URL=https://api.zoo.dev | ||||
| VITE_KC_SITE_BASE_URL=https://zoo.dev | ||||
| VITE_KC_SITE_APP_URL=https://app.zoo.dev | ||||
| VITE_KC_SKIP_AUTH=false | ||||
| VITE_KC_CONNECTION_TIMEOUT_MS=15000 | ||||
|  | ||||
| @ -29,6 +29,13 @@ | ||||
|         { | ||||
|           "name": "isNaN", | ||||
|           "message": "Use Number.isNaN() instead." | ||||
|         }, | ||||
|       ], | ||||
|       "no-restricted-syntax": [ | ||||
|         "error", | ||||
|         { | ||||
|           "selector": "CallExpression[callee.object.name='Array'][callee.property.name='isArray']", | ||||
|           "message": "Use isArray() in lib/utils.ts instead of Array.isArray()." | ||||
|         } | ||||
|       ], | ||||
|       "semi": [ | ||||
|  | ||||
							
								
								
									
										16
									
								
								.github/workflows/build-apps.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										16
									
								
								.github/workflows/build-apps.yml
									
									
									
									
										vendored
									
									
								
							| @ -134,8 +134,6 @@ jobs: | ||||
|           max_attempts: 3 | ||||
|           command: yarn install | ||||
|  | ||||
|       - run: yarn tronb:vite | ||||
|  | ||||
|       - name: Prepare certificate and variables (Windows only) | ||||
|         if: ${{ (env.IS_RELEASE == 'true' || env.IS_NIGHTLY == 'true') && matrix.os == 'windows-2022' }} | ||||
|         run: | | ||||
| @ -165,8 +163,8 @@ jobs: | ||||
|       - name: Build the app (debug) | ||||
|         if: ${{ env.IS_RELEASE == 'false' && env.IS_NIGHTLY == 'false' }} | ||||
|         # electron-builder doesn't have a concept of release vs debug, | ||||
|         # this is just not doing any codesign or release yml generation | ||||
|         run: yarn electron-builder --config | ||||
|         # this is just not doing any codesign or release yml generation, and points to dev infra | ||||
|         run: yarn tronb:package:dev | ||||
|  | ||||
|       - name: Build the app (release) | ||||
|         if: ${{ env.IS_RELEASE == 'true' || env.IS_NIGHTLY == 'true' }} | ||||
| @ -185,7 +183,7 @@ jobs: | ||||
|         with: | ||||
|           timeout_minutes: 10 | ||||
|           max_attempts: 3 | ||||
|           command: yarn electron-builder --config --publish always | ||||
|           command: yarn tronb:package:prod | ||||
|  | ||||
|       - name: List artifacts in out/ | ||||
|         run: ls -R out | ||||
| @ -246,7 +244,7 @@ jobs: | ||||
|         with: | ||||
|           timeout_minutes: 10 | ||||
|           max_attempts: 3 | ||||
|           command: yarn electron-builder --config --publish always | ||||
|           command: yarn tronb:package:prod | ||||
|  | ||||
|       - uses: actions/upload-artifact@v4 | ||||
|         if: ${{ env.IS_RELEASE == 'true' }} | ||||
| @ -390,19 +388,19 @@ jobs: | ||||
|  | ||||
|       - name: Authenticate to Google Cloud | ||||
|         if: ${{ env.IS_NIGHTLY == 'true' }} | ||||
|         uses: 'google-github-actions/auth@v2.1.7' | ||||
|         uses: 'google-github-actions/auth@v2.1.8' | ||||
|         with: | ||||
|           credentials_json: '${{ secrets.GOOGLE_CLOUD_DL_SA }}' | ||||
|  | ||||
|       - name: Set up Google Cloud SDK | ||||
|         if: ${{ env.IS_NIGHTLY == 'true' }} | ||||
|         uses: google-github-actions/setup-gcloud@v2.1.2 | ||||
|         uses: google-github-actions/setup-gcloud@v2.1.4 | ||||
|         with: | ||||
|           project_id: ${{ env.GOOGLE_CLOUD_PROJECT_ID }} | ||||
|  | ||||
|       - name: Upload nightly files to public bucket | ||||
|         if: ${{ env.IS_NIGHTLY == 'true' }} | ||||
|         uses: google-github-actions/upload-cloud-storage@v2.2.1 | ||||
|         uses: google-github-actions/upload-cloud-storage@v2.2.2 | ||||
|         with: | ||||
|           path: out | ||||
|           glob: '*' | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| name: E2E Tests | ||||
| name: E2E Flow Tests | ||||
| on: | ||||
|   push: | ||||
|     branches: [ main ] | ||||
| @ -33,7 +33,7 @@ jobs: | ||||
|             rust: | ||||
|               - 'src/wasm-lib/**' | ||||
| 
 | ||||
|   electron: | ||||
|   flow-tests: | ||||
|     timeout-minutes: 60 | ||||
|     name: playwright:electron:${{ matrix.os }} ${{ matrix.shardIndex }} ${{ matrix.shardTotal }} | ||||
|     strategy: | ||||
| @ -47,23 +47,29 @@ jobs: | ||||
|     needs: check-rust-changes | ||||
|     steps: | ||||
|     - uses: actions/checkout@v4 | ||||
| 
 | ||||
|     - uses: actions/setup-node@v4 | ||||
|       with: | ||||
|         node-version-file: '.nvmrc' | ||||
|         cache: 'yarn' | ||||
| 
 | ||||
|     - uses: KittyCAD/action-install-cli@main | ||||
| 
 | ||||
|     - name: Install dependencies | ||||
|       shell: bash | ||||
|       run: yarn | ||||
| 
 | ||||
|     - name: Cache Playwright Browsers | ||||
|       uses: actions/cache@v4 | ||||
|       with: | ||||
|         path: | | ||||
|           ~/.cache/ms-playwright/ | ||||
|         key: ${{ runner.os }}-playwright-${{ hashFiles('yarn.lock') }} | ||||
| 
 | ||||
|     - name: Install Playwright Browsers | ||||
|       shell: bash | ||||
|       run: yarn playwright install --with-deps | ||||
| 
 | ||||
|     - name: Download Wasm Cache | ||||
|       id: download-wasm | ||||
|       if: needs.check-rust-changes.outputs.rust-changed == 'false' | ||||
| @ -75,29 +81,35 @@ jobs: | ||||
|         workflow: build-and-store-wasm.yml | ||||
|         branch: main | ||||
|         path: src/wasm-lib/pkg | ||||
| 
 | ||||
|     - name: copy wasm blob | ||||
|       if: needs.check-rust-changes.outputs.rust-changed == 'false' | ||||
|       shell: bash | ||||
|       run: cp src/wasm-lib/pkg/wasm_lib_bg.wasm public | ||||
|       continue-on-error: true | ||||
| 
 | ||||
|     - name: Setup Rust | ||||
|       uses: dtolnay/rust-toolchain@stable | ||||
| 
 | ||||
|     - name: Cache Wasm (because rust diff) | ||||
|       if: needs.check-rust-changes.outputs.rust-changed == 'true' | ||||
|       uses: Swatinem/rust-cache@v2 | ||||
|       with: | ||||
|         workspaces: './src/wasm-lib' | ||||
| 
 | ||||
|     - name: OR Cache Wasm (because wasm cache failed) | ||||
|       if: steps.download-wasm.outcome == 'failure' | ||||
|       uses: Swatinem/rust-cache@v2 | ||||
|       with: | ||||
|         workspaces: './src/wasm-lib' | ||||
| 
 | ||||
|     - name: install good sed | ||||
|       if:  ${{ startsWith(matrix.os, 'macos') }} | ||||
|       shell: bash | ||||
|       run: | | ||||
|         brew install gnu-sed | ||||
|         echo "/opt/homebrew/opt/gnu-sed/libexec/gnubin" >> $GITHUB_PATH | ||||
| 
 | ||||
|     - name: Install vector | ||||
|       shell: bash | ||||
|       # TODO: figure out what to do with this, it's failing | ||||
| @ -115,81 +127,33 @@ jobs: | ||||
|         sed -i "s#GH_ACTIONS_AXIOM_TOKEN#${{secrets.GH_ACTIONS_AXIOM_TOKEN}}#g" /tmp/vector.toml | ||||
|         cat /tmp/vector.toml | ||||
|         ${HOME}/.vector/bin/vector --config /tmp/vector.toml & | ||||
| 
 | ||||
|     - name: Build Wasm (because rust diff) | ||||
|       if: needs.check-rust-changes.outputs.rust-changed == 'true' | ||||
|       shell: bash | ||||
|       run: yarn build:wasm | ||||
| 
 | ||||
|     - name: OR Build Wasm (because wasm cache failed) | ||||
|       if: steps.download-wasm.outcome == 'failure' | ||||
|       shell: bash | ||||
|       run: yarn build:wasm | ||||
|     - name: build electron | ||||
| 
 | ||||
|     - name: build web | ||||
|       shell: bash | ||||
|       run: yarn tron:package | ||||
|     # - name: Run ubuntu/chrome snapshots | ||||
|     #   if: ${{ matrix.os == 'namespace-profile-ubuntu-8-cores' && matrix.shardIndex == 1 }} | ||||
|     #   shell: bash | ||||
|     #   # TODO: break this in its own job, for now it's not slowing down the overall execution as ubuntu is the quickest, | ||||
|     #   # but we could do better. This forces a large 1/1 shard of all 20 snapshot tests that runs in about 3 minutes. | ||||
|     #   run: | | ||||
|     #     PLATFORM=web yarn playwright test --config=playwright.config.ts --retries="3" --update-snapshots --grep=@snapshot  --shard=1/1 | ||||
|     #   env: | ||||
|     #     CI: true | ||||
|     #     NODE_ENV: development | ||||
|     #     VITE_KC_DEV_TOKEN: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||
|     #     VITE_KC_SKIP_AUTH: true | ||||
|     #     token: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||
|     #     snapshottoken: ${{ secrets.KITTYCAD_API_TOKEN }} | ||||
|     - uses: actions/upload-artifact@v4 | ||||
|       if: ${{ !cancelled() && (success() || failure()) }} | ||||
|       with: | ||||
|         name: playwright-report-${{ matrix.os }}-snapshot-${{ matrix.shardIndex }}-${{ github.sha }} | ||||
|         path: playwright-report/ | ||||
|         include-hidden-files: true | ||||
|         retention-days: 30 | ||||
|         overwrite: true | ||||
|       run: yarn tronb:vite:dev | ||||
| 
 | ||||
|     - name: Clean up test-results | ||||
|       if: ${{ !cancelled() && (success() || failure()) }} | ||||
|       continue-on-error: true | ||||
|       run: rm -r test-results | ||||
|     - name: check for changes | ||||
|       if: ${{ matrix.os == 'namespace-profile-ubuntu-8-cores' && matrix.shardIndex == 1 }} | ||||
|       shell: bash | ||||
|       id: git-check | ||||
|       run: | | ||||
|           git add . | ||||
|           if git status | grep -q "Changes to be committed" | ||||
|           then echo "modified=true" >> $GITHUB_OUTPUT | ||||
|           else echo "modified=false" >> $GITHUB_OUTPUT | ||||
|           fi | ||||
|     # - name: Commit changes, if any | ||||
|     #   if: steps.git-check.outputs.modified == 'true' | ||||
|     #   shell: bash | ||||
|     #   run: | | ||||
|     #     git add . | ||||
|     #     git config --local user.email "github-actions[bot]@users.noreply.github.com" | ||||
|     #     git config --local user.name "github-actions[bot]" | ||||
|     #     git remote set-url origin https://${{ github.actor }}:${{ secrets.GITHUB_TOKEN }}@github.com/${{ github.repository }}.git | ||||
|     #     git fetch origin | ||||
|     #     echo ${{ github.head_ref }} | ||||
|     #     git checkout ${{ github.head_ref }} | ||||
|     #     git commit -am "A snapshot a day keeps the bugs away! 📷🐛 (OS: ${{matrix.os}})" || true | ||||
|     #     git push | ||||
|     #     git push origin ${{ github.head_ref }} | ||||
|     # only upload artifacts if there's actually changes | ||||
|     - uses: actions/upload-artifact@v4 | ||||
|       if: steps.git-check.outputs.modified == 'true' | ||||
|       with: | ||||
|         name: playwright-report-${{ matrix.os }}-${{ matrix.shardIndex }}-${{ github.sha }} | ||||
|         path: playwright-report/ | ||||
|         include-hidden-files: true | ||||
|         retention-days: 30 | ||||
| 
 | ||||
|     - uses: actions/download-artifact@v4 | ||||
|       if: ${{ !cancelled() && (success() || failure()) }} | ||||
|       continue-on-error: true | ||||
|       with: | ||||
|         name: test-results-${{ matrix.os }}-${{ matrix.shardIndex }}-${{ github.sha }} | ||||
|         path: test-results/ | ||||
| 
 | ||||
|     - name: Run playwright/electron flow (with retries) | ||||
|       id: retry | ||||
|       if: ${{ !cancelled() && (success() || failure()) }} | ||||
| @ -203,6 +167,7 @@ jobs: | ||||
|         VITE_KC_DEV_TOKEN: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||
|         VITE_KC_SKIP_AUTH: true | ||||
|         token: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||
| 
 | ||||
|     - uses: actions/upload-artifact@v4 | ||||
|       if: always() | ||||
|       with: | ||||
| @ -211,6 +176,7 @@ jobs: | ||||
|         include-hidden-files: true | ||||
|         retention-days: 30 | ||||
|         overwrite: true | ||||
| 
 | ||||
|     - uses: actions/upload-artifact@v4 | ||||
|       if: always() | ||||
|       with: | ||||
| @ -219,4 +185,3 @@ jobs: | ||||
|         include-hidden-files: true | ||||
|         retention-days: 30 | ||||
|         overwrite: true | ||||
| 
 | ||||
							
								
								
									
										145
									
								
								.github/workflows/e2e-snapshot-tests.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										145
									
								
								.github/workflows/e2e-snapshot-tests.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,145 @@ | ||||
| name: E2E Snapshot Tests | ||||
| on: | ||||
|   push: | ||||
|     branches: [ main ] | ||||
|   pull_request: | ||||
|     branches: [ main ] | ||||
|  | ||||
| concurrency: | ||||
|   group: ${{ github.workflow }}-${{ github.head_ref || github.run_id }} | ||||
|   cancel-in-progress: true | ||||
|  | ||||
| permissions: | ||||
|   contents: write | ||||
|   pull-requests: write | ||||
|   actions: read | ||||
|  | ||||
|  | ||||
| jobs: | ||||
|  | ||||
|   check-rust-changes: | ||||
|     runs-on: ubuntu-latest | ||||
|     outputs: | ||||
|       rust-changed: ${{ steps.filter.outputs.rust }} | ||||
|  | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|  | ||||
|       - id: filter | ||||
|         name: Check for Rust changes | ||||
|         uses: dorny/paths-filter@v3 | ||||
|         with: | ||||
|           filters: | | ||||
|             rust: | ||||
|               - 'src/wasm-lib/**' | ||||
|  | ||||
|   snapshot-tests: | ||||
|     runs-on: ubuntu-22.04 | ||||
|     needs: check-rust-changes | ||||
|     steps: | ||||
|     - uses: actions/checkout@v4 | ||||
|  | ||||
|     - uses: actions/setup-node@v4 | ||||
|       with: | ||||
|         node-version-file: '.nvmrc' | ||||
|         cache: 'yarn' | ||||
|  | ||||
|     - name: Install dependencies | ||||
|       run: yarn | ||||
|  | ||||
|     - name: Cache Playwright Browsers | ||||
|       uses: actions/cache@v4 | ||||
|       with: | ||||
|         path: | | ||||
|           ~/.cache/ms-playwright/ | ||||
|         key: ${{ runner.os }}-playwright-${{ hashFiles('yarn.lock') }} | ||||
|  | ||||
|     - name: Install Playwright Browsers | ||||
|       run: yarn playwright install --with-deps | ||||
|  | ||||
|     - name: Download Wasm Cache | ||||
|       id: download-wasm | ||||
|       if: needs.check-rust-changes.outputs.rust-changed == 'false' | ||||
|       uses: dawidd6/action-download-artifact@v7 | ||||
|       continue-on-error: true | ||||
|       with: | ||||
|         github_token: ${{secrets.GITHUB_TOKEN}} | ||||
|         name: wasm-bundle | ||||
|         workflow: build-and-store-wasm.yml | ||||
|         branch: main | ||||
|         path: src/wasm-lib/pkg | ||||
|  | ||||
|     - name: copy wasm blob | ||||
|       if: needs.check-rust-changes.outputs.rust-changed == 'false' | ||||
|       run: cp src/wasm-lib/pkg/wasm_lib_bg.wasm public | ||||
|       continue-on-error: true | ||||
|  | ||||
|     - name: Setup Rust | ||||
|       if: needs.check-rust-changes.outputs.rust-changed == 'true' | ||||
|       uses: dtolnay/rust-toolchain@stable | ||||
|  | ||||
|     - name: Cache Wasm (because rust diff) | ||||
|       if: needs.check-rust-changes.outputs.rust-changed == 'true' | ||||
|       uses: Swatinem/rust-cache@v2 | ||||
|       with: | ||||
|         workspaces: './src/wasm-lib' | ||||
|  | ||||
|     - name: OR Cache Wasm (because wasm cache failed) | ||||
|       if: steps.download-wasm.outcome == 'failure' | ||||
|       uses: Swatinem/rust-cache@v2 | ||||
|       with: | ||||
|         workspaces: './src/wasm-lib' | ||||
|  | ||||
|     - name: Build Wasm (because rust diff) | ||||
|       if: needs.check-rust-changes.outputs.rust-changed == 'true' | ||||
|       run: yarn build:wasm | ||||
|  | ||||
|     - name: OR Build Wasm (because wasm cache failed) | ||||
|       if: steps.download-wasm.outcome == 'failure' | ||||
|       run: yarn build:wasm | ||||
|  | ||||
|     - name: build web | ||||
|       run: yarn tronb:vite:dev | ||||
|  | ||||
|     - name: Run chrome snapshots | ||||
|       run: | | ||||
|         PLATFORM=web yarn playwright test --config=playwright.config.ts --retries="3" --update-snapshots --grep=@snapshot | ||||
|       env: | ||||
|         CI: true | ||||
|         NODE_ENV: development | ||||
|         VITE_KC_DEV_TOKEN: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||
|         VITE_KC_SKIP_AUTH: true | ||||
|         token: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||
|         snapshottoken: ${{ secrets.KITTYCAD_API_TOKEN }} | ||||
|  | ||||
|     - name: check for changes | ||||
|       id: git-check | ||||
|       run: | | ||||
|         { | ||||
|           echo 'changes<<EOF' | ||||
|           git diff --name-only e2e/playwright/snapshot-tests.spec.ts-snapshots | ||||
|           echo EOF | ||||
|         } >> "$GITHUB_OUTPUT" | ||||
|  | ||||
|     # only upload artifacts if there's actually changes | ||||
|     - name: Upload changes, if any | ||||
|       if: steps.git-check.outputs.changes != '' | ||||
|       uses: actions/upload-artifact@v4 | ||||
|       with: | ||||
|         name: playwright-snapshots-${{ runner.os }}-${{ github.sha }} | ||||
|         path: ${{ steps.git-check.outputs.changes }} | ||||
|  | ||||
|     - name: Upload report, if any | ||||
|       uses: actions/upload-artifact@v4 | ||||
|       if: steps.git-check.outputs.changes != '' | ||||
|       with: | ||||
|         name: playwright-report-${{ runner.os }}-${{ github.sha }} | ||||
|         path: playwright-report/ | ||||
|         include-hidden-files: true | ||||
|         retention-days: 30 | ||||
|  | ||||
|     - name: Fail the run if we have snapshot updates | ||||
|       if: steps.git-check.outputs.changes != '' | ||||
|       run: exit 1 | ||||
|  | ||||
|     # TODO: check if we could comment on the PR as well | ||||
							
								
								
									
										6
									
								
								.github/workflows/publish-apps-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/publish-apps-release.yml
									
									
									
									
										vendored
									
									
								
							| @ -108,17 +108,17 @@ jobs: | ||||
|         run: yarn files:set-notes | ||||
|  | ||||
|       - name: Authenticate to Google Cloud | ||||
|         uses: 'google-github-actions/auth@v2.1.7' | ||||
|         uses: 'google-github-actions/auth@v2.1.8' | ||||
|         with: | ||||
|           credentials_json: '${{ secrets.GOOGLE_CLOUD_DL_SA }}' | ||||
|  | ||||
|       - name: Set up Google Cloud SDK | ||||
|         uses: google-github-actions/setup-gcloud@v2.1.2 | ||||
|         uses: google-github-actions/setup-gcloud@v2.1.4 | ||||
|         with: | ||||
|           project_id: ${{ env.GOOGLE_CLOUD_PROJECT_ID }} | ||||
|  | ||||
|       - name: Upload release files to public bucket | ||||
|         uses: google-github-actions/upload-cloud-storage@v2.2.1 | ||||
|         uses: google-github-actions/upload-cloud-storage@v2.2.2 | ||||
|         with: | ||||
|           path: out | ||||
|           glob: '*' | ||||
|  | ||||
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -25,6 +25,7 @@ yarn-error.log* | ||||
|  | ||||
| .idea | ||||
| .vscode | ||||
| .helix | ||||
| src/wasm-lib/.idea | ||||
| src/wasm-lib/.vscode | ||||
|  | ||||
|  | ||||
							
								
								
									
										2
									
								
								.helix/config.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								.helix/config.toml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,2 @@ | ||||
| [editor] | ||||
| auto-format = true | ||||
							
								
								
									
										10
									
								
								.helix/languages.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								.helix/languages.toml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | ||||
| [language-server.eslint] | ||||
| args = ["--stdio"] | ||||
| command = "vscode-eslint-language-server" | ||||
|  | ||||
| [[language]] | ||||
| name = "typescript" | ||||
| auto-format = true | ||||
| formatter = { command = "node_modules/.bin/prettier", args = ["--parser", "typescript"] } | ||||
| language-servers = [ { name = "eslint", only-features = [ "diagnostics" ] }, "typescript-language-server" ] | ||||
|  | ||||
| @ -101,7 +101,7 @@ This will start the application and hot-reload on changes. | ||||
|  | ||||
| Devtools can be opened with the usual Cmd-Opt-I (Mac) or Ctrl-Shift-I (Linux and Windows). | ||||
|  | ||||
| To build, run `yarn tron:package`. | ||||
| To build with electron-builder, run `yarn tronb:package:dev` (or `yarn tronb:package:prod` to point to the .env.production variables) | ||||
|  | ||||
| ## Checking out commits / Bisecting | ||||
|  | ||||
|  | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -31,12 +31,12 @@ appearance(data: AppearanceData, solid_set: SolidSet) -> SolidSet | ||||
| // Add color to an extruded solid. | ||||
| exampleSketch = startSketchOn("XZ") | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> lineTo([10, 0], %) | ||||
|   |> lineTo([0, 10], %) | ||||
|   |> lineTo([-10, 0], %) | ||||
|   |> close(%) | ||||
|   |> line(endAbsolute = [10, 0]) | ||||
|   |> line(endAbsolute = [0, 10]) | ||||
|   |> line(endAbsolute = [-10, 0]) | ||||
|   |> close() | ||||
|  | ||||
| example = extrude(5, exampleSketch) | ||||
| example = extrude(exampleSketch, length = 5) | ||||
|   |> appearance({ | ||||
|        color = '#ff0000', | ||||
|        metalness = 50, | ||||
| @ -65,11 +65,11 @@ sketch001 = startSketchOn('XY') | ||||
| fn cube(center) { | ||||
|   return startSketchOn('XY') | ||||
|     |> startProfileAt([center[0] - 10, center[1] - 10], %) | ||||
|     |> lineTo([center[0] + 10, center[1] - 10], %) | ||||
|     |> lineTo([center[0] + 10, center[1] + 10], %) | ||||
|     |> lineTo([center[0] - 10, center[1] + 10], %) | ||||
|     |> close(%) | ||||
|     |> extrude(10, %) | ||||
|     |> line(endAbsolute = [center[0] + 10, center[1] - 10]) | ||||
|     |> line(endAbsolute = [center[0] + 10, center[1] + 10]) | ||||
|     |> line(endAbsolute = [center[0] - 10, center[1] + 10]) | ||||
|     |> close() | ||||
|     |> extrude(length = 10) | ||||
| } | ||||
|  | ||||
| example0 = cube([0, 0]) | ||||
| @ -95,11 +95,11 @@ appearance({ | ||||
| // This example shows setting the appearance _after_ the shell. | ||||
| firstSketch = startSketchOn('XY') | ||||
|   |> startProfileAt([-12, 12], %) | ||||
|   |> line([24, 0], %) | ||||
|   |> line([0, -24], %) | ||||
|   |> line([-24, 0], %) | ||||
|   |> close(%) | ||||
|   |> extrude(6, %) | ||||
|   |> line(end = [24, 0]) | ||||
|   |> line(end = [0, -24]) | ||||
|   |> line(end = [-24, 0]) | ||||
|   |> close() | ||||
|   |> extrude(length = 6) | ||||
|  | ||||
| shell({ faces = ['end'], thickness = 0.25 }, firstSketch) | ||||
|   |> appearance({ | ||||
| @ -116,11 +116,11 @@ shell({ faces = ['end'], thickness = 0.25 }, firstSketch) | ||||
| // This example shows setting the appearance _before_ the shell. | ||||
| firstSketch = startSketchOn('XY') | ||||
|   |> startProfileAt([-12, 12], %) | ||||
|   |> line([24, 0], %) | ||||
|   |> line([0, -24], %) | ||||
|   |> line([-24, 0], %) | ||||
|   |> close(%) | ||||
|   |> extrude(6, %) | ||||
|   |> line(end = [24, 0]) | ||||
|   |> line(end = [0, -24]) | ||||
|   |> line(end = [-24, 0]) | ||||
|   |> close() | ||||
|   |> extrude(length = 6) | ||||
|   |> appearance({ | ||||
|        color = '#ff0000', | ||||
|        metalness = 90, | ||||
| @ -137,12 +137,12 @@ shell({ faces = ['end'], thickness = 0.25 }, firstSketch) | ||||
| // This example shows _before_ the pattern. | ||||
| exampleSketch = startSketchOn('XZ') | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line([0, 2], %) | ||||
|   |> line([3, 1], %) | ||||
|   |> line([0, -4], %) | ||||
|   |> close(%) | ||||
|   |> line(end = [0, 2]) | ||||
|   |> line(end = [3, 1]) | ||||
|   |> line(end = [0, -4]) | ||||
|   |> close() | ||||
|  | ||||
| example = extrude(1, exampleSketch) | ||||
| example = extrude(exampleSketch, length = 1) | ||||
|   |> appearance({ | ||||
|        color = '#ff0000', | ||||
|        metalness = 90, | ||||
| @ -162,12 +162,12 @@ example = extrude(1, exampleSketch) | ||||
| // This example shows _after_ the pattern. | ||||
| exampleSketch = startSketchOn('XZ') | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line([0, 2], %) | ||||
|   |> line([3, 1], %) | ||||
|   |> line([0, -4], %) | ||||
|   |> close(%) | ||||
|   |> line(end = [0, 2]) | ||||
|   |> line(end = [3, 1]) | ||||
|   |> line(end = [0, -4]) | ||||
|   |> close() | ||||
|  | ||||
| example = extrude(1, exampleSketch) | ||||
| example = extrude(exampleSketch, length = 1) | ||||
|   |> patternLinear3d({ | ||||
|        axis = [1, 0, 1], | ||||
|        instances = 7, | ||||
| @ -186,10 +186,10 @@ example = extrude(1, exampleSketch) | ||||
| // Color the result of a 2D pattern that was extruded. | ||||
| exampleSketch = startSketchOn('XZ') | ||||
|   |> startProfileAt([.5, 25], %) | ||||
|   |> line([0, 5], %) | ||||
|   |> line([-1, 0], %) | ||||
|   |> line([0, -5], %) | ||||
|   |> close(%) | ||||
|   |> line(end = [0, 5]) | ||||
|   |> line(end = [-1, 0]) | ||||
|   |> line(end = [0, -5]) | ||||
|   |> close() | ||||
|   |> patternCircular2d({ | ||||
|        center = [0, 0], | ||||
|        instances = 13, | ||||
| @ -197,7 +197,7 @@ exampleSketch = startSketchOn('XZ') | ||||
|        rotateDuplicates = true | ||||
|      }, %) | ||||
|  | ||||
| example = extrude(1, exampleSketch) | ||||
| example = extrude(exampleSketch, length = 1) | ||||
|   |> appearance({ | ||||
|        color = '#ff0000', | ||||
|        metalness = 90, | ||||
| @ -214,11 +214,11 @@ example = extrude(1, exampleSketch) | ||||
| // Create a path for the sweep. | ||||
| sweepPath = startSketchOn('XZ') | ||||
|   |> startProfileAt([0.05, 0.05], %) | ||||
|   |> line([0, 7], %) | ||||
|   |> line(end = [0, 7]) | ||||
|   |> tangentialArc({ offset = 90, radius = 5 }, %) | ||||
|   |> line([-3, 0], %) | ||||
|   |> line(end = [-3, 0]) | ||||
|   |> tangentialArc({ offset = -90, radius = 5 }, %) | ||||
|   |> line([0, 7], %) | ||||
|   |> line(end = [0, 7]) | ||||
|  | ||||
| pipeHole = startSketchOn('XY') | ||||
|   |> circle({ center = [0, 0], radius = 1.5 }, %) | ||||
|  | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -36,12 +36,12 @@ filletRadius = 2 | ||||
|  | ||||
| mountingPlateSketch = startSketchOn("XY") | ||||
|   |> startProfileAt([-width / 2, -length / 2], %) | ||||
|   |> lineTo([width / 2, -length / 2], %, $edge1) | ||||
|   |> lineTo([width / 2, length / 2], %, $edge2) | ||||
|   |> lineTo([-width / 2, length / 2], %, $edge3) | ||||
|   |> close(%, $edge4) | ||||
|   |> line(endAbsolute = [width / 2, -length / 2], tag = $edge1) | ||||
|   |> line(endAbsolute = [width / 2, length / 2], tag = $edge2) | ||||
|   |> line(endAbsolute = [-width / 2, length / 2], tag = $edge3) | ||||
|   |> close(tag = $edge4) | ||||
|  | ||||
| mountingPlate = extrude(thickness, mountingPlateSketch) | ||||
| mountingPlate = extrude(mountingPlateSketch, length = thickness) | ||||
|   |> fillet({ | ||||
|        radius = filletRadius, | ||||
|        tags = [ | ||||
| @ -63,12 +63,12 @@ filletRadius = 1 | ||||
|  | ||||
| mountingPlateSketch = startSketchOn("XY") | ||||
|   |> startProfileAt([-width / 2, -length / 2], %) | ||||
|   |> lineTo([width / 2, -length / 2], %, $edge1) | ||||
|   |> lineTo([width / 2, length / 2], %, $edge2) | ||||
|   |> lineTo([-width / 2, length / 2], %, $edge3) | ||||
|   |> close(%, $edge4) | ||||
|   |> line(endAbsolute = [width / 2, -length / 2], tag = $edge1) | ||||
|   |> line(endAbsolute = [width / 2, length / 2], tag = $edge2) | ||||
|   |> line(endAbsolute = [-width / 2, length / 2], tag = $edge3) | ||||
|   |> close(tag = $edge4) | ||||
|  | ||||
| mountingPlate = extrude(thickness, mountingPlateSketch) | ||||
| mountingPlate = extrude(mountingPlateSketch, length = thickness) | ||||
|   |> fillet({ | ||||
|        radius = filletRadius, | ||||
|        tolerance = 0.000001, | ||||
|  | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -29,14 +29,14 @@ getNextAdjacentEdge(tag: TagIdentifier) -> Uuid | ||||
| ```js | ||||
| exampleSketch = startSketchOn('XZ') | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line([10, 0], %) | ||||
|   |> line(end = [10, 0]) | ||||
|   |> angledLine({ angle = 60, length = 10 }, %) | ||||
|   |> angledLine({ angle = 120, length = 10 }, %) | ||||
|   |> line([-10, 0], %) | ||||
|   |> line(end = [-10, 0]) | ||||
|   |> angledLine({ angle = 240, length = 10 }, %, $referenceEdge) | ||||
|   |> close(%) | ||||
|   |> close() | ||||
|  | ||||
| example = extrude(5, exampleSketch) | ||||
| example = extrude(exampleSketch, length = 5) | ||||
|   |> fillet({ | ||||
|        radius = 3, | ||||
|        tags = [getNextAdjacentEdge(referenceEdge)] | ||||
|  | ||||
| @ -29,14 +29,14 @@ getOppositeEdge(tag: TagIdentifier) -> Uuid | ||||
| ```js | ||||
| exampleSketch = startSketchOn('XZ') | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line([10, 0], %) | ||||
|   |> line(end = [10, 0]) | ||||
|   |> angledLine({ angle = 60, length = 10 }, %) | ||||
|   |> angledLine({ angle = 120, length = 10 }, %) | ||||
|   |> line([-10, 0], %) | ||||
|   |> line(end = [-10, 0]) | ||||
|   |> angledLine({ angle = 240, length = 10 }, %, $referenceEdge) | ||||
|   |> close(%) | ||||
|   |> close() | ||||
|  | ||||
| example = extrude(5, exampleSketch) | ||||
| example = extrude(exampleSketch, length = 5) | ||||
|   |> fillet({ | ||||
|        radius = 3, | ||||
|        tags = [getOppositeEdge(referenceEdge)] | ||||
|  | ||||
| @ -29,14 +29,14 @@ getPreviousAdjacentEdge(tag: TagIdentifier) -> Uuid | ||||
| ```js | ||||
| exampleSketch = startSketchOn('XZ') | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line([10, 0], %) | ||||
|   |> line(end = [10, 0]) | ||||
|   |> angledLine({ angle = 60, length = 10 }, %) | ||||
|   |> angledLine({ angle = 120, length = 10 }, %) | ||||
|   |> line([-10, 0], %) | ||||
|   |> line(end = [-10, 0]) | ||||
|   |> angledLine({ angle = 240, length = 10 }, %, $referenceEdge) | ||||
|   |> close(%) | ||||
|   |> close() | ||||
|  | ||||
| example = extrude(5, exampleSketch) | ||||
| example = extrude(exampleSketch, length = 5) | ||||
|   |> fillet({ | ||||
|        radius = 3, | ||||
|        tags = [getPreviousAdjacentEdge(referenceEdge)] | ||||
|  | ||||
| @ -49,7 +49,7 @@ springSketch = startSketchOn('YZ') | ||||
| // Create a helix around an edge. | ||||
| helper001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line([0, 10], %, $edge001) | ||||
|   |> line(end = [0, 10], tag = $edge001) | ||||
|  | ||||
| helixPath = helix({ | ||||
|   angleStart = 0, | ||||
|  | ||||
| @ -30,7 +30,7 @@ helixRevolutions(data: HelixRevolutionsData, solid: Solid) -> Solid | ||||
| ```js | ||||
| part001 = startSketchOn('XY') | ||||
|   |> circle({ center = [5, 5], radius = 10 }, %) | ||||
|   |> extrude(10, %) | ||||
|   |> extrude(length = 10) | ||||
|   |> helixRevolutions({ | ||||
|        angleStart = 0, | ||||
|        ccw = true, | ||||
|  | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -31,11 +31,11 @@ hollow(thickness: number, solid: Solid) -> Solid | ||||
| // Hollow a basic sketch. | ||||
| firstSketch = startSketchOn('XY') | ||||
|   |> startProfileAt([-12, 12], %) | ||||
|   |> line([24, 0], %) | ||||
|   |> line([0, -24], %) | ||||
|   |> line([-24, 0], %) | ||||
|   |> close(%) | ||||
|   |> extrude(6, %) | ||||
|   |> line(end = [24, 0]) | ||||
|   |> line(end = [0, -24]) | ||||
|   |> line(end = [-24, 0]) | ||||
|   |> close() | ||||
|   |> extrude(length = 6) | ||||
|   |> hollow(0.25, %) | ||||
| ``` | ||||
|  | ||||
| @ -45,11 +45,11 @@ firstSketch = startSketchOn('XY') | ||||
| // Hollow a basic sketch. | ||||
| firstSketch = startSketchOn('-XZ') | ||||
|   |> startProfileAt([-12, 12], %) | ||||
|   |> line([24, 0], %) | ||||
|   |> line([0, -24], %) | ||||
|   |> line([-24, 0], %) | ||||
|   |> close(%) | ||||
|   |> extrude(6, %) | ||||
|   |> line(end = [24, 0]) | ||||
|   |> line(end = [0, -24]) | ||||
|   |> line(end = [-24, 0]) | ||||
|   |> close() | ||||
|   |> extrude(length = 6) | ||||
|   |> hollow(0.5, %) | ||||
| ``` | ||||
|  | ||||
| @ -60,25 +60,25 @@ firstSketch = startSketchOn('-XZ') | ||||
| size = 100 | ||||
| case = startSketchOn('-XZ') | ||||
|   |> startProfileAt([-size, -size], %) | ||||
|   |> line([2 * size, 0], %) | ||||
|   |> line([0, 2 * size], %) | ||||
|   |> line(end = [2 * size, 0]) | ||||
|   |> line(end = [0, 2 * size]) | ||||
|   |> tangentialArcTo([-size, size], %) | ||||
|   |> close(%) | ||||
|   |> extrude(65, %) | ||||
|   |> close() | ||||
|   |> extrude(length = 65) | ||||
|  | ||||
| thing1 = startSketchOn(case, 'end') | ||||
|   |> circle({ | ||||
|        center = [-size / 2, -size / 2], | ||||
|        radius = 25 | ||||
|      }, %) | ||||
|   |> extrude(50, %) | ||||
|   |> extrude(length = 50) | ||||
|  | ||||
| thing2 = startSketchOn(case, 'end') | ||||
|   |> circle({ | ||||
|        center = [size / 2, -size / 2], | ||||
|        radius = 25 | ||||
|      }, %) | ||||
|   |> extrude(50, %) | ||||
|   |> extrude(length = 50) | ||||
|  | ||||
| hollow(0.5, case) | ||||
| ``` | ||||
|  | ||||
| @ -58,7 +58,6 @@ layout: manual | ||||
| * [`legAngY`](kcl/legAngY) | ||||
| * [`legLen`](kcl/legLen) | ||||
| * [`line`](kcl/line) | ||||
| * [`lineTo`](kcl/lineTo) | ||||
| * [`ln`](kcl/ln) | ||||
| * [`loft`](kcl/loft) | ||||
| * [`log`](kcl/log) | ||||
|  | ||||
| @ -38,7 +38,7 @@ assertEqual(n, 3, 0.0001, "5/2 = 2.5, rounded up makes 3") | ||||
| // Draw n cylinders. | ||||
| startSketchOn('XZ') | ||||
|   |> circle({ center = [0, 0], radius = 2 }, %) | ||||
|   |> extrude(5, %) | ||||
|   |> extrude(length = 5) | ||||
|   |> patternTransform(n, fn(id) { | ||||
|     return { translate = [4 * id, 0, 0] } | ||||
|   }, %) | ||||
|  | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -34,18 +34,18 @@ loft(sketches: [Sketch], v_degree: NonZeroU32, bez_approximate_rational: bool, b | ||||
| // Loft a square and a triangle. | ||||
| squareSketch = startSketchOn('XY') | ||||
|   |> startProfileAt([-100, 200], %) | ||||
|   |> line([200, 0], %) | ||||
|   |> line([0, -200], %) | ||||
|   |> line([-200, 0], %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
|   |> line(end = [200, 0]) | ||||
|   |> line(end = [0, -200]) | ||||
|   |> line(end = [-200, 0]) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
|  | ||||
| triangleSketch = startSketchOn(offsetPlane('XY', 75)) | ||||
|   |> startProfileAt([0, 125], %) | ||||
|   |> line([-15, -30], %) | ||||
|   |> line([30, 0], %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
|   |> line(end = [-15, -30]) | ||||
|   |> line(end = [30, 0]) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
|  | ||||
| loft([squareSketch, triangleSketch]) | ||||
| ``` | ||||
| @ -56,11 +56,11 @@ loft([squareSketch, triangleSketch]) | ||||
| // Loft a square, a circle, and another circle. | ||||
| squareSketch = startSketchOn('XY') | ||||
|   |> startProfileAt([-100, 200], %) | ||||
|   |> line([200, 0], %) | ||||
|   |> line([0, -200], %) | ||||
|   |> line([-200, 0], %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
|   |> line(end = [200, 0]) | ||||
|   |> line(end = [0, -200]) | ||||
|   |> line(end = [-200, 0]) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
|  | ||||
| circleSketch0 = startSketchOn(offsetPlane('XY', 75)) | ||||
|   |> circle({ center = [0, 100], radius = 50 }, %) | ||||
| @ -81,11 +81,11 @@ loft([ | ||||
| // Loft a square, a circle, and another circle with options. | ||||
| squareSketch = startSketchOn('XY') | ||||
|   |> startProfileAt([-100, 200], %) | ||||
|   |> line([200, 0], %) | ||||
|   |> line([0, -200], %) | ||||
|   |> line([-200, 0], %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
|   |> line(end = [200, 0]) | ||||
|   |> line(end = [0, -200]) | ||||
|   |> line(end = [-200, 0]) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
|  | ||||
| circleSketch0 = startSketchOn(offsetPlane('XY', 75)) | ||||
|   |> circle({ center = [0, 100], radius = 50 }, %) | ||||
|  | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -31,7 +31,7 @@ patternCircular3d(data: CircularPattern3dData, solid_set: SolidSet) -> [Solid] | ||||
| exampleSketch = startSketchOn('XZ') | ||||
|   |> circle({ center = [0, 0], radius = 1 }, %) | ||||
|  | ||||
| example = extrude(-5, exampleSketch) | ||||
| example = extrude(exampleSketch, length = -5) | ||||
|   |> patternCircular3d({ | ||||
|        axis = [1, -1, 0], | ||||
|        center = [10, -20, 0], | ||||
|  | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -30,12 +30,12 @@ patternLinear3d(data: LinearPattern3dData, solid_set: SolidSet) -> [Solid] | ||||
| ```js | ||||
| exampleSketch = startSketchOn('XZ') | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line([0, 2], %) | ||||
|   |> line([3, 1], %) | ||||
|   |> line([0, -4], %) | ||||
|   |> close(%) | ||||
|   |> line(end = [0, 2]) | ||||
|   |> line(end = [3, 1]) | ||||
|   |> line(end = [0, -4]) | ||||
|   |> close() | ||||
|  | ||||
| example = extrude(1, exampleSketch) | ||||
| example = extrude(exampleSketch, length = 1) | ||||
|   |> patternLinear3d({ | ||||
|        axis = [1, 0, 1], | ||||
|        instances = 7, | ||||
|  | ||||
| @ -63,7 +63,7 @@ fn transform(id) { | ||||
| // Sketch 4 cylinders. | ||||
| sketch001 = startSketchOn('XZ') | ||||
|   |> circle({ center = [0, 0], radius = 2 }, %) | ||||
|   |> extrude(5, %) | ||||
|   |> extrude(length = 5) | ||||
|   |> patternTransform(4, transform, %) | ||||
| ``` | ||||
|  | ||||
| @ -79,7 +79,7 @@ fn transform(id) { | ||||
|  | ||||
| sketch001 = startSketchOn('XZ') | ||||
|   |> circle({ center = [0, 0], radius = 2 }, %) | ||||
|   |> extrude(5, %) | ||||
|   |> extrude(length = 5) | ||||
|   |> patternTransform(4, transform, %) | ||||
| ``` | ||||
|  | ||||
| @ -97,12 +97,12 @@ fn cube(length, center) { | ||||
|  | ||||
|   return startSketchOn('XY') | ||||
|     |> startProfileAt(p0, %) | ||||
|     |> lineTo(p1, %) | ||||
|     |> lineTo(p2, %) | ||||
|     |> lineTo(p3, %) | ||||
|     |> lineTo(p0, %) | ||||
|     |> close(%) | ||||
|     |> extrude(length, %) | ||||
|     |> line(endAbsolute = p1) | ||||
|     |> line(endAbsolute = p2) | ||||
|     |> line(endAbsolute = p3) | ||||
|     |> line(endAbsolute = p0) | ||||
|     |> close() | ||||
|     |> extrude(length = length) | ||||
| } | ||||
|  | ||||
| width = 20 | ||||
| @ -135,12 +135,12 @@ fn cube(length, center) { | ||||
|  | ||||
|   return startSketchOn('XY') | ||||
|     |> startProfileAt(p0, %) | ||||
|     |> lineTo(p1, %) | ||||
|     |> lineTo(p2, %) | ||||
|     |> lineTo(p3, %) | ||||
|     |> lineTo(p0, %) | ||||
|     |> close(%) | ||||
|     |> extrude(length, %) | ||||
|     |> line(endAbsolute = p1) | ||||
|     |> line(endAbsolute = p2) | ||||
|     |> line(endAbsolute = p3) | ||||
|     |> line(endAbsolute = p0) | ||||
|     |> close() | ||||
|     |> extrude(length = length) | ||||
| } | ||||
|  | ||||
| width = 20 | ||||
| @ -179,7 +179,7 @@ fn layer() { | ||||
|   return startSketchOn("XY") | ||||
|     // or some other plane idk | ||||
|     |> circle({ center = [0, 0], radius = 1 }, %, $tag1) | ||||
|     |> extrude(h, %) | ||||
|     |> extrude(length = h) | ||||
| } | ||||
| // The vase is 100 layers tall. | ||||
| // The 100 layers are replica of each other, with a slight transformation applied to each. | ||||
| @ -205,7 +205,7 @@ startSketchOn('XY') | ||||
|        center = [0, 0], | ||||
|        inscribed = false | ||||
|      }, %) | ||||
|   |> extrude(4, %) | ||||
|   |> extrude(length = 4) | ||||
|   |> patternTransform(3, transform, %) | ||||
| ``` | ||||
|  | ||||
|  | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -30,12 +30,12 @@ segEnd(tag: TagIdentifier) -> [number] | ||||
| w = 15 | ||||
| cube = startSketchOn('XY') | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line([w, 0], %, $line1) | ||||
|   |> line([0, w], %, $line2) | ||||
|   |> line([-w, 0], %, $line3) | ||||
|   |> line([0, -w], %, $line4) | ||||
|   |> close(%) | ||||
|   |> extrude(5, %) | ||||
|   |> line(end = [w, 0], tag = $line1) | ||||
|   |> line(end = [0, w], tag = $line2) | ||||
|   |> line(end = [-w, 0], tag = $line3) | ||||
|   |> line(end = [0, -w], tag = $line4) | ||||
|   |> close() | ||||
|   |> extrude(length = 5) | ||||
|  | ||||
| fn cylinder(radius, tag) { | ||||
|   return startSketchOn('XY') | ||||
| @ -44,7 +44,7 @@ fn cylinder(radius, tag) { | ||||
|          radius = radius, | ||||
|          center = segEnd(tag) | ||||
|        }, %) | ||||
|     |> extrude(radius, %) | ||||
|     |> extrude(length = radius) | ||||
| } | ||||
|  | ||||
| cylinder(1, line1) | ||||
|  | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -30,12 +30,12 @@ segStart(tag: TagIdentifier) -> [number] | ||||
| w = 15 | ||||
| cube = startSketchOn('XY') | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line([w, 0], %, $line1) | ||||
|   |> line([0, w], %, $line2) | ||||
|   |> line([-w, 0], %, $line3) | ||||
|   |> line([0, -w], %, $line4) | ||||
|   |> close(%) | ||||
|   |> extrude(5, %) | ||||
|   |> line(end = [w, 0], tag = $line1) | ||||
|   |> line(end = [0, w], tag = $line2) | ||||
|   |> line(end = [-w, 0], tag = $line3) | ||||
|   |> line(end = [0, -w], tag = $line4) | ||||
|   |> close() | ||||
|   |> extrude(length = 5) | ||||
|  | ||||
| fn cylinder(radius, tag) { | ||||
|   return startSketchOn('XY') | ||||
| @ -44,7 +44,7 @@ fn cylinder(radius, tag) { | ||||
|          radius = radius, | ||||
|          center = segStart(tag) | ||||
|        }, %) | ||||
|     |> extrude(radius, %) | ||||
|     |> extrude(length = radius) | ||||
| } | ||||
|  | ||||
| cylinder(1, line1) | ||||
|  | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -31,11 +31,11 @@ shell(data: ShellData, solid_set: SolidSet) -> SolidSet | ||||
| // Remove the end face for the extrusion. | ||||
| firstSketch = startSketchOn('XY') | ||||
|   |> startProfileAt([-12, 12], %) | ||||
|   |> line([24, 0], %) | ||||
|   |> line([0, -24], %) | ||||
|   |> line([-24, 0], %) | ||||
|   |> close(%) | ||||
|   |> extrude(6, %) | ||||
|   |> line(end = [24, 0]) | ||||
|   |> line(end = [0, -24]) | ||||
|   |> line(end = [-24, 0]) | ||||
|   |> close() | ||||
|   |> extrude(length = 6) | ||||
|  | ||||
| // Remove the end face for the extrusion. | ||||
| shell({ faces = ['end'], thickness = 0.25 }, firstSketch) | ||||
| @ -47,11 +47,11 @@ shell({ faces = ['end'], thickness = 0.25 }, firstSketch) | ||||
| // Remove the start face for the extrusion. | ||||
| firstSketch = startSketchOn('-XZ') | ||||
|   |> startProfileAt([-12, 12], %) | ||||
|   |> line([24, 0], %) | ||||
|   |> line([0, -24], %) | ||||
|   |> line([-24, 0], %) | ||||
|   |> close(%) | ||||
|   |> extrude(6, %) | ||||
|   |> line(end = [24, 0]) | ||||
|   |> line(end = [0, -24]) | ||||
|   |> line(end = [-24, 0]) | ||||
|   |> close() | ||||
|   |> extrude(length = 6) | ||||
|  | ||||
| // Remove the start face for the extrusion. | ||||
| shell({ faces = ['start'], thickness = 0.25 }, firstSketch) | ||||
| @ -63,11 +63,11 @@ shell({ faces = ['start'], thickness = 0.25 }, firstSketch) | ||||
| // Remove a tagged face and the end face for the extrusion. | ||||
| firstSketch = startSketchOn('XY') | ||||
|   |> startProfileAt([-12, 12], %) | ||||
|   |> line([24, 0], %) | ||||
|   |> line([0, -24], %) | ||||
|   |> line([-24, 0], %, $myTag) | ||||
|   |> close(%) | ||||
|   |> extrude(6, %) | ||||
|   |> line(end = [24, 0]) | ||||
|   |> line(end = [0, -24]) | ||||
|   |> line(end = [-24, 0], tag = $myTag) | ||||
|   |> close() | ||||
|   |> extrude(length = 6) | ||||
|  | ||||
| // Remove a tagged face for the extrusion. | ||||
| shell({ faces = [myTag], thickness = 0.25 }, firstSketch) | ||||
| @ -79,11 +79,11 @@ shell({ faces = [myTag], thickness = 0.25 }, firstSketch) | ||||
| // Remove multiple faces at once. | ||||
| firstSketch = startSketchOn('XY') | ||||
|   |> startProfileAt([-12, 12], %) | ||||
|   |> line([24, 0], %) | ||||
|   |> line([0, -24], %) | ||||
|   |> line([-24, 0], %, $myTag) | ||||
|   |> close(%) | ||||
|   |> extrude(6, %) | ||||
|   |> line(end = [24, 0]) | ||||
|   |> line(end = [0, -24]) | ||||
|   |> line(end = [-24, 0], tag = $myTag) | ||||
|   |> close() | ||||
|   |> extrude(length = 6) | ||||
|  | ||||
| // Remove a tagged face and the end face for the extrusion. | ||||
| shell({ | ||||
| @ -99,25 +99,25 @@ shell({ | ||||
| size = 100 | ||||
| case = startSketchOn('-XZ') | ||||
|   |> startProfileAt([-size, -size], %) | ||||
|   |> line([2 * size, 0], %) | ||||
|   |> line([0, 2 * size], %) | ||||
|   |> line(end = [2 * size, 0]) | ||||
|   |> line(end = [0, 2 * size]) | ||||
|   |> tangentialArcTo([-size, size], %) | ||||
|   |> close(%) | ||||
|   |> extrude(65, %) | ||||
|   |> close() | ||||
|   |> extrude(length = 65) | ||||
|  | ||||
| thing1 = startSketchOn(case, 'end') | ||||
|   |> circle({ | ||||
|        center = [-size / 2, -size / 2], | ||||
|        radius = 25 | ||||
|      }, %) | ||||
|   |> extrude(50, %) | ||||
|   |> extrude(length = 50) | ||||
|  | ||||
| thing2 = startSketchOn(case, 'end') | ||||
|   |> circle({ | ||||
|        center = [size / 2, -size / 2], | ||||
|        radius = 25 | ||||
|      }, %) | ||||
|   |> extrude(50, %) | ||||
|   |> extrude(length = 50) | ||||
|  | ||||
| // We put "case" in the shell function to shell the entire object. | ||||
| shell({ faces = ['start'], thickness = 5 }, case) | ||||
| @ -130,25 +130,25 @@ shell({ faces = ['start'], thickness = 5 }, case) | ||||
| size = 100 | ||||
| case = startSketchOn('XY') | ||||
|   |> startProfileAt([-size, -size], %) | ||||
|   |> line([2 * size, 0], %) | ||||
|   |> line([0, 2 * size], %) | ||||
|   |> line(end = [2 * size, 0]) | ||||
|   |> line(end = [0, 2 * size]) | ||||
|   |> tangentialArcTo([-size, size], %) | ||||
|   |> close(%) | ||||
|   |> extrude(65, %) | ||||
|   |> close() | ||||
|   |> extrude(length = 65) | ||||
|  | ||||
| thing1 = startSketchOn(case, 'end') | ||||
|   |> circle({ | ||||
|        center = [-size / 2, -size / 2], | ||||
|        radius = 25 | ||||
|      }, %) | ||||
|   |> extrude(50, %) | ||||
|   |> extrude(length = 50) | ||||
|  | ||||
| thing2 = startSketchOn(case, 'end') | ||||
|   |> circle({ | ||||
|        center = [size / 2, -size / 2], | ||||
|        radius = 25 | ||||
|      }, %) | ||||
|   |> extrude(50, %) | ||||
|   |> extrude(length = 50) | ||||
|  | ||||
| // We put "thing1" in the shell function to shell the end face of the object. | ||||
| shell({ faces = ['end'], thickness = 5 }, thing1) | ||||
| @ -164,25 +164,25 @@ shell({ faces = ['end'], thickness = 5 }, thing1) | ||||
| size = 100 | ||||
| case = startSketchOn('XY') | ||||
|   |> startProfileAt([-size, -size], %) | ||||
|   |> line([2 * size, 0], %) | ||||
|   |> line([0, 2 * size], %) | ||||
|   |> line(end = [2 * size, 0]) | ||||
|   |> line(end = [0, 2 * size]) | ||||
|   |> tangentialArcTo([-size, size], %) | ||||
|   |> close(%) | ||||
|   |> extrude(65, %) | ||||
|   |> close() | ||||
|   |> extrude(length = 65) | ||||
|  | ||||
| thing1 = startSketchOn(case, 'end') | ||||
|   |> circle({ | ||||
|        center = [-size / 2, -size / 2], | ||||
|        radius = 25 | ||||
|      }, %) | ||||
|   |> extrude(50, %) | ||||
|   |> extrude(length = 50) | ||||
|  | ||||
| thing2 = startSketchOn(case, 'end') | ||||
|   |> circle({ | ||||
|        center = [size / 2, -size / 2], | ||||
|        radius = 25 | ||||
|      }, %) | ||||
|   |> extrude(50, %) | ||||
|   |> extrude(length = 50) | ||||
|  | ||||
| // We put "thing1" and "thing2" in the shell function to shell the end face of the object. | ||||
| shell({ faces = ['end'], thickness = 5 }, [thing1, thing2]) | ||||
|  | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										9318
									
								
								docs/kcl/std.json
									
									
									
									
									
								
							
							
						
						
									
										9318
									
								
								docs/kcl/std.json
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -34,11 +34,11 @@ sweep(data: SweepData, sketch: Sketch) -> Solid | ||||
| // Create a path for the sweep. | ||||
| sweepPath = startSketchOn('XZ') | ||||
|   |> startProfileAt([0.05, 0.05], %) | ||||
|   |> line([0, 7], %) | ||||
|   |> line(end = [0, 7]) | ||||
|   |> tangentialArc({ offset = 90, radius = 5 }, %) | ||||
|   |> line([-3, 0], %) | ||||
|   |> line(end = [-3, 0]) | ||||
|   |> tangentialArc({ offset = -90, radius = 5 }, %) | ||||
|   |> line([0, 7], %) | ||||
|   |> line(end = [0, 7]) | ||||
|  | ||||
| // Create a hole for the pipe. | ||||
| pipeHole = startSketchOn('XY') | ||||
|  | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -47,21 +47,6 @@ myObj = { a = 0, b = "thing" } | ||||
| We support two different ways of getting properties from objects, you can call | ||||
| `myObj.a` or `myObj["a"]` both work. | ||||
|  | ||||
|  | ||||
| ## Functions | ||||
|  | ||||
| We also have support for defining your own functions. Functions can take in any | ||||
| type of argument. Below is an example of the syntax: | ||||
|  | ||||
| ``` | ||||
| fn myFn(x) { | ||||
|   return x | ||||
| } | ||||
| ``` | ||||
|  | ||||
| As you can see above `myFn` just returns whatever it is given. | ||||
|  | ||||
|  | ||||
| ## Binary expressions | ||||
|  | ||||
| You can also do math! Let's show an example below: | ||||
| @ -76,6 +61,120 @@ You can nest expressions in parenthesis as well: | ||||
| myMathExpression = 3 + (1 * 2 / (3 - 7)) | ||||
| ``` | ||||
|  | ||||
| ## Functions | ||||
|  | ||||
| We also have support for defining your own functions. Functions can take in any | ||||
| type of argument. Below is an example of the syntax: | ||||
|  | ||||
| ``` | ||||
| fn myFn(x) { | ||||
|   return x | ||||
| } | ||||
| ``` | ||||
|  | ||||
| As you can see above `myFn` just returns whatever it is given. | ||||
|  | ||||
| KCL's early drafts used positional arguments, but we now use keyword arguments. If you declare a | ||||
| function like this: | ||||
|  | ||||
| ``` | ||||
| fn add(left, right) { | ||||
|   return left + right | ||||
| } | ||||
| ``` | ||||
|  | ||||
| You can call it like this: | ||||
|  | ||||
| ``` | ||||
| total = add(left = 1, right = 2) | ||||
| ``` | ||||
|  | ||||
| Functions can also declare one *unlabeled* arg. If you do want to declare an unlabeled arg, it must | ||||
| be the first arg declared. | ||||
|  | ||||
| ``` | ||||
| // The @ indicates an argument can be used without a label. | ||||
| // Note that only the first argument can use @. | ||||
| fn increment(@x) { | ||||
|   return x + 1 | ||||
| } | ||||
|  | ||||
| fn add(@x, delta) { | ||||
|   return x + delta | ||||
| } | ||||
|  | ||||
| two = increment(1) | ||||
| three = add(1, delta = 2) | ||||
| ``` | ||||
|  | ||||
| ## Pipelines | ||||
|  | ||||
| It can be hard to read repeated function calls, because of all the nested brackets. | ||||
|  | ||||
| ``` | ||||
| i = 1 | ||||
| x = h(g(f(i))) | ||||
| ``` | ||||
|  | ||||
| You can make this easier to read by breaking it into many declarations, but that is a bit annoying. | ||||
|  | ||||
| ``` | ||||
| i  = 1 | ||||
| x0 = f(i) | ||||
| x1 = g(x0) | ||||
| x  = h(x1) | ||||
| ``` | ||||
|  | ||||
| Instead, you can use the pipeline operator (`|>`) to simplify this. | ||||
|  | ||||
| Basically, `x |> f(%)` is a shorthand for `f(x)`. The left-hand side of the `|>` gets put into | ||||
| the `%` in the right-hand side. | ||||
|  | ||||
| So, this means `x |> f(%) |> g(%)` is shorthand for `g(f(x))`. The code example above, with its | ||||
| somewhat-clunky `x0` and `x1` constants could be rewritten as | ||||
|  | ||||
| ``` | ||||
| i = 1 | ||||
| x = i | ||||
| |> f(%) | ||||
| |> g(%) | ||||
| |> h(%) | ||||
| ``` | ||||
|  | ||||
| This helps keep your code neat and avoid unnecessary declarations. | ||||
|  | ||||
| ## Pipelines and keyword arguments | ||||
|  | ||||
| Say you have a long pipeline of sketch functions, like this: | ||||
|  | ||||
| ``` | ||||
| startSketch() | ||||
| |> line(%, end = [3, 4]) | ||||
| |> line(%, end = [10, 10]) | ||||
| |> line(%, end = [-13, -14]) | ||||
| |> close(%) | ||||
| ``` | ||||
|  | ||||
| In this example, each function call outputs a sketch, and it gets put into the next function call via | ||||
| the `%`, into the first (unlabeled) argument. | ||||
|  | ||||
| If a function call uses an unlabeled first parameter, it will default to `%` if it's not given. This | ||||
| means that `|> line(%, end = [3, 4])` and `|> line(end = [3, 4])` are equivalent! So the above | ||||
| could be rewritten as  | ||||
|  | ||||
| ``` | ||||
| startSketch() | ||||
| |> line(end = [3, 4]) | ||||
| |> line(end = [10, 10]) | ||||
| |> line(end = [-13, -14]) | ||||
| |> close() | ||||
| ``` | ||||
|  | ||||
| Note that we are still in the process of migrating KCL's standard library to use keyword arguments. So some | ||||
| functions are still unfortunately using positional arguments. We're moving them over, so keep checking back. | ||||
| Some functions like `angledLine`, `startProfileAt` etc are still using the old positional argument syntax. | ||||
| Check the docs page for each function and look at its examples to see. | ||||
|  | ||||
| ## Tags | ||||
|  | ||||
| Tags are used to give a name (tag) to a specific path. | ||||
| @ -88,17 +187,17 @@ way: | ||||
| ``` | ||||
| startSketchOn('XZ') | ||||
|   |> startProfileAt(origin, %) | ||||
|   |> angledLine([0, 191.26], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001) - 90, | ||||
|        196.99 | ||||
|      ], %, $rectangleSegmentB001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001), | ||||
|        -segLen(rectangleSegmentA001) | ||||
|      ], %, $rectangleSegmentC001) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
|   |> angledLine({angle = 0, length = 191.26}, %, $rectangleSegmentA001) | ||||
|   |> angledLine({ | ||||
|        angle = segAng(rectangleSegmentA001) - 90, | ||||
|        length = 196.99, | ||||
|      }, %, $rectangleSegmentB001) | ||||
|   |> angledLine({ | ||||
|        angle = segAng(rectangleSegmentA001), | ||||
|        length = -segLen(rectangleSegmentA001), | ||||
|      }, %, $rectangleSegmentC001) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| ``` | ||||
|  | ||||
| ### Tag Identifier | ||||
| @ -121,17 +220,17 @@ However if the code was written like this: | ||||
| fn rect(origin) { | ||||
|   return startSketchOn('XZ') | ||||
|     |> startProfileAt(origin, %) | ||||
|     |> angledLine([0, 191.26], %, $rectangleSegmentA001) | ||||
|     |> angledLine([ | ||||
|          segAng(rectangleSegmentA001) - 90, | ||||
|          196.99 | ||||
|        ], %, $rectangleSegmentB001) | ||||
|     |> angledLine([ | ||||
|          segAng(rectangleSegmentA001), | ||||
|          -segLen(rectangleSegmentA001) | ||||
|        ], %, $rectangleSegmentC001) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%) | ||||
|     |> angledLine({angle = 0, length = 191.26}, %, $rectangleSegmentA001) | ||||
|     |> angledLine({ | ||||
|          angle = segAng(rectangleSegmentA001) - 90, | ||||
|          length = 196.99 | ||||
|        }, %, $rectangleSegmentB001) | ||||
|     |> angledLine({ | ||||
|          angle = segAng(rectangleSegmentA001), | ||||
|          length = -segLen(rectangleSegmentA001) | ||||
|        }, %, $rectangleSegmentC001) | ||||
|     |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|     |> close() | ||||
| } | ||||
|  | ||||
| rect([0, 0]) | ||||
| @ -149,17 +248,17 @@ For example the following code works. | ||||
| fn rect(origin) { | ||||
|   return startSketchOn('XZ') | ||||
|     |> startProfileAt(origin, %) | ||||
|     |> angledLine([0, 191.26], %, $rectangleSegmentA001) | ||||
|     |> angledLine([ | ||||
|          segAng(rectangleSegmentA001) - 90, | ||||
|          196.99 | ||||
|        ], %, $rectangleSegmentB001) | ||||
|     |> angledLine([ | ||||
|          segAng(rectangleSegmentA001), | ||||
|          -segLen(rectangleSegmentA001) | ||||
|        ], %, $rectangleSegmentC001) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%) | ||||
|     |> angledLine({angle = 0, length = 191.26}, %, $rectangleSegmentA001) | ||||
|     |> angledLine({ | ||||
|          angle = segAng(rectangleSegmentA001) - 90, | ||||
|          length = 196.99 | ||||
|        }, %, $rectangleSegmentB001) | ||||
|     |> angledLine({ | ||||
|          angle = segAng(rectangleSegmentA001), | ||||
|          length = -segLen(rectangleSegmentA001) | ||||
|        }, %, $rectangleSegmentC001) | ||||
|     |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|     |> close() | ||||
| } | ||||
|  | ||||
| rect([0, 0]) | ||||
|  | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user
	