Compare commits
	
		
			50 Commits
		
	
	
		
			nightly-v2
			...
			pierremtb/
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| fecf5c2ee7 | |||
| 8ef31a0be1 | |||
| 3adb42b5f2 | |||
| 20016b101e | |||
| 8d9dbf36c3 | |||
| 440704ed9f | |||
| 2261217a5d | |||
| 10da986649 | |||
| 10789d9c3c | |||
| 67cc4f5835 | |||
| 2692f2b73a | |||
| 965cb18059 | |||
| a022b8ef6c | |||
| 4d24bf7c94 | |||
| 9a537da183 | |||
| df81b76b8b | |||
| ac3f7ab712 | |||
| d531728675 | |||
| 1d78fc15ac | |||
| c32aebc8ad | |||
| 997ebce3eb | |||
| dac91d3b79 | |||
| 1eaf371b44 | |||
| 0698432abf | |||
| 54da18d8ab | |||
| 2fe5ef7034 | |||
| 16b5eeadb1 | |||
| 7be4001839 | |||
| ffb2559787 | |||
| 0592d3b5da | |||
| 31e4d60045 | |||
| c0817b00e4 | |||
| 4ea1d16fb6 | |||
| d049bf33e8 | |||
| 7b11047d07 | |||
| 412e9568f2 | |||
| 9be208e5e1 | |||
| 842ef5ede9 | |||
| 3f855d7bad | |||
| 0a1a6e50cf | |||
| d4e955289c | |||
| c147a219f4 | |||
| 38513a1e25 | |||
| c0c5c790ca | |||
| 8b60f75220 | |||
| f91ad4331f | |||
| 59103a2118 | |||
| 9737c2550a | |||
| bf9d01a8dd | |||
| 702e322f90 | 
| @ -1,3 +1,3 @@ | |||||||
| [codespell] | [codespell] | ||||||
| ignore-words-list: crate,everytime,inout,co-ordinate,ot,nwo,atleast,ue,afterall | ignore-words-list: crate,everytime,inout,co-ordinate,ot,nwo,atleast,ue,afterall,ser | ||||||
| skip: **/target,node_modules,build,dist,./out,**/Cargo.lock,./docs/kcl/*.md,.yarn.lock,**/yarn.lock,./openapi/*.json,./packages/codemirror-lang-kcl/test/all.test.ts,tsconfig.tsbuildinfo | skip: **/target,node_modules,build,dist,./out,**/Cargo.lock,./docs/kcl/*.md,.yarn.lock,**/yarn.lock,./openapi/*.json,./packages/codemirror-lang-kcl/test/all.test.ts,tsconfig.tsbuildinfo | ||||||
|  | |||||||
							
								
								
									
										2
									
								
								.github/workflows/build-and-store-wasm.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -27,7 +27,7 @@ jobs: | |||||||
|  |  | ||||||
|  |  | ||||||
|       # Upload the WASM bundle as an artifact |       # Upload the WASM bundle as an artifact | ||||||
|       - uses: actions/upload-artifact@v3 |       - uses: actions/upload-artifact@v4 | ||||||
|         with: |         with: | ||||||
|           name: wasm-bundle |           name: wasm-bundle | ||||||
|           path: src/wasm-lib/pkg |           path: src/wasm-lib/pkg | ||||||
|  | |||||||
							
								
								
									
										8
									
								
								.github/workflows/build-apps.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -126,7 +126,13 @@ jobs: | |||||||
|           node-version-file: '.nvmrc' |           node-version-file: '.nvmrc' | ||||||
|           cache: 'yarn' # Set this to npm, yarn or pnpm. |           cache: 'yarn' # Set this to npm, yarn or pnpm. | ||||||
|  |  | ||||||
|       - run: yarn install |       - name: yarn install | ||||||
|  |         # Windows is picky sometimes and fails on fetch. Step takes about ~30s | ||||||
|  |         uses: nick-fields/retry@v3.0.0 | ||||||
|  |         with: | ||||||
|  |           timeout_minutes: 2 | ||||||
|  |           max_attempts: 3 | ||||||
|  |           command: yarn install | ||||||
|  |  | ||||||
|       - run: yarn tronb:vite |       - run: yarn tronb:vite | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										44
									
								
								.github/workflows/cargo-bench.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -1,44 +0,0 @@ | |||||||
| on: |  | ||||||
|   push: |  | ||||||
|     branches: |  | ||||||
|       - main |  | ||||||
|     paths: |  | ||||||
|       - '**.rs' |  | ||||||
|       - '**/Cargo.toml' |  | ||||||
|       - '**/Cargo.lock' |  | ||||||
|       - '**/rust-toolchain.toml' |  | ||||||
|       - .github/workflows/cargo-bench.yml |  | ||||||
|   pull_request: |  | ||||||
|     paths: |  | ||||||
|       - '**.rs' |  | ||||||
|       - '**/Cargo.toml' |  | ||||||
|       - '**/Cargo.lock' |  | ||||||
|       - '**/rust-toolchain.toml' |  | ||||||
|       - .github/workflows/cargo-bench.yml |  | ||||||
|   workflow_dispatch: |  | ||||||
| permissions: read-all |  | ||||||
| concurrency: |  | ||||||
|   group: ${{ github.workflow }}-${{ github.head_ref || github.run_id }} |  | ||||||
|   cancel-in-progress: true |  | ||||||
| name: cargo bench |  | ||||||
| jobs: |  | ||||||
|   cargo-bench: |  | ||||||
|     name: Benchmark with iai |  | ||||||
|     runs-on: ubuntu-latest-8-cores |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|       - uses: dtolnay/rust-toolchain@stable |  | ||||||
|       - name: Install dependencies |  | ||||||
|         run: | |  | ||||||
|           cargo install cargo-criterion |  | ||||||
|           sudo apt update |  | ||||||
|           sudo apt install -y valgrind |  | ||||||
|       - name: Rust Cache |  | ||||||
|         uses: Swatinem/rust-cache@v2.6.1 |  | ||||||
|       - name: Benchmark kcl library |  | ||||||
|         shell: bash |  | ||||||
|         run: |- |  | ||||||
|           cd src/wasm-lib/kcl; cargo bench --all-features -- iai |  | ||||||
|         env: |  | ||||||
|           KITTYCAD_API_TOKEN: ${{secrets.KITTYCAD_API_TOKEN}} |  | ||||||
|  |  | ||||||
							
								
								
									
										32
									
								
								.github/workflows/codemirror-lang-kcl.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,32 @@ | |||||||
|  | name: CodeMirror Lang KCL | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   pull_request: | ||||||
|  |   push: | ||||||
|  |     branches: | ||||||
|  |       - main | ||||||
|  |  | ||||||
|  | concurrency: | ||||||
|  |   group: ${{ github.workflow }}-${{ github.head_ref || github.run_id }} | ||||||
|  |   cancel-in-progress: true | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   yarn-unit-test: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - uses: actions/checkout@v4 | ||||||
|  |  | ||||||
|  |       - uses: actions/setup-node@v4 | ||||||
|  |         with: | ||||||
|  |           node-version-file: '.nvmrc' | ||||||
|  |           cache: 'yarn' | ||||||
|  |  | ||||||
|  |       - run: yarn install | ||||||
|  |         working-directory: packages/codemirror-lang-kcl | ||||||
|  |  | ||||||
|  |       - run: yarn tsc | ||||||
|  |         working-directory: packages/codemirror-lang-kcl | ||||||
|  |  | ||||||
|  |       - name: run unit tests | ||||||
|  |         run: yarn test | ||||||
|  |         working-directory: packages/codemirror-lang-kcl | ||||||
| @ -24,5 +24,3 @@ once fixed in engine will just start working here with no language changes. | |||||||
|     chamfer cases work currently. |     chamfer cases work currently. | ||||||
|  |  | ||||||
| - **Appearance**: Changing the appearance on a loft does not work. | - **Appearance**: Changing the appearance on a loft does not work. | ||||||
|  |  | ||||||
| - **Helix**: Currently sweeping a helix does not work. |  | ||||||
|  | |||||||
| @ -53,7 +53,6 @@ layout: manual | |||||||
| * [`hollow`](kcl/hollow) | * [`hollow`](kcl/hollow) | ||||||
| * [`import`](kcl/import) | * [`import`](kcl/import) | ||||||
| * [`inch`](kcl/inch) | * [`inch`](kcl/inch) | ||||||
| * [`int`](kcl/int) |  | ||||||
| * [`lastSegX`](kcl/lastSegX) | * [`lastSegX`](kcl/lastSegX) | ||||||
| * [`lastSegY`](kcl/lastSegY) | * [`lastSegY`](kcl/lastSegY) | ||||||
| * [`legAngX`](kcl/legAngX) | * [`legAngX`](kcl/legAngX) | ||||||
|  | |||||||
| @ -4,6 +4,8 @@ excerpt: "Convert a number to an integer." | |||||||
| layout: manual | layout: manual | ||||||
| --- | --- | ||||||
|  |  | ||||||
|  | **WARNING:** This function is deprecated. | ||||||
|  |  | ||||||
| Convert a number to an integer. | Convert a number to an integer. | ||||||
|  |  | ||||||
| DEPRECATED use floor(), ceil(), or round(). | DEPRECATED use floor(), ceil(), or round(). | ||||||
|  | |||||||
							
								
								
									
										20596
									
								
								docs/kcl/std.json
									
									
									
									
									
								
							
							
						
						
							
								
								
									
										28
									
								
								docs/kcl/types/Face.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,28 @@ | |||||||
|  | --- | ||||||
|  | title: "Face" | ||||||
|  | excerpt: "A face." | ||||||
|  | layout: manual | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | A face. | ||||||
|  |  | ||||||
|  | **Type:** `object` | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## Properties | ||||||
|  |  | ||||||
|  | | Property | Type | Description | Required | | ||||||
|  | |----------|------|-------------|----------| | ||||||
|  | | `id` |`string`| The id of the face. | No | | ||||||
|  | | `value` |`string`| The tag of the face. | No | | ||||||
|  | | `xAxis` |[`Point3d`](/docs/kcl/types/Point3d)| What should the face’s X axis be? | No | | ||||||
|  | | `yAxis` |[`Point3d`](/docs/kcl/types/Point3d)| What should the face’s Y axis be? | No | | ||||||
|  | | `zAxis` |[`Point3d`](/docs/kcl/types/Point3d)| The z-axis (normal). | No | | ||||||
|  | | `solid` |[`Solid`](/docs/kcl/types/Solid)| The solid the face is on. | No | | ||||||
|  | | `units` |[`UnitLen`](/docs/kcl/types/UnitLen)| A face. | No | | ||||||
|  | | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`|  | No | | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -20,6 +20,7 @@ A helix. | |||||||
| | `revolutions` |`number`| Number of revolutions. | No | | | `revolutions` |`number`| Number of revolutions. | No | | ||||||
| | `angleStart` |`number`| Start angle (in degrees). | No | | | `angleStart` |`number`| Start angle (in degrees). | No | | ||||||
| | `ccw` |`boolean`| Is the helix rotation counter clockwise? | No | | | `ccw` |`boolean`| Is the helix rotation counter clockwise? | No | | ||||||
|  | | `units` |[`UnitLen`](/docs/kcl/types/UnitLen)| A helix. | No | | ||||||
| | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`|  | No | | | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`|  | No | | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
| @ -20,6 +20,7 @@ A helix. | |||||||
| | `revolutions` |`number`| Number of revolutions. | No | | | `revolutions` |`number`| Number of revolutions. | No | | ||||||
| | `angleStart` |`number`| Start angle (in degrees). | No | | | `angleStart` |`number`| Start angle (in degrees). | No | | ||||||
| | `ccw` |`boolean`| Is the helix rotation counter clockwise? | No | | | `ccw` |`boolean`| Is the helix rotation counter clockwise? | No | | ||||||
|  | | `units` |[`UnitLen`](/docs/kcl/types/UnitLen)| A helix. | No | | ||||||
| | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`|  | No | | | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`|  | No | | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
| @ -168,7 +168,6 @@ Any KCL value. | |||||||
|  |  | ||||||
|  |  | ||||||
| ---- | ---- | ||||||
| A plane. |  | ||||||
|  |  | ||||||
| **Type:** `object` | **Type:** `object` | ||||||
|  |  | ||||||
| @ -181,17 +180,10 @@ A plane. | |||||||
| | Property | Type | Description | Required | | | Property | Type | Description | Required | | ||||||
| |----------|------|-------------|----------| | |----------|------|-------------|----------| | ||||||
| | `type` |enum: [`Plane`](/docs/kcl/types/Plane)|  | No | | | `type` |enum: [`Plane`](/docs/kcl/types/Plane)|  | No | | ||||||
| | `id` |`string`| The id of the plane. | No | | | `value` |[`Plane`](/docs/kcl/types/Plane)| Any KCL value. | No | | ||||||
| | `value` |[`PlaneType`](/docs/kcl/types/PlaneType)| Any KCL value. | No | |  | ||||||
| | `origin` |[`Point3d`](/docs/kcl/types/Point3d)| Origin of the plane. | No | |  | ||||||
| | `xAxis` |[`Point3d`](/docs/kcl/types/Point3d)| What should the plane’s X axis be? | No | |  | ||||||
| | `yAxis` |[`Point3d`](/docs/kcl/types/Point3d)| What should the plane’s Y axis be? | No | |  | ||||||
| | `zAxis` |[`Point3d`](/docs/kcl/types/Point3d)| The z-axis (normal). | No | |  | ||||||
| | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`|  | No | |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ---- | ---- | ||||||
| A face. |  | ||||||
|  |  | ||||||
| **Type:** `object` | **Type:** `object` | ||||||
|  |  | ||||||
| @ -203,14 +195,8 @@ A face. | |||||||
|  |  | ||||||
| | Property | Type | Description | Required | | | Property | Type | Description | Required | | ||||||
| |----------|------|-------------|----------| | |----------|------|-------------|----------| | ||||||
| | `type` |enum: `Face`|  | No | | | `type` |enum: [`Face`](/docs/kcl/types/Face)|  | No | | ||||||
| | `id` |`string`| The id of the face. | No | | | `value` |[`Face`](/docs/kcl/types/Face)| Any KCL value. | No | | ||||||
| | `value` |`string`| The tag of the face. | No | |  | ||||||
| | `xAxis` |[`Point3d`](/docs/kcl/types/Point3d)| What should the face’s X axis be? | No | |  | ||||||
| | `yAxis` |[`Point3d`](/docs/kcl/types/Point3d)| What should the face’s Y axis be? | No | |  | ||||||
| | `zAxis` |[`Point3d`](/docs/kcl/types/Point3d)| The z-axis (normal). | No | |  | ||||||
| | `solid` |[`Solid`](/docs/kcl/types/Solid)| The solid the face is on. | No | |  | ||||||
| | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`|  | No | |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ---- | ---- | ||||||
| @ -246,7 +232,6 @@ A face. | |||||||
|  |  | ||||||
|  |  | ||||||
| ---- | ---- | ||||||
| An solid is a collection of extrude surfaces. |  | ||||||
|  |  | ||||||
| **Type:** `object` | **Type:** `object` | ||||||
|  |  | ||||||
| @ -259,14 +244,7 @@ An solid is a collection of extrude surfaces. | |||||||
| | Property | Type | Description | Required | | | Property | Type | Description | Required | | ||||||
| |----------|------|-------------|----------| | |----------|------|-------------|----------| | ||||||
| | `type` |enum: [`Solid`](/docs/kcl/types/Solid)|  | No | | | `type` |enum: [`Solid`](/docs/kcl/types/Solid)|  | No | | ||||||
| | `id` |`string`| The id of the solid. | No | | | `value` |[`Solid`](/docs/kcl/types/Solid)| Any KCL value. | No | | ||||||
| | `value` |`[` [`ExtrudeSurface`](/docs/kcl/types/ExtrudeSurface) `]`| The extrude surfaces. | No | |  | ||||||
| | `sketch` |[`Sketch`](/docs/kcl/types/Sketch)| The sketch. | No | |  | ||||||
| | `height` |`number`| The height of the solid. | No | |  | ||||||
| | `startCapId` |`string`| The id of the extrusion start cap | No | |  | ||||||
| | `endCapId` |`string`| The id of the extrusion end cap | No | |  | ||||||
| | `edgeCuts` |`[` [`EdgeCut`](/docs/kcl/types/EdgeCut) `]`| Chamfers or fillets on this solid. | No | |  | ||||||
| | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`| Metadata. | No | |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ---- | ---- | ||||||
| @ -286,7 +264,6 @@ An solid is a collection of extrude surfaces. | |||||||
|  |  | ||||||
|  |  | ||||||
| ---- | ---- | ||||||
| A helix. |  | ||||||
|  |  | ||||||
| **Type:** `object` | **Type:** `object` | ||||||
|  |  | ||||||
| @ -299,11 +276,7 @@ A helix. | |||||||
| | Property | Type | Description | Required | | | Property | Type | Description | Required | | ||||||
| |----------|------|-------------|----------| | |----------|------|-------------|----------| | ||||||
| | `type` |enum: [`Helix`](/docs/kcl/types/Helix)|  | No | | | `type` |enum: [`Helix`](/docs/kcl/types/Helix)|  | No | | ||||||
| | `value` |`string`| The id of the helix. | No | | | `value` |[`Helix`](/docs/kcl/types/Helix)| Any KCL value. | No | | ||||||
| | `revolutions` |`number`| Number of revolutions. | No | |  | ||||||
| | `angleStart` |`number`| Start angle (in degrees). | No | |  | ||||||
| | `ccw` |`boolean`| Is the helix rotation counter clockwise? | No | |  | ||||||
| | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`|  | No | |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ---- | ---- | ||||||
|  | |||||||
| @ -22,6 +22,7 @@ A plane. | |||||||
| | `xAxis` |[`Point3d`](/docs/kcl/types/Point3d)| What should the plane’s X axis be? | No | | | `xAxis` |[`Point3d`](/docs/kcl/types/Point3d)| What should the plane’s X axis be? | No | | ||||||
| | `yAxis` |[`Point3d`](/docs/kcl/types/Point3d)| What should the plane’s Y axis be? | No | | | `yAxis` |[`Point3d`](/docs/kcl/types/Point3d)| What should the plane’s Y axis be? | No | | ||||||
| | `zAxis` |[`Point3d`](/docs/kcl/types/Point3d)| The z-axis (normal). | No | | | `zAxis` |[`Point3d`](/docs/kcl/types/Point3d)| The z-axis (normal). | No | | ||||||
|  | | `units` |[`UnitLen`](/docs/kcl/types/UnitLen)| A plane. | No | | ||||||
| | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`|  | No | | | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`|  | No | | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
| @ -21,6 +21,7 @@ A sketch is a collection of paths. | |||||||
| | `on` |[`SketchSurface`](/docs/kcl/types/SketchSurface)| What the sketch is on (can be a plane or a face). | No | | | `on` |[`SketchSurface`](/docs/kcl/types/SketchSurface)| What the sketch is on (can be a plane or a face). | No | | ||||||
| | `start` |[`BasePath`](/docs/kcl/types/BasePath)| The starting path. | No | | | `start` |[`BasePath`](/docs/kcl/types/BasePath)| The starting path. | No | | ||||||
| | `tags` |`object`| Tag identifiers that have been declared in this sketch. | No | | | `tags` |`object`| Tag identifiers that have been declared in this sketch. | No | | ||||||
|  | | `units` |[`UnitLen`](/docs/kcl/types/UnitLen)| A sketch is a collection of paths. | No | | ||||||
| | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`| Metadata. | No | | | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`| Metadata. | No | | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
| @ -30,6 +30,7 @@ A sketch is a collection of paths. | |||||||
| | `on` |[`SketchSurface`](/docs/kcl/types/SketchSurface)| What the sketch is on (can be a plane or a face). | No | | | `on` |[`SketchSurface`](/docs/kcl/types/SketchSurface)| What the sketch is on (can be a plane or a face). | No | | ||||||
| | `start` |[`BasePath`](/docs/kcl/types/BasePath)| The starting path. | No | | | `start` |[`BasePath`](/docs/kcl/types/BasePath)| The starting path. | No | | ||||||
| | `tags` |`object`| Tag identifiers that have been declared in this sketch. | No | | | `tags` |`object`| Tag identifiers that have been declared in this sketch. | No | | ||||||
|  | | `units` |[`UnitLen`](/docs/kcl/types/UnitLen)| A sketch or a group of sketches. | No | | ||||||
| | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`| Metadata. | No | | | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`| Metadata. | No | | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
| @ -31,6 +31,7 @@ A plane. | |||||||
| | `xAxis` |[`Point3d`](/docs/kcl/types/Point3d)| What should the plane’s X axis be? | No | | | `xAxis` |[`Point3d`](/docs/kcl/types/Point3d)| What should the plane’s X axis be? | No | | ||||||
| | `yAxis` |[`Point3d`](/docs/kcl/types/Point3d)| What should the plane’s Y axis be? | No | | | `yAxis` |[`Point3d`](/docs/kcl/types/Point3d)| What should the plane’s Y axis be? | No | | ||||||
| | `zAxis` |[`Point3d`](/docs/kcl/types/Point3d)| The z-axis (normal). | No | | | `zAxis` |[`Point3d`](/docs/kcl/types/Point3d)| The z-axis (normal). | No | | ||||||
|  | | `units` |[`UnitLen`](/docs/kcl/types/UnitLen)| A sketch type. | No | | ||||||
| | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`|  | No | | | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`|  | No | | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -54,6 +55,7 @@ A face. | |||||||
| | `yAxis` |[`Point3d`](/docs/kcl/types/Point3d)| What should the face’s Y axis be? | No | | | `yAxis` |[`Point3d`](/docs/kcl/types/Point3d)| What should the face’s Y axis be? | No | | ||||||
| | `zAxis` |[`Point3d`](/docs/kcl/types/Point3d)| The z-axis (normal). | No | | | `zAxis` |[`Point3d`](/docs/kcl/types/Point3d)| The z-axis (normal). | No | | ||||||
| | `solid` |[`Solid`](/docs/kcl/types/Solid)| The solid the face is on. | No | | | `solid` |[`Solid`](/docs/kcl/types/Solid)| The solid the face is on. | No | | ||||||
|  | | `units` |[`UnitLen`](/docs/kcl/types/UnitLen)| A sketch type. | No | | ||||||
| | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`|  | No | | | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`|  | No | | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
| @ -23,6 +23,7 @@ An solid is a collection of extrude surfaces. | |||||||
| | `startCapId` |`string`| The id of the extrusion start cap | No | | | `startCapId` |`string`| The id of the extrusion start cap | No | | ||||||
| | `endCapId` |`string`| The id of the extrusion end cap | No | | | `endCapId` |`string`| The id of the extrusion end cap | No | | ||||||
| | `edgeCuts` |`[` [`EdgeCut`](/docs/kcl/types/EdgeCut) `]`| Chamfers or fillets on this solid. | No | | | `edgeCuts` |`[` [`EdgeCut`](/docs/kcl/types/EdgeCut) `]`| Chamfers or fillets on this solid. | No | | ||||||
|  | | `units` |[`UnitLen`](/docs/kcl/types/UnitLen)| An solid is a collection of extrude surfaces. | No | | ||||||
| | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`| Metadata. | No | | | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`| Metadata. | No | | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
| @ -32,6 +32,7 @@ An solid is a collection of extrude surfaces. | |||||||
| | `startCapId` |`string`| The id of the extrusion start cap | No | | | `startCapId` |`string`| The id of the extrusion start cap | No | | ||||||
| | `endCapId` |`string`| The id of the extrusion end cap | No | | | `endCapId` |`string`| The id of the extrusion end cap | No | | ||||||
| | `edgeCuts` |`[` [`EdgeCut`](/docs/kcl/types/EdgeCut) `]`| Chamfers or fillets on this solid. | No | | | `edgeCuts` |`[` [`EdgeCut`](/docs/kcl/types/EdgeCut) `]`| Chamfers or fillets on this solid. | No | | ||||||
|  | | `units` |[`UnitLen`](/docs/kcl/types/UnitLen)| A solid or a group of solids. | No | | ||||||
| | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`| Metadata. | No | | | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`| Metadata. | No | | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										107
									
								
								docs/kcl/types/UnitLen.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,107 @@ | |||||||
|  | --- | ||||||
|  | title: "UnitLen" | ||||||
|  | excerpt: "" | ||||||
|  | layout: manual | ||||||
|  | --- | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | **This schema accepts exactly one of the following:** | ||||||
|  |  | ||||||
|  |  | ||||||
|  | **Type:** `object` | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## Properties | ||||||
|  |  | ||||||
|  | | Property | Type | Description | Required | | ||||||
|  | |----------|------|-------------|----------| | ||||||
|  | | `type` |enum: `Mm`|  | No | | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ---- | ||||||
|  |  | ||||||
|  | **Type:** `object` | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## Properties | ||||||
|  |  | ||||||
|  | | Property | Type | Description | Required | | ||||||
|  | |----------|------|-------------|----------| | ||||||
|  | | `type` |enum: `Cm`|  | No | | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ---- | ||||||
|  |  | ||||||
|  | **Type:** `object` | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## Properties | ||||||
|  |  | ||||||
|  | | Property | Type | Description | Required | | ||||||
|  | |----------|------|-------------|----------| | ||||||
|  | | `type` |enum: `M`|  | No | | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ---- | ||||||
|  |  | ||||||
|  | **Type:** `object` | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## Properties | ||||||
|  |  | ||||||
|  | | Property | Type | Description | Required | | ||||||
|  | |----------|------|-------------|----------| | ||||||
|  | | `type` |enum: `Inches`|  | No | | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ---- | ||||||
|  |  | ||||||
|  | **Type:** `object` | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## Properties | ||||||
|  |  | ||||||
|  | | Property | Type | Description | Required | | ||||||
|  | |----------|------|-------------|----------| | ||||||
|  | | `type` |enum: `Feet`|  | No | | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ---- | ||||||
|  |  | ||||||
|  | **Type:** `object` | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## Properties | ||||||
|  |  | ||||||
|  | | Property | Type | Description | Required | | ||||||
|  | |----------|------|-------------|----------| | ||||||
|  | | `type` |enum: `Yards`|  | No | | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ---- | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -280,7 +280,7 @@ test( | |||||||
|  |  | ||||||
|       await expect(page.getByRole('link', { name: 'bracket' })).toBeVisible() |       await expect(page.getByRole('link', { name: 'bracket' })).toBeVisible() | ||||||
|       await expect(page.getByText('router-template-slate')).toBeVisible() |       await expect(page.getByText('router-template-slate')).toBeVisible() | ||||||
|       await expect(page.getByText('New Project')).toBeVisible() |       await expect(page.getByText('Create project')).toBeVisible() | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     await test.step('Opening the router-template project should load', async () => { |     await test.step('Opening the router-template project should load', async () => { | ||||||
|  | |||||||
| @ -45,46 +45,6 @@ test.describe('Command bar tests', () => { | |||||||
|     ) |     ) | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   // TODO: fix this test after the electron migration |  | ||||||
|   test.fixme('Fillet from command bar', async ({ page, homePage }) => { |  | ||||||
|     await page.addInitScript(async () => { |  | ||||||
|       localStorage.setItem( |  | ||||||
|         'persistCode', |  | ||||||
|         `sketch001 = startSketchOn('XY') |  | ||||||
|     |> startProfileAt([-5, -5], %) |  | ||||||
|     |> line([0, 10], %) |  | ||||||
|     |> line([10, 0], %) |  | ||||||
|     |> line([0, -10], %) |  | ||||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) |  | ||||||
|     |> close(%) |  | ||||||
|   extrude001 = extrude(-10, sketch001)` |  | ||||||
|       ) |  | ||||||
|     }) |  | ||||||
|  |  | ||||||
|     const u = await getUtils(page) |  | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |  | ||||||
|     await homePage.goToModelingScene() |  | ||||||
|     await u.openDebugPanel() |  | ||||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') |  | ||||||
|     await u.closeDebugPanel() |  | ||||||
|  |  | ||||||
|     const selectSegment = () => page.getByText(`line([0, -10], %)`).click() |  | ||||||
|  |  | ||||||
|     await selectSegment() |  | ||||||
|     await page.waitForTimeout(100) |  | ||||||
|     await page.getByRole('button', { name: 'Fillet' }).click() |  | ||||||
|     await page.waitForTimeout(100) |  | ||||||
|     await page.keyboard.press('Enter') // skip selection |  | ||||||
|     await page.waitForTimeout(100) |  | ||||||
|     await page.keyboard.press('Enter') // accept default radius |  | ||||||
|     await page.waitForTimeout(100) |  | ||||||
|     await page.keyboard.press('Enter') // submit |  | ||||||
|     await page.waitForTimeout(100) |  | ||||||
|     await expect(page.locator('.cm-activeLine')).toContainText( |  | ||||||
|       `fillet({ radius = ${KCL_DEFAULT_LENGTH}, tags = [seg01] }, %)` |  | ||||||
|     ) |  | ||||||
|   }) |  | ||||||
|  |  | ||||||
|   test('Command bar can change a setting, and switch back and forth between arguments', async ({ |   test('Command bar can change a setting, and switch back and forth between arguments', async ({ | ||||||
|     page, |     page, | ||||||
|     homePage, |     homePage, | ||||||
|  | |||||||
| @ -38,14 +38,14 @@ test.describe('Debug pane', () => { | |||||||
|       // Set the code in the code editor. |       // Set the code in the code editor. | ||||||
|       await u.codeLocator.click() |       await u.codeLocator.click() | ||||||
|       await page.keyboard.type(code, { delay: 0 }) |       await page.keyboard.type(code, { delay: 0 }) | ||||||
|       // Scroll to the feature tree. |       // Scroll to the artifact graph. | ||||||
|       await tree.scrollIntoViewIfNeeded() |       await tree.scrollIntoViewIfNeeded() | ||||||
|       // Expand the feature tree. |       // Expand the artifact graph. | ||||||
|       await tree.getByText('Feature Tree').click() |       await tree.getByText('Artifact Graph').click() | ||||||
|       // Just expanded the details, making the element taller, so scroll again. |       // Just expanded the details, making the element taller, so scroll again. | ||||||
|       await tree.getByText('Plane').first().scrollIntoViewIfNeeded() |       await tree.getByText('Plane').first().scrollIntoViewIfNeeded() | ||||||
|     }) |     }) | ||||||
|     // Extract the artifact IDs from the debug feature tree. |     // Extract the artifact IDs from the debug artifact graph. | ||||||
|     const initialSegmentIds = await segment.innerText({ timeout: 5_000 }) |     const initialSegmentIds = await segment.innerText({ timeout: 5_000 }) | ||||||
|     // The artifact ID should include a UUID. |     // The artifact ID should include a UUID. | ||||||
|     expect(initialSegmentIds).toMatch( |     expect(initialSegmentIds).toMatch( | ||||||
|  | |||||||
| @ -135,4 +135,20 @@ export class CmdBarFixture { | |||||||
|       await promptEditCommand.first().click() |       await promptEditCommand.first().click() | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   get cmdSearchInput() { | ||||||
|  |     return this.page.getByTestId('cmd-bar-search') | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   get argumentInput() { | ||||||
|  |     return this.page.getByTestId('cmd-bar-arg-value') | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   get cmdOptions() { | ||||||
|  |     return this.page.getByTestId('cmd-bar-option') | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   chooseCommand = async (commandName: string) => { | ||||||
|  |     await this.cmdOptions.getByText(commandName).click() | ||||||
|  |   } | ||||||
| } | } | ||||||
|  | |||||||
| @ -103,7 +103,7 @@ export class HomePageFixture { | |||||||
|       .toEqual(expectedState) |       .toEqual(expectedState) | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   createAndGoToProject = async (projectTitle: string) => { |   createAndGoToProject = async (projectTitle = 'project-$nnn') => { | ||||||
|     await expect(this.projectSection).not.toHaveText('Loading your Projects...') |     await expect(this.projectSection).not.toHaveText('Loading your Projects...') | ||||||
|     await this.projectButtonNew.click() |     await this.projectButtonNew.click() | ||||||
|     await this.projectTextName.click() |     await this.projectTextName.click() | ||||||
|  | |||||||
| @ -15,6 +15,8 @@ export class ToolbarFixture { | |||||||
|   extrudeButton!: Locator |   extrudeButton!: Locator | ||||||
|   loftButton!: Locator |   loftButton!: Locator | ||||||
|   sweepButton!: Locator |   sweepButton!: Locator | ||||||
|  |   filletButton!: Locator | ||||||
|  |   chamferButton!: Locator | ||||||
|   shellButton!: Locator |   shellButton!: Locator | ||||||
|   offsetPlaneButton!: Locator |   offsetPlaneButton!: Locator | ||||||
|   startSketchBtn!: Locator |   startSketchBtn!: Locator | ||||||
| @ -42,6 +44,8 @@ export class ToolbarFixture { | |||||||
|     this.extrudeButton = page.getByTestId('extrude') |     this.extrudeButton = page.getByTestId('extrude') | ||||||
|     this.loftButton = page.getByTestId('loft') |     this.loftButton = page.getByTestId('loft') | ||||||
|     this.sweepButton = page.getByTestId('sweep') |     this.sweepButton = page.getByTestId('sweep') | ||||||
|  |     this.filletButton = page.getByTestId('fillet3d') | ||||||
|  |     this.chamferButton = page.getByTestId('chamfer3d') | ||||||
|     this.shellButton = page.getByTestId('shell') |     this.shellButton = page.getByTestId('shell') | ||||||
|     this.offsetPlaneButton = page.getByTestId('plane-offset') |     this.offsetPlaneButton = page.getByTestId('plane-offset') | ||||||
|     this.startSketchBtn = page.getByTestId('sketch') |     this.startSketchBtn = page.getByTestId('sketch') | ||||||
| @ -59,6 +63,10 @@ export class ToolbarFixture { | |||||||
|     this.exeIndicator = page.getByTestId('model-state-indicator-execution-done') |     this.exeIndicator = page.getByTestId('model-state-indicator-execution-done') | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   get logoLink() { | ||||||
|  |     return this.page.getByTestId('app-logo') | ||||||
|  |   } | ||||||
|  |  | ||||||
|   startSketchPlaneSelection = async () => |   startSketchPlaneSelection = async () => | ||||||
|     doAndWaitForImageDiff(this.page, () => this.startSketchBtn.click(), 500) |     doAndWaitForImageDiff(this.page, () => this.startSketchBtn.click(), 500) | ||||||
|  |  | ||||||
|  | |||||||
| @ -829,12 +829,6 @@ loftPointAndClickCases.forEach(({ shouldPreselect }) => { | |||||||
|         }) |         }) | ||||||
|         await selectSketches() |         await selectSketches() | ||||||
|         await cmdBar.progressCmdBar() |         await cmdBar.progressCmdBar() | ||||||
|         await cmdBar.expectState({ |  | ||||||
|           stage: 'review', |  | ||||||
|           headerArguments: { Selection: '2 faces' }, |  | ||||||
|           commandName: 'Loft', |  | ||||||
|         }) |  | ||||||
|         await cmdBar.progressCmdBar() |  | ||||||
|       }) |       }) | ||||||
|     } else { |     } else { | ||||||
|       await test.step(`Preselect the two sketches`, async () => { |       await test.step(`Preselect the two sketches`, async () => { | ||||||
| @ -844,12 +838,6 @@ loftPointAndClickCases.forEach(({ shouldPreselect }) => { | |||||||
|       await test.step(`Go through the command bar flow with preselected sketches`, async () => { |       await test.step(`Go through the command bar flow with preselected sketches`, async () => { | ||||||
|         await toolbar.loftButton.click() |         await toolbar.loftButton.click() | ||||||
|         await cmdBar.progressCmdBar() |         await cmdBar.progressCmdBar() | ||||||
|         await cmdBar.expectState({ |  | ||||||
|           stage: 'review', |  | ||||||
|           headerArguments: { Selection: '2 faces' }, |  | ||||||
|           commandName: 'Loft', |  | ||||||
|         }) |  | ||||||
|         await cmdBar.progressCmdBar() |  | ||||||
|       }) |       }) | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @ -975,37 +963,31 @@ sketch002 = startSketchOn('XZ') | |||||||
|     await toolbar.sweepButton.click() |     await toolbar.sweepButton.click() | ||||||
|     await cmdBar.expectState({ |     await cmdBar.expectState({ | ||||||
|       commandName: 'Sweep', |       commandName: 'Sweep', | ||||||
|       currentArgKey: 'profile', |       currentArgKey: 'target', | ||||||
|       currentArgValue: '', |       currentArgValue: '', | ||||||
|       headerArguments: { |       headerArguments: { | ||||||
|         Path: '', |         Target: '', | ||||||
|         Profile: '', |         Trajectory: '', | ||||||
|       }, |       }, | ||||||
|       highlightedHeaderArg: 'profile', |       highlightedHeaderArg: 'target', | ||||||
|       stage: 'arguments', |       stage: 'arguments', | ||||||
|     }) |     }) | ||||||
|     await clickOnSketch1() |     await clickOnSketch1() | ||||||
|     await cmdBar.expectState({ |     await cmdBar.expectState({ | ||||||
|       commandName: 'Sweep', |       commandName: 'Sweep', | ||||||
|       currentArgKey: 'path', |       currentArgKey: 'trajectory', | ||||||
|       currentArgValue: '', |       currentArgValue: '', | ||||||
|       headerArguments: { |       headerArguments: { | ||||||
|         Path: '', |         Target: '1 face', | ||||||
|         Profile: '1 face', |         Trajectory: '', | ||||||
|       }, |       }, | ||||||
|       highlightedHeaderArg: 'path', |       highlightedHeaderArg: 'trajectory', | ||||||
|       stage: 'arguments', |       stage: 'arguments', | ||||||
|     }) |     }) | ||||||
|     await clickOnSketch2() |     await clickOnSketch2() | ||||||
|     await cmdBar.expectState({ |     await page.waitForTimeout(500) | ||||||
|       commandName: 'Sweep', |  | ||||||
|       headerArguments: { |  | ||||||
|         Path: '1 face', |  | ||||||
|         Profile: '1 face', |  | ||||||
|       }, |  | ||||||
|       stage: 'review', |  | ||||||
|     }) |  | ||||||
|     await cmdBar.progressCmdBar() |     await cmdBar.progressCmdBar() | ||||||
|  |     await page.waitForTimeout(500) | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   await test.step(`Confirm code is added to the editor, scene has changed`, async () => { |   await test.step(`Confirm code is added to the editor, scene has changed`, async () => { | ||||||
| @ -1032,6 +1014,505 @@ sketch002 = startSketchOn('XZ') | |||||||
|   }) |   }) | ||||||
| }) | }) | ||||||
|  |  | ||||||
|  | test(`Sweep point-and-click failing validation`, async ({ | ||||||
|  |   context, | ||||||
|  |   page, | ||||||
|  |   homePage, | ||||||
|  |   scene, | ||||||
|  |   toolbar, | ||||||
|  |   cmdBar, | ||||||
|  | }) => { | ||||||
|  |   const initialCode = `sketch001 = startSketchOn('YZ') | ||||||
|  |   |> circle({ | ||||||
|  |        center = [0, 0], | ||||||
|  |        radius = 500 | ||||||
|  |      }, %) | ||||||
|  | sketch002 = startSketchOn('XZ') | ||||||
|  |   |> startProfileAt([0, 0], %) | ||||||
|  |   |> xLine(-500, %) | ||||||
|  |   |> lineTo([-2000, 500], %) | ||||||
|  | ` | ||||||
|  |   await context.addInitScript((initialCode) => { | ||||||
|  |     localStorage.setItem('persistCode', initialCode) | ||||||
|  |   }, initialCode) | ||||||
|  |   await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||||
|  |   await homePage.goToModelingScene() | ||||||
|  |   await scene.waitForExecutionDone() | ||||||
|  |  | ||||||
|  |   // One dumb hardcoded screen pixel value | ||||||
|  |   const testPoint = { x: 700, y: 250 } | ||||||
|  |   const [clickOnSketch1] = scene.makeMouseHelpers(testPoint.x, testPoint.y) | ||||||
|  |   const [clickOnSketch2] = scene.makeMouseHelpers(testPoint.x - 50, testPoint.y) | ||||||
|  |  | ||||||
|  |   await test.step(`Look for sketch001`, async () => { | ||||||
|  |     await toolbar.closePane('code') | ||||||
|  |     await scene.expectPixelColor([53, 53, 53], testPoint, 15) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   await test.step(`Go through the command bar flow and fail validation with a toast`, async () => { | ||||||
|  |     await toolbar.sweepButton.click() | ||||||
|  |     await cmdBar.expectState({ | ||||||
|  |       commandName: 'Sweep', | ||||||
|  |       currentArgKey: 'target', | ||||||
|  |       currentArgValue: '', | ||||||
|  |       headerArguments: { | ||||||
|  |         Target: '', | ||||||
|  |         Trajectory: '', | ||||||
|  |       }, | ||||||
|  |       highlightedHeaderArg: 'target', | ||||||
|  |       stage: 'arguments', | ||||||
|  |     }) | ||||||
|  |     await clickOnSketch1() | ||||||
|  |     await cmdBar.expectState({ | ||||||
|  |       commandName: 'Sweep', | ||||||
|  |       currentArgKey: 'trajectory', | ||||||
|  |       currentArgValue: '', | ||||||
|  |       headerArguments: { | ||||||
|  |         Target: '1 face', | ||||||
|  |         Trajectory: '', | ||||||
|  |       }, | ||||||
|  |       highlightedHeaderArg: 'trajectory', | ||||||
|  |       stage: 'arguments', | ||||||
|  |     }) | ||||||
|  |     await clickOnSketch2() | ||||||
|  |     await page.waitForTimeout(500) | ||||||
|  |     await cmdBar.progressCmdBar() | ||||||
|  |     await expect( | ||||||
|  |       page.getByText('Unable to sweep with the provided selection') | ||||||
|  |     ).toBeVisible() | ||||||
|  |   }) | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | test(`Fillet point-and-click`, async ({ | ||||||
|  |   context, | ||||||
|  |   page, | ||||||
|  |   homePage, | ||||||
|  |   scene, | ||||||
|  |   editor, | ||||||
|  |   toolbar, | ||||||
|  |   cmdBar, | ||||||
|  | }) => { | ||||||
|  |   // Code samples | ||||||
|  |   const initialCode = `sketch001 = startSketchOn('XY') | ||||||
|  |   |> startProfileAt([-12, -6], %) | ||||||
|  |   |> line([0, 12], %) | ||||||
|  |   |> line([24, 0], %) | ||||||
|  |   |> line([0, -12], %) | ||||||
|  |   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||||
|  |   |> close(%) | ||||||
|  | extrude001 = extrude(-12, sketch001) | ||||||
|  | ` | ||||||
|  |   const firstFilletDeclaration = 'fillet({ radius = 5, tags = [seg01] }, %)' | ||||||
|  |   const secondFilletDeclaration = | ||||||
|  |     'fillet({       radius = 5,       tags = [getOppositeEdge(seg01)]     }, %)' | ||||||
|  |  | ||||||
|  |   // Locators | ||||||
|  |   const firstEdgeLocation = { x: 600, y: 193 } | ||||||
|  |   const secondEdgeLocation = { x: 600, y: 383 } | ||||||
|  |   const bodyLocation = { x: 630, y: 290 } | ||||||
|  |   const [clickOnFirstEdge] = scene.makeMouseHelpers( | ||||||
|  |     firstEdgeLocation.x, | ||||||
|  |     firstEdgeLocation.y | ||||||
|  |   ) | ||||||
|  |   const [clickOnSecondEdge] = scene.makeMouseHelpers( | ||||||
|  |     secondEdgeLocation.x, | ||||||
|  |     secondEdgeLocation.y | ||||||
|  |   ) | ||||||
|  |  | ||||||
|  |   // Colors | ||||||
|  |   const edgeColorWhite: [number, number, number] = [248, 248, 248] | ||||||
|  |   const edgeColorYellow: [number, number, number] = [251, 251, 40] // Mac:B=67 Ubuntu:B=12 | ||||||
|  |   const bodyColor: [number, number, number] = [155, 155, 155] | ||||||
|  |   const filletColor: [number, number, number] = [127, 127, 127] | ||||||
|  |   const backgroundColor: [number, number, number] = [30, 30, 30] | ||||||
|  |   const lowTolerance = 20 | ||||||
|  |   const highTolerance = 40 | ||||||
|  |  | ||||||
|  |   // Setup | ||||||
|  |   await test.step(`Initial test setup`, async () => { | ||||||
|  |     await context.addInitScript((initialCode) => { | ||||||
|  |       localStorage.setItem('persistCode', initialCode) | ||||||
|  |     }, initialCode) | ||||||
|  |     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||||
|  |     await homePage.goToModelingScene() | ||||||
|  |  | ||||||
|  |     // verify modeling scene is loaded | ||||||
|  |     await scene.expectPixelColor( | ||||||
|  |       backgroundColor, | ||||||
|  |       secondEdgeLocation, | ||||||
|  |       lowTolerance | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     // wait for stream to load | ||||||
|  |     await scene.expectPixelColor(bodyColor, bodyLocation, highTolerance) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   // Test 1: Command bar flow with preselected edges | ||||||
|  |   await test.step(`Select first edge`, async () => { | ||||||
|  |     await scene.expectPixelColor( | ||||||
|  |       edgeColorWhite, | ||||||
|  |       firstEdgeLocation, | ||||||
|  |       lowTolerance | ||||||
|  |     ) | ||||||
|  |     await clickOnFirstEdge() | ||||||
|  |     await scene.expectPixelColor( | ||||||
|  |       edgeColorYellow, | ||||||
|  |       firstEdgeLocation, | ||||||
|  |       highTolerance // Ubuntu color mismatch can require high tolerance | ||||||
|  |     ) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   await test.step(`Apply fillet to the preselected edge`, async () => { | ||||||
|  |     await page.waitForTimeout(100) | ||||||
|  |     await toolbar.filletButton.click() | ||||||
|  |     await cmdBar.expectState({ | ||||||
|  |       commandName: 'Fillet', | ||||||
|  |       highlightedHeaderArg: 'selection', | ||||||
|  |       currentArgKey: 'selection', | ||||||
|  |       currentArgValue: '', | ||||||
|  |       headerArguments: { | ||||||
|  |         Selection: '', | ||||||
|  |         Radius: '', | ||||||
|  |       }, | ||||||
|  |       stage: 'arguments', | ||||||
|  |     }) | ||||||
|  |     await cmdBar.progressCmdBar() | ||||||
|  |     await cmdBar.expectState({ | ||||||
|  |       commandName: 'Fillet', | ||||||
|  |       highlightedHeaderArg: 'radius', | ||||||
|  |       currentArgKey: 'radius', | ||||||
|  |       currentArgValue: '5', | ||||||
|  |       headerArguments: { | ||||||
|  |         Selection: '1 face', | ||||||
|  |         Radius: '', | ||||||
|  |       }, | ||||||
|  |       stage: 'arguments', | ||||||
|  |     }) | ||||||
|  |     await cmdBar.progressCmdBar() | ||||||
|  |     await cmdBar.expectState({ | ||||||
|  |       commandName: 'Fillet', | ||||||
|  |       headerArguments: { | ||||||
|  |         Selection: '1 face', | ||||||
|  |         Radius: '5', | ||||||
|  |       }, | ||||||
|  |       stage: 'review', | ||||||
|  |     }) | ||||||
|  |     await cmdBar.progressCmdBar() | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   await test.step(`Confirm code is added to the editor`, async () => { | ||||||
|  |     await editor.expectEditor.toContain(firstFilletDeclaration) | ||||||
|  |     await editor.expectState({ | ||||||
|  |       diagnostics: [], | ||||||
|  |       activeLines: ['|>fillet({radius=5,tags=[seg01]},%)'], | ||||||
|  |       highlightedCode: '', | ||||||
|  |     }) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   await test.step(`Confirm scene has changed`, async () => { | ||||||
|  |     await scene.expectPixelColor(filletColor, firstEdgeLocation, lowTolerance) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   // Test 2: Command bar flow without preselected edges | ||||||
|  |   await test.step(`Open fillet UI without selecting edges`, async () => { | ||||||
|  |     await page.waitForTimeout(100) | ||||||
|  |     await toolbar.filletButton.click() | ||||||
|  |     await cmdBar.expectState({ | ||||||
|  |       stage: 'arguments', | ||||||
|  |       currentArgKey: 'selection', | ||||||
|  |       currentArgValue: '', | ||||||
|  |       headerArguments: { | ||||||
|  |         Selection: '', | ||||||
|  |         Radius: '', | ||||||
|  |       }, | ||||||
|  |       highlightedHeaderArg: 'selection', | ||||||
|  |       commandName: 'Fillet', | ||||||
|  |     }) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   await test.step(`Select second edge`, async () => { | ||||||
|  |     await scene.expectPixelColor( | ||||||
|  |       edgeColorWhite, | ||||||
|  |       secondEdgeLocation, | ||||||
|  |       lowTolerance | ||||||
|  |     ) | ||||||
|  |     await clickOnSecondEdge() | ||||||
|  |     await scene.expectPixelColor( | ||||||
|  |       edgeColorYellow, | ||||||
|  |       secondEdgeLocation, | ||||||
|  |       highTolerance // Ubuntu color mismatch can require high tolerance | ||||||
|  |     ) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   await test.step(`Apply fillet to the second edge`, async () => { | ||||||
|  |     await cmdBar.expectState({ | ||||||
|  |       commandName: 'Fillet', | ||||||
|  |       highlightedHeaderArg: 'selection', | ||||||
|  |       currentArgKey: 'selection', | ||||||
|  |       currentArgValue: '', | ||||||
|  |       headerArguments: { | ||||||
|  |         Selection: '', | ||||||
|  |         Radius: '', | ||||||
|  |       }, | ||||||
|  |       stage: 'arguments', | ||||||
|  |     }) | ||||||
|  |     await cmdBar.progressCmdBar() | ||||||
|  |     await cmdBar.expectState({ | ||||||
|  |       commandName: 'Fillet', | ||||||
|  |       highlightedHeaderArg: 'radius', | ||||||
|  |       currentArgKey: 'radius', | ||||||
|  |       currentArgValue: '5', | ||||||
|  |       headerArguments: { | ||||||
|  |         Selection: '1 sweepEdge', | ||||||
|  |         Radius: '', | ||||||
|  |       }, | ||||||
|  |       stage: 'arguments', | ||||||
|  |     }) | ||||||
|  |     await cmdBar.progressCmdBar() | ||||||
|  |     await cmdBar.expectState({ | ||||||
|  |       commandName: 'Fillet', | ||||||
|  |       headerArguments: { | ||||||
|  |         Selection: '1 sweepEdge', | ||||||
|  |         Radius: '5', | ||||||
|  |       }, | ||||||
|  |       stage: 'review', | ||||||
|  |     }) | ||||||
|  |     await cmdBar.progressCmdBar() | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   await test.step(`Confirm code is added to the editor`, async () => { | ||||||
|  |     await editor.expectEditor.toContain(secondFilletDeclaration) | ||||||
|  |     await editor.expectState({ | ||||||
|  |       diagnostics: [], | ||||||
|  |       activeLines: ['radius=5,'], | ||||||
|  |       highlightedCode: '', | ||||||
|  |     }) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   await test.step(`Confirm scene has changed`, async () => { | ||||||
|  |     await scene.expectPixelColor( | ||||||
|  |       backgroundColor, | ||||||
|  |       secondEdgeLocation, | ||||||
|  |       lowTolerance | ||||||
|  |     ) | ||||||
|  |   }) | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | test(`Chamfer point-and-click`, async ({ | ||||||
|  |   context, | ||||||
|  |   page, | ||||||
|  |   homePage, | ||||||
|  |   scene, | ||||||
|  |   editor, | ||||||
|  |   toolbar, | ||||||
|  |   cmdBar, | ||||||
|  | }) => { | ||||||
|  |   // Code samples | ||||||
|  |   const initialCode = `sketch001 = startSketchOn('XY') | ||||||
|  |   |> startProfileAt([-12, -6], %) | ||||||
|  |   |> line([0, 12], %) | ||||||
|  |   |> line([24, 0], %) | ||||||
|  |   |> line([0, -12], %) | ||||||
|  |   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||||
|  |   |> close(%) | ||||||
|  | extrude001 = extrude(-12, sketch001) | ||||||
|  | ` | ||||||
|  |   const firstChamferDeclaration = 'chamfer({ length = 5, tags = [seg01] }, %)' | ||||||
|  |   const secondChamferDeclaration = | ||||||
|  |     'chamfer({       length = 5,       tags = [getOppositeEdge(seg01)]     }, %)' | ||||||
|  |  | ||||||
|  |   // Locators | ||||||
|  |   const firstEdgeLocation = { x: 600, y: 193 } | ||||||
|  |   const secondEdgeLocation = { x: 600, y: 383 } | ||||||
|  |   const bodyLocation = { x: 630, y: 290 } | ||||||
|  |   const [clickOnFirstEdge] = scene.makeMouseHelpers( | ||||||
|  |     firstEdgeLocation.x, | ||||||
|  |     firstEdgeLocation.y | ||||||
|  |   ) | ||||||
|  |   const [clickOnSecondEdge] = scene.makeMouseHelpers( | ||||||
|  |     secondEdgeLocation.x, | ||||||
|  |     secondEdgeLocation.y | ||||||
|  |   ) | ||||||
|  |  | ||||||
|  |   // Colors | ||||||
|  |   const edgeColorWhite: [number, number, number] = [248, 248, 248] | ||||||
|  |   const edgeColorYellow: [number, number, number] = [251, 251, 40] // Mac:B=67 Ubuntu:B=12 | ||||||
|  |   const bodyColor: [number, number, number] = [155, 155, 155] | ||||||
|  |   const chamferColor: [number, number, number] = [168, 168, 168] | ||||||
|  |   const backgroundColor: [number, number, number] = [30, 30, 30] | ||||||
|  |   const lowTolerance = 20 | ||||||
|  |   const highTolerance = 40 | ||||||
|  |  | ||||||
|  |   // Setup | ||||||
|  |   await test.step(`Initial test setup`, async () => { | ||||||
|  |     await context.addInitScript((initialCode) => { | ||||||
|  |       localStorage.setItem('persistCode', initialCode) | ||||||
|  |     }, initialCode) | ||||||
|  |     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||||
|  |     await homePage.goToModelingScene() | ||||||
|  |  | ||||||
|  |     // verify modeling scene is loaded | ||||||
|  |     await scene.expectPixelColor( | ||||||
|  |       backgroundColor, | ||||||
|  |       secondEdgeLocation, | ||||||
|  |       lowTolerance | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     // wait for stream to load | ||||||
|  |     await scene.expectPixelColor(bodyColor, bodyLocation, highTolerance) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   // Test 1: Command bar flow with preselected edges | ||||||
|  |   await test.step(`Select first edge`, async () => { | ||||||
|  |     await scene.expectPixelColor( | ||||||
|  |       edgeColorWhite, | ||||||
|  |       firstEdgeLocation, | ||||||
|  |       lowTolerance | ||||||
|  |     ) | ||||||
|  |     await clickOnFirstEdge() | ||||||
|  |     await scene.expectPixelColor( | ||||||
|  |       edgeColorYellow, | ||||||
|  |       firstEdgeLocation, | ||||||
|  |       highTolerance // Ubuntu color mismatch can require high tolerance | ||||||
|  |     ) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   await test.step(`Apply chamfer to the preselected edge`, async () => { | ||||||
|  |     await page.waitForTimeout(100) | ||||||
|  |     await toolbar.chamferButton.click() | ||||||
|  |     await cmdBar.expectState({ | ||||||
|  |       commandName: 'Chamfer', | ||||||
|  |       highlightedHeaderArg: 'selection', | ||||||
|  |       currentArgKey: 'selection', | ||||||
|  |       currentArgValue: '', | ||||||
|  |       headerArguments: { | ||||||
|  |         Selection: '', | ||||||
|  |         Length: '', | ||||||
|  |       }, | ||||||
|  |       stage: 'arguments', | ||||||
|  |     }) | ||||||
|  |     await cmdBar.progressCmdBar() | ||||||
|  |     await cmdBar.expectState({ | ||||||
|  |       commandName: 'Chamfer', | ||||||
|  |       highlightedHeaderArg: 'length', | ||||||
|  |       currentArgKey: 'length', | ||||||
|  |       currentArgValue: '5', | ||||||
|  |       headerArguments: { | ||||||
|  |         Selection: '1 face', | ||||||
|  |         Length: '', | ||||||
|  |       }, | ||||||
|  |       stage: 'arguments', | ||||||
|  |     }) | ||||||
|  |     await cmdBar.progressCmdBar() | ||||||
|  |     await cmdBar.expectState({ | ||||||
|  |       commandName: 'Chamfer', | ||||||
|  |       headerArguments: { | ||||||
|  |         Selection: '1 face', | ||||||
|  |         Length: '5', | ||||||
|  |       }, | ||||||
|  |       stage: 'review', | ||||||
|  |     }) | ||||||
|  |     await cmdBar.progressCmdBar() | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   await test.step(`Confirm code is added to the editor`, async () => { | ||||||
|  |     await editor.expectEditor.toContain(firstChamferDeclaration) | ||||||
|  |     await editor.expectState({ | ||||||
|  |       diagnostics: [], | ||||||
|  |       activeLines: ['|>chamfer({length=5,tags=[seg01]},%)'], | ||||||
|  |       highlightedCode: '', | ||||||
|  |     }) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   await test.step(`Confirm scene has changed`, async () => { | ||||||
|  |     await scene.expectPixelColor(chamferColor, firstEdgeLocation, lowTolerance) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   // Test 2: Command bar flow without preselected edges | ||||||
|  |   await test.step(`Open chamfer UI without selecting edges`, async () => { | ||||||
|  |     await page.waitForTimeout(100) | ||||||
|  |     await toolbar.chamferButton.click() | ||||||
|  |     await cmdBar.expectState({ | ||||||
|  |       stage: 'arguments', | ||||||
|  |       currentArgKey: 'selection', | ||||||
|  |       currentArgValue: '', | ||||||
|  |       headerArguments: { | ||||||
|  |         Selection: '', | ||||||
|  |         Length: '', | ||||||
|  |       }, | ||||||
|  |       highlightedHeaderArg: 'selection', | ||||||
|  |       commandName: 'Chamfer', | ||||||
|  |     }) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   await test.step(`Select second edge`, async () => { | ||||||
|  |     await scene.expectPixelColor( | ||||||
|  |       edgeColorWhite, | ||||||
|  |       secondEdgeLocation, | ||||||
|  |       lowTolerance | ||||||
|  |     ) | ||||||
|  |     await clickOnSecondEdge() | ||||||
|  |     await scene.expectPixelColor( | ||||||
|  |       edgeColorYellow, | ||||||
|  |       secondEdgeLocation, | ||||||
|  |       highTolerance // Ubuntu color mismatch can require high tolerance | ||||||
|  |     ) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   await test.step(`Apply chamfer to the second edge`, async () => { | ||||||
|  |     await cmdBar.expectState({ | ||||||
|  |       commandName: 'Chamfer', | ||||||
|  |       highlightedHeaderArg: 'selection', | ||||||
|  |       currentArgKey: 'selection', | ||||||
|  |       currentArgValue: '', | ||||||
|  |       headerArguments: { | ||||||
|  |         Selection: '', | ||||||
|  |         Length: '', | ||||||
|  |       }, | ||||||
|  |       stage: 'arguments', | ||||||
|  |     }) | ||||||
|  |     await cmdBar.progressCmdBar() | ||||||
|  |     await cmdBar.expectState({ | ||||||
|  |       commandName: 'Chamfer', | ||||||
|  |       highlightedHeaderArg: 'length', | ||||||
|  |       currentArgKey: 'length', | ||||||
|  |       currentArgValue: '5', | ||||||
|  |       headerArguments: { | ||||||
|  |         Selection: '1 sweepEdge', | ||||||
|  |         Length: '', | ||||||
|  |       }, | ||||||
|  |       stage: 'arguments', | ||||||
|  |     }) | ||||||
|  |     await cmdBar.progressCmdBar() | ||||||
|  |     await cmdBar.expectState({ | ||||||
|  |       commandName: 'Chamfer', | ||||||
|  |       headerArguments: { | ||||||
|  |         Selection: '1 sweepEdge', | ||||||
|  |         Length: '5', | ||||||
|  |       }, | ||||||
|  |       stage: 'review', | ||||||
|  |     }) | ||||||
|  |     await cmdBar.progressCmdBar() | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   await test.step(`Confirm code is added to the editor`, async () => { | ||||||
|  |     await editor.expectEditor.toContain(secondChamferDeclaration) | ||||||
|  |     await editor.expectState({ | ||||||
|  |       diagnostics: [], | ||||||
|  |       activeLines: ['length=5,'], | ||||||
|  |       highlightedCode: '', | ||||||
|  |     }) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   await test.step(`Confirm scene has changed`, async () => { | ||||||
|  |     await scene.expectPixelColor( | ||||||
|  |       backgroundColor, | ||||||
|  |       secondEdgeLocation, | ||||||
|  |       lowTolerance | ||||||
|  |     ) | ||||||
|  |   }) | ||||||
|  | }) | ||||||
|  |  | ||||||
| const shellPointAndClickCapCases = [ | const shellPointAndClickCapCases = [ | ||||||
|   { shouldPreselect: true }, |   { shouldPreselect: true }, | ||||||
|   { shouldPreselect: false }, |   { shouldPreselect: false }, | ||||||
| @ -1085,6 +1566,7 @@ shellPointAndClickCapCases.forEach(({ shouldPreselect }) => { | |||||||
|         await clickOnCap() |         await clickOnCap() | ||||||
|         await page.waitForTimeout(500) |         await page.waitForTimeout(500) | ||||||
|         await cmdBar.progressCmdBar() |         await cmdBar.progressCmdBar() | ||||||
|  |         await page.waitForTimeout(500) | ||||||
|         await cmdBar.progressCmdBar() |         await cmdBar.progressCmdBar() | ||||||
|         await cmdBar.expectState({ |         await cmdBar.expectState({ | ||||||
|           stage: 'review', |           stage: 'review', | ||||||
| @ -1105,6 +1587,7 @@ shellPointAndClickCapCases.forEach(({ shouldPreselect }) => { | |||||||
|       await test.step(`Go through the command bar flow with a preselected face (cap)`, async () => { |       await test.step(`Go through the command bar flow with a preselected face (cap)`, async () => { | ||||||
|         await toolbar.shellButton.click() |         await toolbar.shellButton.click() | ||||||
|         await cmdBar.progressCmdBar() |         await cmdBar.progressCmdBar() | ||||||
|  |         await page.waitForTimeout(500) | ||||||
|         await cmdBar.progressCmdBar() |         await cmdBar.progressCmdBar() | ||||||
|         await cmdBar.expectState({ |         await cmdBar.expectState({ | ||||||
|           stage: 'review', |           stage: 'review', | ||||||
| @ -1186,6 +1669,7 @@ extrude001 = extrude(40, sketch001) | |||||||
|     await page.waitForTimeout(500) |     await page.waitForTimeout(500) | ||||||
|     await page.keyboard.up('Shift') |     await page.keyboard.up('Shift') | ||||||
|     await cmdBar.progressCmdBar() |     await cmdBar.progressCmdBar() | ||||||
|  |     await page.waitForTimeout(500) | ||||||
|     await cmdBar.progressCmdBar() |     await cmdBar.progressCmdBar() | ||||||
|     await cmdBar.expectState({ |     await cmdBar.expectState({ | ||||||
|       stage: 'review', |       stage: 'review', | ||||||
| @ -1309,3 +1793,61 @@ shellSketchOnFacesCases.forEach((initialCode, index) => { | |||||||
|     }) |     }) | ||||||
|   }) |   }) | ||||||
| }) | }) | ||||||
|  |  | ||||||
|  | test(`Shell dry-run validation rejects sweeps`, async ({ | ||||||
|  |   context, | ||||||
|  |   page, | ||||||
|  |   homePage, | ||||||
|  |   scene, | ||||||
|  |   editor, | ||||||
|  |   toolbar, | ||||||
|  |   cmdBar, | ||||||
|  | }) => { | ||||||
|  |   const initialCode = `sketch001 = startSketchOn('YZ') | ||||||
|  |   |> circle({ | ||||||
|  |        center = [0, 0], | ||||||
|  |        radius = 500 | ||||||
|  |      }, %) | ||||||
|  | sketch002 = startSketchOn('XZ') | ||||||
|  |   |> startProfileAt([0, 0], %) | ||||||
|  |   |> xLine(-2000, %) | ||||||
|  | sweep001 = sweep({ path = sketch002 }, sketch001) | ||||||
|  | ` | ||||||
|  |   await context.addInitScript((initialCode) => { | ||||||
|  |     localStorage.setItem('persistCode', initialCode) | ||||||
|  |   }, initialCode) | ||||||
|  |   await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||||
|  |   await homePage.goToModelingScene() | ||||||
|  |   await scene.waitForExecutionDone() | ||||||
|  |  | ||||||
|  |   // One dumb hardcoded screen pixel value | ||||||
|  |   const testPoint = { x: 500, y: 250 } | ||||||
|  |   const [clickOnSweep] = scene.makeMouseHelpers(testPoint.x, testPoint.y) | ||||||
|  |  | ||||||
|  |   await test.step(`Confirm sweep exists`, async () => { | ||||||
|  |     await toolbar.closePane('code') | ||||||
|  |     await scene.expectPixelColor([231, 231, 231], testPoint, 15) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   await test.step(`Go through the Shell flow and fail validation with a toast`, async () => { | ||||||
|  |     await toolbar.shellButton.click() | ||||||
|  |     await cmdBar.expectState({ | ||||||
|  |       stage: 'arguments', | ||||||
|  |       currentArgKey: 'selection', | ||||||
|  |       currentArgValue: '', | ||||||
|  |       headerArguments: { | ||||||
|  |         Selection: '', | ||||||
|  |         Thickness: '', | ||||||
|  |       }, | ||||||
|  |       highlightedHeaderArg: 'selection', | ||||||
|  |       commandName: 'Shell', | ||||||
|  |     }) | ||||||
|  |     await clickOnSweep() | ||||||
|  |     await page.waitForTimeout(500) | ||||||
|  |     await cmdBar.progressCmdBar() | ||||||
|  |     await expect( | ||||||
|  |       page.getByText('Unable to shell with the provided selection') | ||||||
|  |     ).toBeVisible() | ||||||
|  |     await page.waitForTimeout(1000) | ||||||
|  |   }) | ||||||
|  | }) | ||||||
|  | |||||||
| @ -172,7 +172,7 @@ test( | |||||||
|       await expect(page.getByRole('link', { name: 'bracket' })).toBeVisible() |       await expect(page.getByRole('link', { name: 'bracket' })).toBeVisible() | ||||||
|       await expect(page.getByText('broken-code')).toBeVisible() |       await expect(page.getByText('broken-code')).toBeVisible() | ||||||
|       await expect(page.getByText('bracket')).toBeVisible() |       await expect(page.getByText('bracket')).toBeVisible() | ||||||
|       await expect(page.getByText('New Project')).toBeVisible() |       await expect(page.getByText('Create project')).toBeVisible() | ||||||
|     }) |     }) | ||||||
|     await test.step('opening broken code project should clear the scene and show the error', async () => { |     await test.step('opening broken code project should clear the scene and show the error', async () => { | ||||||
|       // Go back home. |       // Go back home. | ||||||
| @ -253,7 +253,7 @@ test( | |||||||
|       await expect(page.getByRole('link', { name: 'bracket' })).toBeVisible() |       await expect(page.getByRole('link', { name: 'bracket' })).toBeVisible() | ||||||
|       await expect(page.getByText('empty')).toBeVisible() |       await expect(page.getByText('empty')).toBeVisible() | ||||||
|       await expect(page.getByText('bracket')).toBeVisible() |       await expect(page.getByText('bracket')).toBeVisible() | ||||||
|       await expect(page.getByText('New Project')).toBeVisible() |       await expect(page.getByText('Create project')).toBeVisible() | ||||||
|     }) |     }) | ||||||
|     await test.step('opening empty code project should clear the scene', async () => { |     await test.step('opening empty code project should clear the scene', async () => { | ||||||
|       // Go back home. |       // Go back home. | ||||||
| @ -985,6 +985,126 @@ test.describe(`Project management commands`, () => { | |||||||
|       }) |       }) | ||||||
|     } |     } | ||||||
|   ) |   ) | ||||||
|  |   test(`Create a new project with a colliding name`, async ({ | ||||||
|  |     context, | ||||||
|  |     homePage, | ||||||
|  |     toolbar, | ||||||
|  |     cmdBar, | ||||||
|  |   }) => { | ||||||
|  |     const projectName = 'test-project' | ||||||
|  |     await test.step(`Setup`, async () => { | ||||||
|  |       await context.folderSetupFn(async (dir) => { | ||||||
|  |         const projectDir = path.join(dir, projectName) | ||||||
|  |         await Promise.all([fsp.mkdir(projectDir, { recursive: true })]) | ||||||
|  |         await Promise.all([ | ||||||
|  |           fsp.copyFile( | ||||||
|  |             executorInputPath('router-template-slate.kcl'), | ||||||
|  |             path.join(projectDir, 'main.kcl') | ||||||
|  |           ), | ||||||
|  |         ]) | ||||||
|  |       }) | ||||||
|  |       await homePage.expectState({ | ||||||
|  |         projectCards: [ | ||||||
|  |           { | ||||||
|  |             title: projectName, | ||||||
|  |             fileCount: 1, | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |         sortBy: 'last-modified-desc', | ||||||
|  |       }) | ||||||
|  |     }) | ||||||
|  |  | ||||||
|  |     await test.step('Create a new project with the same name', async () => { | ||||||
|  |       await cmdBar.openCmdBar() | ||||||
|  |       await cmdBar.chooseCommand('create project') | ||||||
|  |       await cmdBar.expectState({ | ||||||
|  |         stage: 'arguments', | ||||||
|  |         commandName: 'Create project', | ||||||
|  |         currentArgKey: 'name', | ||||||
|  |         currentArgValue: '', | ||||||
|  |         headerArguments: { | ||||||
|  |           Name: '', | ||||||
|  |         }, | ||||||
|  |         highlightedHeaderArg: 'name', | ||||||
|  |       }) | ||||||
|  |       await cmdBar.argumentInput.fill(projectName) | ||||||
|  |       await cmdBar.progressCmdBar() | ||||||
|  |     }) | ||||||
|  |  | ||||||
|  |     await test.step(`Check the project was created with a non-colliding name`, async () => { | ||||||
|  |       await toolbar.logoLink.click() | ||||||
|  |       await homePage.expectState({ | ||||||
|  |         projectCards: [ | ||||||
|  |           { | ||||||
|  |             title: projectName + '-1', | ||||||
|  |             fileCount: 1, | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             title: projectName, | ||||||
|  |             fileCount: 1, | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |         sortBy: 'last-modified-desc', | ||||||
|  |       }) | ||||||
|  |     }) | ||||||
|  |  | ||||||
|  |     await test.step('Create another project with the same name', async () => { | ||||||
|  |       await cmdBar.openCmdBar() | ||||||
|  |       await cmdBar.chooseCommand('create project') | ||||||
|  |       await cmdBar.expectState({ | ||||||
|  |         stage: 'arguments', | ||||||
|  |         commandName: 'Create project', | ||||||
|  |         currentArgKey: 'name', | ||||||
|  |         currentArgValue: '', | ||||||
|  |         headerArguments: { | ||||||
|  |           Name: '', | ||||||
|  |         }, | ||||||
|  |         highlightedHeaderArg: 'name', | ||||||
|  |       }) | ||||||
|  |       await cmdBar.argumentInput.fill(projectName) | ||||||
|  |       await cmdBar.progressCmdBar() | ||||||
|  |     }) | ||||||
|  |  | ||||||
|  |     await test.step(`Check the second project was created with a non-colliding name`, async () => { | ||||||
|  |       await toolbar.logoLink.click() | ||||||
|  |       await homePage.expectState({ | ||||||
|  |         projectCards: [ | ||||||
|  |           { | ||||||
|  |             title: projectName + '-2', | ||||||
|  |             fileCount: 1, | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             title: projectName + '-1', | ||||||
|  |             fileCount: 1, | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             title: projectName, | ||||||
|  |             fileCount: 1, | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |         sortBy: 'last-modified-desc', | ||||||
|  |       }) | ||||||
|  |     }) | ||||||
|  |   }) | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | test(`Create a few projects using the default project name`, async ({ | ||||||
|  |   homePage, | ||||||
|  |   toolbar, | ||||||
|  | }) => { | ||||||
|  |   for (let i = 0; i < 12; i++) { | ||||||
|  |     await test.step(`Create project ${i}`, async () => { | ||||||
|  |       await homePage.expectState({ | ||||||
|  |         projectCards: Array.from({ length: i }, (_, i) => ({ | ||||||
|  |           title: `project-${i.toString().padStart(3, '0')}`, | ||||||
|  |           fileCount: 1, | ||||||
|  |         })).toReversed(), | ||||||
|  |         sortBy: 'last-modified-desc', | ||||||
|  |       }) | ||||||
|  |       await homePage.createAndGoToProject() | ||||||
|  |       await toolbar.logoLink.click() | ||||||
|  |     }) | ||||||
|  |   } | ||||||
| }) | }) | ||||||
|  |  | ||||||
| test( | test( | ||||||
| @ -1391,7 +1511,7 @@ extrude001 = extrude(200, sketch001)`) | |||||||
|     await page.getByTestId('app-logo').click() |     await page.getByTestId('app-logo').click() | ||||||
|  |  | ||||||
|     await expect( |     await expect( | ||||||
|       page.getByRole('button', { name: 'New project' }) |       page.getByRole('button', { name: 'Create project' }) | ||||||
|     ).toBeVisible() |     ).toBeVisible() | ||||||
|  |  | ||||||
|     for (let i = 1; i <= 10; i++) { |     for (let i = 1; i <= 10; i++) { | ||||||
| @ -1465,7 +1585,7 @@ test( | |||||||
|  |  | ||||||
|       await expect(page.getByRole('link', { name: 'bracket' })).toBeVisible() |       await expect(page.getByRole('link', { name: 'bracket' })).toBeVisible() | ||||||
|       await expect(page.getByText('router-template-slate')).toBeVisible() |       await expect(page.getByText('router-template-slate')).toBeVisible() | ||||||
|       await expect(page.getByText('New Project')).toBeVisible() |       await expect(page.getByText('Create project')).toBeVisible() | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     await test.step('Opening the router-template project should load the stream', async () => { |     await test.step('Opening the router-template project should load the stream', async () => { | ||||||
| @ -1494,7 +1614,7 @@ test( | |||||||
|  |  | ||||||
|       await expect(page.getByRole('link', { name: 'bracket' })).toBeVisible() |       await expect(page.getByRole('link', { name: 'bracket' })).toBeVisible() | ||||||
|       await expect(page.getByText('router-template-slate')).toBeVisible() |       await expect(page.getByText('router-template-slate')).toBeVisible() | ||||||
|       await expect(page.getByText('New Project')).toBeVisible() |       await expect(page.getByText('Create project')).toBeVisible() | ||||||
|     }) |     }) | ||||||
|   } |   } | ||||||
| ) | ) | ||||||
|  | |||||||
| Before Width: | Height: | Size: 54 KiB After Width: | Height: | Size: 54 KiB | 
| Before Width: | Height: | Size: 52 KiB After Width: | Height: | Size: 52 KiB | 
| Before Width: | Height: | Size: 59 KiB After Width: | Height: | Size: 60 KiB | 
| Before Width: | Height: | Size: 53 KiB After Width: | Height: | Size: 54 KiB | 
| Before Width: | Height: | Size: 50 KiB After Width: | Height: | Size: 44 KiB | 
| Before Width: | Height: | Size: 50 KiB After Width: | Height: | Size: 50 KiB | 
| Before Width: | Height: | Size: 66 KiB After Width: | Height: | Size: 66 KiB | 
| Before Width: | Height: | Size: 144 KiB After Width: | Height: | Size: 145 KiB | 
| Before Width: | Height: | Size: 128 KiB After Width: | Height: | Size: 129 KiB | 
| Before Width: | Height: | Size: 48 KiB After Width: | Height: | Size: 48 KiB | 
| Before Width: | Height: | Size: 53 KiB After Width: | Height: | Size: 54 KiB | 
| Before Width: | Height: | Size: 48 KiB After Width: | Height: | Size: 48 KiB | 
| Before Width: | Height: | Size: 52 KiB After Width: | Height: | Size: 53 KiB | 
| Before Width: | Height: | Size: 53 KiB After Width: | Height: | Size: 53 KiB | 
| Before Width: | Height: | Size: 47 KiB After Width: | Height: | Size: 47 KiB | 
| @ -1078,7 +1078,7 @@ export async function createProject({ | |||||||
|   returnHome?: boolean |   returnHome?: boolean | ||||||
| }) { | }) { | ||||||
|   await test.step(`Create project and navigate to it`, async () => { |   await test.step(`Create project and navigate to it`, async () => { | ||||||
|     await page.getByRole('button', { name: 'New project' }).click() |     await page.getByRole('button', { name: 'Create project' }).click() | ||||||
|     await page.getByRole('textbox', { name: 'Name' }).fill(name) |     await page.getByRole('textbox', { name: 'Name' }).fill(name) | ||||||
|     await page.getByRole('button', { name: 'Continue' }).click() |     await page.getByRole('button', { name: 'Continue' }).click() | ||||||
|  |  | ||||||
|  | |||||||
| @ -8,8 +8,8 @@ import { UnitLength_type } from '@kittycad/lib/dist/types/src/models' | |||||||
|  |  | ||||||
| test.describe('Testing in-app sample loading', () => { | test.describe('Testing in-app sample loading', () => { | ||||||
|   /** |   /** | ||||||
|    * Note this test implicitly depends on the KCL sample "car-wheel.kcl", |    * Note this test implicitly depends on the KCL sample "a-parametric-bearing-pillow-block", | ||||||
|    * its title, and its units settings. https://github.com/KittyCAD/kcl-samples/blob/main/car-wheel/car-wheel.kcl |    * its title, and its units settings. https://github.com/KittyCAD/kcl-samples/blob/main/a-parametric-bearing-pillow-block/main.kcl | ||||||
|    */ |    */ | ||||||
|   test('Web: should overwrite current code, cannot create new file', async ({ |   test('Web: should overwrite current code, cannot create new file', async ({ | ||||||
|     editor, |     editor, | ||||||
| @ -29,8 +29,8 @@ test.describe('Testing in-app sample loading', () => { | |||||||
|  |  | ||||||
|     // Locators and constants |     // Locators and constants | ||||||
|     const newSample = { |     const newSample = { | ||||||
|       file: 'car-wheel' + FILE_EXT, |       file: 'a-parametric-bearing-pillow-block' + FILE_EXT, | ||||||
|       title: 'Car Wheel', |       title: 'A Parametric Bearing Pillow Block', | ||||||
|     } |     } | ||||||
|     const commandBarButton = page.getByRole('button', { name: 'Commands' }) |     const commandBarButton = page.getByRole('button', { name: 'Commands' }) | ||||||
|     const samplesCommandOption = page.getByRole('option', { |     const samplesCommandOption = page.getByRole('option', { | ||||||
| @ -75,8 +75,8 @@ test.describe('Testing in-app sample loading', () => { | |||||||
|  |  | ||||||
|   /** |   /** | ||||||
|    * Note this test implicitly depends on the KCL samples: |    * Note this test implicitly depends on the KCL samples: | ||||||
|    * "car-wheel.kcl": https://github.com/KittyCAD/kcl-samples/blob/main/car-wheel/car-wheel.kcl |    * "a-parametric-bearing-pillow-block": https://github.com/KittyCAD/kcl-samples/blob/main/a-parametric-bearing-pillow-block/main.kcl | ||||||
|    * "gear-rack.kcl": https://github.com/KittyCAD/kcl-samples/blob/main/gear-rack/gear-rack.kcl |    * "gear-rack": https://github.com/KittyCAD/kcl-samples/blob/main/gear-rack/main.kcl | ||||||
|    */ |    */ | ||||||
|   test( |   test( | ||||||
|     'Desktop: should create new file by default, optionally overwrite', |     'Desktop: should create new file by default, optionally overwrite', | ||||||
| @ -93,8 +93,8 @@ test.describe('Testing in-app sample loading', () => { | |||||||
|  |  | ||||||
|       // Locators and constants |       // Locators and constants | ||||||
|       const sampleOne = { |       const sampleOne = { | ||||||
|         file: 'car-wheel' + FILE_EXT, |         file: 'a-parametric-bearing-pillow-block' + FILE_EXT, | ||||||
|         title: 'Car Wheel', |         title: 'A Parametric Bearing Pillow Block', | ||||||
|       } |       } | ||||||
|       const sampleTwo = { |       const sampleTwo = { | ||||||
|         file: 'gear-rack' + FILE_EXT, |         file: 'gear-rack' + FILE_EXT, | ||||||
|  | |||||||
| @ -906,53 +906,6 @@ test.describe('Testing selections', () => { | |||||||
|     ).not.toBeDisabled() |     ).not.toBeDisabled() | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('Fillet button states test', async ({ page, homePage }) => { |  | ||||||
|     const u = await getUtils(page) |  | ||||||
|     await page.addInitScript(async () => { |  | ||||||
|       localStorage.setItem( |  | ||||||
|         'persistCode', |  | ||||||
|         `sketch001 = startSketchOn('XZ') |  | ||||||
|     |> startProfileAt([-5, -5], %) |  | ||||||
|     |> line([0, 10], %) |  | ||||||
|     |> line([10, 0], %) |  | ||||||
|     |> line([0, -10], %) |  | ||||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) |  | ||||||
|     |> close(%)` |  | ||||||
|       ) |  | ||||||
|     }) |  | ||||||
|  |  | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |  | ||||||
|     await homePage.goToModelingScene() |  | ||||||
|     await u.openDebugPanel() |  | ||||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') |  | ||||||
|     await u.closeDebugPanel() |  | ||||||
|  |  | ||||||
|     const selectSegment = () => page.getByText(`line([10, 0], %)`).click() |  | ||||||
|     const selectClose = () => page.getByText(`close(%)`).click() |  | ||||||
|     const clickEmpty = () => page.mouse.click(950, 100) |  | ||||||
|  |  | ||||||
|     // Now that we don't disable toolbar buttons based on selection, |  | ||||||
|     // but rather based on a "selection" step in the command palette, |  | ||||||
|     // the fillet button should always be enabled with a good network connection. |  | ||||||
|     // I'm not sure if this test is actually useful anymore. |  | ||||||
|     await selectSegment() |  | ||||||
|     await expect(page.getByRole('button', { name: 'Fillet' })).toBeEnabled() |  | ||||||
|     await clickEmpty() |  | ||||||
|     await expect(page.getByRole('button', { name: 'Fillet' })).toBeEnabled() |  | ||||||
|  |  | ||||||
|     // test fillet button with the body in the scene |  | ||||||
|     const codeToAdd = `${await u.codeLocator.allInnerTexts()} |  | ||||||
|   extrude001 = extrude(10, sketch001)` |  | ||||||
|     await u.codeLocator.clear() |  | ||||||
|     await u.codeLocator.fill(codeToAdd) |  | ||||||
|     await selectSegment() |  | ||||||
|     await expect(page.getByRole('button', { name: 'Fillet' })).toBeEnabled() |  | ||||||
|     await selectClose() |  | ||||||
|     await expect(page.getByRole('button', { name: 'Fillet' })).toBeEnabled() |  | ||||||
|     await clickEmpty() |  | ||||||
|     await expect(page.getByRole('button', { name: 'Fillet' })).toBeEnabled() |  | ||||||
|   }) |  | ||||||
|  |  | ||||||
|   const removeAfterFirstParenthesis = (inputString: string) => { |   const removeAfterFirstParenthesis = (inputString: string) => { | ||||||
|     const index = inputString.indexOf('(') |     const index = inputString.indexOf('(') | ||||||
|     if (index !== -1) { |     if (index !== -1) { | ||||||
|  | |||||||
							
								
								
									
										14
									
								
								package.json
									
									
									
									
									
								
							
							
						
						| @ -65,7 +65,7 @@ | |||||||
|     "vscode-languageserver-protocol": "^3.17.5", |     "vscode-languageserver-protocol": "^3.17.5", | ||||||
|     "vscode-uri": "^3.0.8", |     "vscode-uri": "^3.0.8", | ||||||
|     "web-vitals": "^3.5.2", |     "web-vitals": "^3.5.2", | ||||||
|     "xstate": "^5.17.4", |     "xstate": "^5.19.2", | ||||||
|     "yargs": "^17.7.2" |     "yargs": "^17.7.2" | ||||||
|   }, |   }, | ||||||
|   "scripts": { |   "scripts": { | ||||||
| @ -113,9 +113,9 @@ | |||||||
|     "test:unit": "vitest run --mode development --exclude **/kclSamples.test.ts", |     "test:unit": "vitest run --mode development --exclude **/kclSamples.test.ts", | ||||||
|     "test:unit:kcl-samples": "vitest run --mode development ./src/lang/kclSamples.test.ts", |     "test:unit:kcl-samples": "vitest run --mode development ./src/lang/kclSamples.test.ts", | ||||||
|     "test:playwright:electron": "playwright test --config=playwright.electron.config.ts --grep-invert='@snapshot'", |     "test:playwright:electron": "playwright test --config=playwright.electron.config.ts --grep-invert='@snapshot'", | ||||||
|     "test:playwright:electron:windows": "playwright test --config=playwright.electron.config.ts --grep-invert=\"@skipWin|@snapshot\"", |     "test:playwright:electron:windows": "playwright test --config=playwright.electron.config.ts --grep-invert=\"@skipWin|@snapshot\" --quiet", | ||||||
|     "test:playwright:electron:macos": "playwright test --config=playwright.electron.config.ts --grep-invert='@skipMacos|@snapshot'", |     "test:playwright:electron:macos": "playwright test --config=playwright.electron.config.ts --grep-invert='@skipMacos|@snapshot' --quiet", | ||||||
|     "test:playwright:electron:ubuntu": "playwright test --config=playwright.electron.config.ts --grep-invert='@skipLinux|@snapshot'", |     "test:playwright:electron:ubuntu": "playwright test --config=playwright.electron.config.ts --grep-invert='@skipLinux|@snapshot' --quiet", | ||||||
|     "test:playwright:electron:local": "yarn tron:package && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep-invert='@snapshot'", |     "test:playwright:electron:local": "yarn tron:package && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep-invert='@snapshot'", | ||||||
|     "test:playwright:electron:windows:local": "yarn tron:package && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep-invert=\"@skipWin|@snapshot\"", |     "test:playwright:electron:windows:local": "yarn tron:package && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep-invert=\"@skipWin|@snapshot\"", | ||||||
|     "test:playwright:electron:macos:local": "yarn tron:package && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep-invert='@skipMacos|@snapshot'", |     "test:playwright:electron:macos:local": "yarn tron:package && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep-invert='@skipMacos|@snapshot'", | ||||||
| @ -154,7 +154,6 @@ | |||||||
|     "@playwright/test": "^1.49.0", |     "@playwright/test": "^1.49.0", | ||||||
|     "@testing-library/jest-dom": "^5.14.1", |     "@testing-library/jest-dom": "^5.14.1", | ||||||
|     "@testing-library/react": "^15.0.2", |     "@testing-library/react": "^15.0.2", | ||||||
|     "@types/d3-force": "^3.0.10", |  | ||||||
|     "@types/diff": "^6.0.0", |     "@types/diff": "^6.0.0", | ||||||
|     "@types/electron": "^1.6.10", |     "@types/electron": "^1.6.10", | ||||||
|     "@types/isomorphic-fetch": "^0.0.39", |     "@types/isomorphic-fetch": "^0.0.39", | ||||||
| @ -175,7 +174,6 @@ | |||||||
|     "@vitest/web-worker": "^1.5.0", |     "@vitest/web-worker": "^1.5.0", | ||||||
|     "@xstate/cli": "^0.5.17", |     "@xstate/cli": "^0.5.17", | ||||||
|     "autoprefixer": "^10.4.19", |     "autoprefixer": "^10.4.19", | ||||||
|     "d3-force": "^3.0.0", |  | ||||||
|     "electron": "32.1.2", |     "electron": "32.1.2", | ||||||
|     "electron-builder": "24.13.3", |     "electron-builder": "24.13.3", | ||||||
|     "electron-notarize": "1.2.2", |     "electron-notarize": "1.2.2", | ||||||
| @ -201,9 +199,9 @@ | |||||||
|     "setimmediate": "^1.0.5", |     "setimmediate": "^1.0.5", | ||||||
|     "tailwindcss": "^3.4.1", |     "tailwindcss": "^3.4.1", | ||||||
|     "ts-node": "^10.0.0", |     "ts-node": "^10.0.0", | ||||||
|     "typescript": "^5.7.2", |     "typescript": "^5.7.3", | ||||||
|     "typescript-eslint": "^8.19.1", |     "typescript-eslint": "^8.19.1", | ||||||
|     "vite": "^5.4.6", |     "vite": "^5.4.12", | ||||||
|     "vite-plugin-package-version": "^1.1.0", |     "vite-plugin-package-version": "^1.1.0", | ||||||
|     "vite-tsconfig-paths": "^4.3.2", |     "vite-tsconfig-paths": "^4.3.2", | ||||||
|     "vitest": "^1.6.0", |     "vitest": "^1.6.0", | ||||||
|  | |||||||
							
								
								
									
										1
									
								
								packages/codemirror-lang-kcl/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -4,4 +4,5 @@ dist | |||||||
| tsconfig.tsbuildinfo | tsconfig.tsbuildinfo | ||||||
| *.d.ts | *.d.ts | ||||||
| *.js | *.js | ||||||
|  | !postcss.config.js | ||||||
| !rollup.config.js | !rollup.config.js | ||||||
|  | |||||||
| @ -28,6 +28,7 @@ | |||||||
|     "@rollup/plugin-typescript": "^12.1.2", |     "@rollup/plugin-typescript": "^12.1.2", | ||||||
|     "rollup": "^4.29.1", |     "rollup": "^4.29.1", | ||||||
|     "rollup-plugin-dts": "^6.1.1", |     "rollup-plugin-dts": "^6.1.1", | ||||||
|  |     "vite-tsconfig-paths": "^4.3.2", | ||||||
|     "vitest": "^2.1.8" |     "vitest": "^2.1.8" | ||||||
|   }, |   }, | ||||||
|   "files": [ |   "files": [ | ||||||
|  | |||||||
							
								
								
									
										1
									
								
								packages/codemirror-lang-kcl/postcss.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1 @@ | |||||||
|  | // This is here to prevent using the one in the root of the project. | ||||||
| @ -398,7 +398,7 @@ check-error@^2.1.1: | |||||||
|   resolved "https://registry.yarnpkg.com/check-error/-/check-error-2.1.1.tgz#87eb876ae71ee388fa0471fe423f494be1d96ccc" |   resolved "https://registry.yarnpkg.com/check-error/-/check-error-2.1.1.tgz#87eb876ae71ee388fa0471fe423f494be1d96ccc" | ||||||
|   integrity sha512-OAlb+T7V4Op9OwdkjmguYRqncdlx5JiofwOAUkmTF+jNdHwzTaTs4sRAGpzLF3oOz5xAyDGrPgeIDFQmDOTiJw== |   integrity sha512-OAlb+T7V4Op9OwdkjmguYRqncdlx5JiofwOAUkmTF+jNdHwzTaTs4sRAGpzLF3oOz5xAyDGrPgeIDFQmDOTiJw== | ||||||
|  |  | ||||||
| debug@^4.3.7: | debug@^4.1.1, debug@^4.3.7: | ||||||
|   version "4.4.0" |   version "4.4.0" | ||||||
|   resolved "https://registry.yarnpkg.com/debug/-/debug-4.4.0.tgz#2b3f2aea2ffeb776477460267377dc8710faba8a" |   resolved "https://registry.yarnpkg.com/debug/-/debug-4.4.0.tgz#2b3f2aea2ffeb776477460267377dc8710faba8a" | ||||||
|   integrity sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA== |   integrity sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA== | ||||||
| @ -471,6 +471,11 @@ function-bind@^1.1.2: | |||||||
|   resolved "https://registry.yarnpkg.com/function-bind/-/function-bind-1.1.2.tgz#2c02d864d97f3ea6c8830c464cbd11ab6eab7a1c" |   resolved "https://registry.yarnpkg.com/function-bind/-/function-bind-1.1.2.tgz#2c02d864d97f3ea6c8830c464cbd11ab6eab7a1c" | ||||||
|   integrity sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA== |   integrity sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA== | ||||||
|  |  | ||||||
|  | globrex@^0.1.2: | ||||||
|  |   version "0.1.2" | ||||||
|  |   resolved "https://registry.yarnpkg.com/globrex/-/globrex-0.1.2.tgz#dd5d9ec826232730cd6793a5e33a9302985e6098" | ||||||
|  |   integrity sha512-uHJgbwAMwNFf5mLst7IWLNg14x1CkeqglJb/K3doi4dw6q2IvAAmM/Y81kevy83wP+Sst+nutFTYOGg3d1lsxg== | ||||||
|  |  | ||||||
| hasown@^2.0.2: | hasown@^2.0.2: | ||||||
|   version "2.0.2" |   version "2.0.2" | ||||||
|   resolved "https://registry.yarnpkg.com/hasown/-/hasown-2.0.2.tgz#003eaf91be7adc372e84ec59dc37252cedb80003" |   resolved "https://registry.yarnpkg.com/hasown/-/hasown-2.0.2.tgz#003eaf91be7adc372e84ec59dc37252cedb80003" | ||||||
| @ -647,6 +652,11 @@ tinyspy@^3.0.2: | |||||||
|   resolved "https://registry.yarnpkg.com/tinyspy/-/tinyspy-3.0.2.tgz#86dd3cf3d737b15adcf17d7887c84a75201df20a" |   resolved "https://registry.yarnpkg.com/tinyspy/-/tinyspy-3.0.2.tgz#86dd3cf3d737b15adcf17d7887c84a75201df20a" | ||||||
|   integrity sha512-n1cw8k1k0x4pgA2+9XrOkFydTerNcJ1zWCO5Nn9scWHTD+5tp8dghT2x1uduQePZTZgd3Tupf+x9BxJjeJi77Q== |   integrity sha512-n1cw8k1k0x4pgA2+9XrOkFydTerNcJ1zWCO5Nn9scWHTD+5tp8dghT2x1uduQePZTZgd3Tupf+x9BxJjeJi77Q== | ||||||
|  |  | ||||||
|  | tsconfck@^3.0.3: | ||||||
|  |   version "3.1.4" | ||||||
|  |   resolved "https://registry.yarnpkg.com/tsconfck/-/tsconfck-3.1.4.tgz#de01a15334962e2feb526824339b51be26712229" | ||||||
|  |   integrity sha512-kdqWFGVJqe+KGYvlSO9NIaWn9jT1Ny4oKVzAJsKii5eoE9snzTJzL4+MMVOMn+fikWGFmKEylcXL710V/kIPJQ== | ||||||
|  |  | ||||||
| typescript@^5.7.2: | typescript@^5.7.2: | ||||||
|   version "5.7.2" |   version "5.7.2" | ||||||
|   resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.7.2.tgz#3169cf8c4c8a828cde53ba9ecb3d2b1d5dd67be6" |   resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.7.2.tgz#3169cf8c4c8a828cde53ba9ecb3d2b1d5dd67be6" | ||||||
| @ -663,6 +673,15 @@ vite-node@2.1.8: | |||||||
|     pathe "^1.1.2" |     pathe "^1.1.2" | ||||||
|     vite "^5.0.0" |     vite "^5.0.0" | ||||||
|  |  | ||||||
|  | vite-tsconfig-paths@^4.3.2: | ||||||
|  |   version "4.3.2" | ||||||
|  |   resolved "https://registry.yarnpkg.com/vite-tsconfig-paths/-/vite-tsconfig-paths-4.3.2.tgz#321f02e4b736a90ff62f9086467faf4e2da857a9" | ||||||
|  |   integrity sha512-0Vd/a6po6Q+86rPlntHye7F31zA2URZMbH8M3saAZ/xR9QoGN/L21bxEGfXdWmFdNkqPpRdxFT7nmNe12e9/uA== | ||||||
|  |   dependencies: | ||||||
|  |     debug "^4.1.1" | ||||||
|  |     globrex "^0.1.2" | ||||||
|  |     tsconfck "^3.0.3" | ||||||
|  |  | ||||||
| vite@^5.0.0: | vite@^5.0.0: | ||||||
|   version "5.4.11" |   version "5.4.11" | ||||||
|   resolved "https://registry.yarnpkg.com/vite/-/vite-5.4.11.tgz#3b415cd4aed781a356c1de5a9ebafb837715f6e5" |   resolved "https://registry.yarnpkg.com/vite/-/vite-5.4.11.tgz#3b415cd4aed781a356c1de5a9ebafb837715f6e5" | ||||||
|  | |||||||
| @ -29,7 +29,7 @@ | |||||||
|     "vscode-uri": "^3.0.8" |     "vscode-uri": "^3.0.8" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@types/node": "^20.14.9", |     "@types/node": "^22.10.6", | ||||||
|     "ts-node": "^10.9.2" |     "ts-node": "^10.9.2" | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  | |||||||
| @ -109,12 +109,12 @@ | |||||||
|   resolved "https://registry.yarnpkg.com/@tsconfig/node16/-/node16-1.0.4.tgz#0b92dcc0cc1c81f6f306a381f28e31b1a56536e9" |   resolved "https://registry.yarnpkg.com/@tsconfig/node16/-/node16-1.0.4.tgz#0b92dcc0cc1c81f6f306a381f28e31b1a56536e9" | ||||||
|   integrity sha512-vxhUy4J8lyeyinH7Azl1pdd43GJhZH/tP2weN8TntQblOY+A0XbT8DJk1/oCPuOOyg/Ja757rG0CgHcWC8OfMA== |   integrity sha512-vxhUy4J8lyeyinH7Azl1pdd43GJhZH/tP2weN8TntQblOY+A0XbT8DJk1/oCPuOOyg/Ja757rG0CgHcWC8OfMA== | ||||||
|  |  | ||||||
| "@types/node@^20.14.9": | "@types/node@^22.10.6": | ||||||
|   version "20.14.9" |   version "22.10.6" | ||||||
|   resolved "https://registry.yarnpkg.com/@types/node/-/node-20.14.9.tgz#12e8e765ab27f8c421a1820c99f5f313a933b420" |   resolved "https://registry.yarnpkg.com/@types/node/-/node-22.10.6.tgz#5c6795e71635876039f853cbccd59f523d9e4239" | ||||||
|   integrity sha512-06OCtnTXtWOZBJlRApleWndH4JsRVs1pDCc8dLSQp+7PpUpX3ePdHyeNSFTeSe7FtKyQkrlPvHwJOW3SLd8Oyg== |   integrity sha512-qNiuwC4ZDAUNcY47xgaSuS92cjf8JbSUoaKS77bmLG1rU7MlATVSiw/IlrjtIyyskXBZ8KkNfjK/P5na7rgXbQ== | ||||||
|   dependencies: |   dependencies: | ||||||
|     undici-types "~5.26.4" |     undici-types "~6.20.0" | ||||||
|  |  | ||||||
| acorn-walk@^8.1.1: | acorn-walk@^8.1.1: | ||||||
|   version "8.3.3" |   version "8.3.3" | ||||||
| @ -187,10 +187,10 @@ typescript@^5.7.2: | |||||||
|   resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.7.2.tgz#3169cf8c4c8a828cde53ba9ecb3d2b1d5dd67be6" |   resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.7.2.tgz#3169cf8c4c8a828cde53ba9ecb3d2b1d5dd67be6" | ||||||
|   integrity sha512-i5t66RHxDvVN40HfDd1PsEThGNnlMCMT3jMUuoh9/0TaqWevNontacunWyN02LA9/fIbEWlcHZcgTKb9QoaLfg== |   integrity sha512-i5t66RHxDvVN40HfDd1PsEThGNnlMCMT3jMUuoh9/0TaqWevNontacunWyN02LA9/fIbEWlcHZcgTKb9QoaLfg== | ||||||
|  |  | ||||||
| undici-types@~5.26.4: | undici-types@~6.20.0: | ||||||
|   version "5.26.5" |   version "6.20.0" | ||||||
|   resolved "https://registry.yarnpkg.com/undici-types/-/undici-types-5.26.5.tgz#bcd539893d00b56e964fd2657a4866b221a65617" |   resolved "https://registry.yarnpkg.com/undici-types/-/undici-types-6.20.0.tgz#8171bf22c1f588d1554d55bf204bc624af388433" | ||||||
|   integrity sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA== |   integrity sha512-Ny6QZ2Nju20vw1SRHe3d9jVu6gJ+4e3+MMpqu7pqE5HT6WsTSlce++GQmK5UXS8mzV8DSYHrQH+Xrf2jVcuKNg== | ||||||
|  |  | ||||||
| v8-compile-cache-lib@^3.0.1: | v8-compile-cache-lib@^3.0.1: | ||||||
|   version "3.0.1" |   version "3.0.1" | ||||||
|  | |||||||
| @ -1,172 +1,212 @@ | |||||||
| [ | [ | ||||||
|   { |   { | ||||||
|     "file": "80-20-rail.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "80-20-rail/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "80/20 Rail", |     "title": "80/20 Rail", | ||||||
|     "description": "An 80/20 extruded aluminum linear rail. T-slot profile adjustable by profile height, rail length, and origin position" |     "description": "An 80/20 extruded aluminum linear rail. T-slot profile adjustable by profile height, rail length, and origin position" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "a-parametric-bearing-pillow-block.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "a-parametric-bearing-pillow-block/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "A Parametric Bearing Pillow Block", |     "title": "A Parametric Bearing Pillow Block", | ||||||
|     "description": "A bearing pillow block, also known as a plummer block or pillow block bearing, is a pedestal used to provide support for a rotating shaft with the help of compatible bearings and various accessories. Housing a bearing, the pillow block provides a secure and stable foundation that allows the shaft to rotate smoothly within its machinery setup. These components are essential in a wide range of mechanical systems and machinery, playing a key role in reducing friction and supporting radial and axial loads." |     "description": "A bearing pillow block, also known as a plummer block or pillow block bearing, is a pedestal used to provide support for a rotating shaft with the help of compatible bearings and various accessories. Housing a bearing, the pillow block provides a secure and stable foundation that allows the shaft to rotate smoothly within its machinery setup. These components are essential in a wide range of mechanical systems and machinery, playing a key role in reducing friction and supporting radial and axial loads." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "ball-bearing.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "ball-bearing/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Ball Bearing", |     "title": "Ball Bearing", | ||||||
|     "description": "A ball bearing is a type of rolling-element bearing that uses balls to maintain the separation between the bearing races. The primary purpose of a ball bearing is to reduce rotational friction and support radial and axial loads." |     "description": "A ball bearing is a type of rolling-element bearing that uses balls to maintain the separation between the bearing races. The primary purpose of a ball bearing is to reduce rotational friction and support radial and axial loads." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "bracket.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "bracket/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Shelf Bracket", |     "title": "Shelf Bracket", | ||||||
|     "description": "This is a bracket that holds a shelf. It is made of aluminum and is designed to hold a force of 300 lbs. The bracket is 6 inches wide and the force is applied at the end of the shelf, 12 inches from the wall. The bracket has a factor of safety of 1.2. The legs of the bracket are 5 inches and 2 inches long. The thickness of the bracket is calculated from the constraints provided." |     "description": "This is a bracket that holds a shelf. It is made of aluminum and is designed to hold a force of 300 lbs. The bracket is 6 inches wide and the force is applied at the end of the shelf, 12 inches from the wall. The bracket has a factor of safety of 1.2. The legs of the bracket are 5 inches and 2 inches long. The thickness of the bracket is calculated from the constraints provided." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "brake-caliper.kcl", |     "file": "main.kcl", | ||||||
|     "title": "Brake Caliper", |     "pathFromProjectDirectoryToFirstFile": "car-wheel-assembly/main.kcl", | ||||||
|     "description": "Brake calipers are used to squeeze the brake pads against the rotor, causing larger and larger amounts of friction depending on how hard the brakes are pressed." |     "multipleFiles": true, | ||||||
|   }, |  | ||||||
|   { |  | ||||||
|     "file": "car-wheel.kcl", |  | ||||||
|     "title": "Car Wheel", |  | ||||||
|     "description": "A sports car wheel with a circular lug pattern and spokes." |  | ||||||
|   }, |  | ||||||
|   { |  | ||||||
|     "file": "car-wheel-assembly.kcl", |  | ||||||
|     "title": "Car Wheel Assembly", |     "title": "Car Wheel Assembly", | ||||||
|     "description": "A car wheel assembly with a rotor, tire, and lug nuts." |     "description": "A car wheel assembly with a rotor, tire, and lug nuts." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "dodecahedron.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "dodecahedron/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Hollow Dodecahedron", |     "title": "Hollow Dodecahedron", | ||||||
|     "description": "A regular dodecahedron or pentagonal dodecahedron is a dodecahedron composed of regular pentagonal faces, three meeting at each vertex. This example shows constructing the individual faces of the dodecahedron and extruding inwards." |     "description": "A regular dodecahedron or pentagonal dodecahedron is a dodecahedron composed of regular pentagonal faces, three meeting at each vertex. This example shows constructing the individual faces of the dodecahedron and extruding inwards." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "enclosure.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "enclosure/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Enclosure", |     "title": "Enclosure", | ||||||
|     "description": "An enclosure body and sealing lid for storing items" |     "description": "An enclosure body and sealing lid for storing items" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "flange-with-patterns.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "flange-with-patterns/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Flange", |     "title": "Flange", | ||||||
|     "description": "A flange is a flat rim, collar, or rib, typically forged or cast, that is used to strengthen an object, guide it, or attach it to another object. Flanges are known for their use in various applications, including piping, plumbing, and mechanical engineering, among others." |     "description": "A flange is a flat rim, collar, or rib, typically forged or cast, that is used to strengthen an object, guide it, or attach it to another object. Flanges are known for their use in various applications, including piping, plumbing, and mechanical engineering, among others." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "flange-xy.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "flange-xy/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Flange with XY coordinates", |     "title": "Flange with XY coordinates", | ||||||
|     "description": "A flange is a flat rim, collar, or rib, typically forged or cast, that is used to strengthen an object, guide it, or attach it to another object. Flanges are known for their use in various applications, including piping, plumbing, and mechanical engineering, among others." |     "description": "A flange is a flat rim, collar, or rib, typically forged or cast, that is used to strengthen an object, guide it, or attach it to another object. Flanges are known for their use in various applications, including piping, plumbing, and mechanical engineering, among others." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "focusrite-scarlett-mounting-bracket.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "focusrite-scarlett-mounting-bracket/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "A mounting bracket for the Focusrite Scarlett Solo audio interface", |     "title": "A mounting bracket for the Focusrite Scarlett Solo audio interface", | ||||||
|     "description": "This is a bracket that holds an audio device underneath a desk or shelf. The audio device has dimensions of 144mm wide, 80mm length and 45mm depth with fillets of 6mm. This mounting bracket is designed to be 3D printed with PLA material" |     "description": "This is a bracket that holds an audio device underneath a desk or shelf. The audio device has dimensions of 144mm wide, 80mm length and 45mm depth with fillets of 6mm. This mounting bracket is designed to be 3D printed with PLA material" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "food-service-spatula.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "food-service-spatula/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Food Service Spatula", |     "title": "Food Service Spatula", | ||||||
|     "description": "Use these spatulas for mixing, flipping, and scraping." |     "description": "Use these spatulas for mixing, flipping, and scraping." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "french-press.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "french-press/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "French Press", |     "title": "French Press", | ||||||
|     "description": "A french press immersion coffee maker" |     "description": "A french press immersion coffee maker" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "gear.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "gear/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Spur Gear", |     "title": "Spur Gear", | ||||||
|     "description": "A rotating machine part having cut teeth or, in the case of a cogwheel, inserted teeth (called cogs), which mesh with another toothed part to transmit torque. Geared devices can change the speed, torque, and direction of a power source. The two elements that define a gear are its circular shape and the teeth that are integrated into its outer edge, which are designed to fit into the teeth of another gear." |     "description": "A rotating machine part having cut teeth or, in the case of a cogwheel, inserted teeth (called cogs), which mesh with another toothed part to transmit torque. Geared devices can change the speed, torque, and direction of a power source. The two elements that define a gear are its circular shape and the teeth that are integrated into its outer edge, which are designed to fit into the teeth of another gear." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "gear-rack.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "gear-rack/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "100mm Gear Rack", |     "title": "100mm Gear Rack", | ||||||
|     "description": "A flat bar or rail that is engraved with teeth along its length. These teeth are designed to mesh with the teeth of a gear, known as a pinion. When the pinion, a small cylindrical gear, rotates, its teeth engage with the teeth on the rack, causing the rack to move linearly. Conversely, linear motion applied to the rack will cause the pinion to rotate." |     "description": "A flat bar or rail that is engraved with teeth along its length. These teeth are designed to mesh with the teeth of a gear, known as a pinion. When the pinion, a small cylindrical gear, rotates, its teeth engage with the teeth on the rack, causing the rack to move linearly. Conversely, linear motion applied to the rack will cause the pinion to rotate." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "hex-nut.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "hex-nut/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Hex nut", |     "title": "Hex nut", | ||||||
|     "description": "A hex nut is a type of fastener with a threaded hole and a hexagonal outer shape, used in a wide variety of applications to secure parts together. The hexagonal shape allows for a greater torque to be applied with wrenches or tools, making it one of the most common nut types in hardware." |     "description": "A hex nut is a type of fastener with a threaded hole and a hexagonal outer shape, used in a wide variety of applications to secure parts together. The hexagonal shape allows for a greater torque to be applied with wrenches or tools, making it one of the most common nut types in hardware." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "i-beam.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "i-beam/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "I-beam", |     "title": "I-beam", | ||||||
|     "description": "A structural metal beam with an I shaped cross section. Often used in construction" |     "description": "A structural metal beam with an I shaped cross section. Often used in construction" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "kitt.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "kitt/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Kitt", |     "title": "Kitt", | ||||||
|     "description": "The beloved KittyCAD mascot in a voxelized style." |     "description": "The beloved KittyCAD mascot in a voxelized style." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "lego.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "lego/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Lego Brick", |     "title": "Lego Brick", | ||||||
|     "description": "A standard Lego brick. This is a small, plastic construction block toy that can be interlocked with other blocks to build various structures, models, and figures. There are a lot of hacks used in this code." |     "description": "A standard Lego brick. This is a small, plastic construction block toy that can be interlocked with other blocks to build various structures, models, and figures. There are a lot of hacks used in this code." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "lug-nut.kcl", |     "file": "main.kcl", | ||||||
|     "title": "Lug Nut", |     "pathFromProjectDirectoryToFirstFile": "mounting-plate/main.kcl", | ||||||
|     "description": "lug Nuts are essential components used to create secure connections, whether for electrical purposes, like terminating wires or grounding, or for mechanical purposes, such as providing mounting points or reinforcing structural joints." |     "multipleFiles": false, | ||||||
|   }, |  | ||||||
|   { |  | ||||||
|     "file": "mounting-plate.kcl", |  | ||||||
|     "title": "Mounting Plate", |     "title": "Mounting Plate", | ||||||
|     "description": "A flat piece of material, often metal or plastic, that serves as a support or base for attaching, securing, or mounting various types of equipment, devices, or components." |     "description": "A flat piece of material, often metal or plastic, that serves as a support or base for attaching, securing, or mounting various types of equipment, devices, or components." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "multi-axis-robot.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "multi-axis-robot/main.kcl", | ||||||
|  |     "multipleFiles": true, | ||||||
|     "title": "Robot Arm", |     "title": "Robot Arm", | ||||||
|     "description": "A 4 axis robotic arm for industrial use. These machines can be used for assembly, packaging, organization of goods, and quality inspection processes" |     "description": "A 4 axis robotic arm for industrial use. These machines can be used for assembly, packaging, organization of goods, and quality inspection processes" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "pipe.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "pipe/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Pipe", |     "title": "Pipe", | ||||||
|     "description": "A tubular section or hollow cylinder, usually but not necessarily of circular cross-section, used mainly to convey substances that can flow." |     "description": "A tubular section or hollow cylinder, usually but not necessarily of circular cross-section, used mainly to convey substances that can flow." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "pipe-flange-assembly.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "pipe-flange-assembly/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Pipe and Flange Assembly", |     "title": "Pipe and Flange Assembly", | ||||||
|     "description": "A crucial component in various piping systems, designed to facilitate the connection, disconnection, and access to piping for inspection, cleaning, and modifications. This assembly combines pipes (long cylindrical conduits) with flanges (plate-like fittings) to create a secure yet detachable joint." |     "description": "A crucial component in various piping systems, designed to facilitate the connection, disconnection, and access to piping for inspection, cleaning, and modifications. This assembly combines pipes (long cylindrical conduits) with flanges (plate-like fittings) to create a secure yet detachable joint." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "pipe-with-bend.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "pipe-with-bend/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Pipe with bend", |     "title": "Pipe with bend", | ||||||
|     "description": "A tubular section or hollow cylinder, usually but not necessarily of circular cross-section, used mainly to convey substances that can flow." |     "description": "A tubular section or hollow cylinder, usually but not necessarily of circular cross-section, used mainly to convey substances that can flow." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "poopy-shoe.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "poopy-shoe/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Poopy Shoe", |     "title": "Poopy Shoe", | ||||||
|     "description": "poop shute for bambu labs printer - optimized for printing." |     "description": "poop shute for bambu labs printer - optimized for printing." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "router-template-cross-bar.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "router-template-cross-bar/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Router template for a cross bar", |     "title": "Router template for a cross bar", | ||||||
|     "description": "A guide for routing a notch into a cross bar." |     "description": "A guide for routing a notch into a cross bar." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "router-template-slate.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "router-template-slate/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Router template for a slate", |     "title": "Router template for a slate", | ||||||
|     "description": "A guide for routing a slate for a cross bar." |     "description": "A guide for routing a slate for a cross bar." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "sheet-metal-bracket.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "sheet-metal-bracket/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Sheet Metal Bracket", |     "title": "Sheet Metal Bracket", | ||||||
|     "description": "A component typically made from flat sheet metal through various manufacturing processes such as bending, punching, cutting, and forming. These brackets are used to support, attach, or mount other hardware components, often providing a structural or functional base for assembly." |     "description": "A component typically made from flat sheet metal through various manufacturing processes such as bending, punching, cutting, and forming. These brackets are used to support, attach, or mount other hardware components, often providing a structural or functional base for assembly." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "socket-head-cap-screw.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "socket-head-cap-screw/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Socket Head Cap Screw", |     "title": "Socket Head Cap Screw", | ||||||
|     "description": "This is for a #10-24 screw that is 1.00 inches long. A socket head cap screw is a type of fastener that is widely used in a variety of applications requiring a high strength fastening solution. It is characterized by its cylindrical head and internal hexagonal drive, which allows for tightening with an Allen wrench or hex key." |     "description": "This is for a #10-24 screw that is 1.00 inches long. A socket head cap screw is a type of fastener that is widely used in a variety of applications requiring a high strength fastening solution. It is characterized by its cylindrical head and internal hexagonal drive, which allows for tightening with an Allen wrench or hex key." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "tire.kcl", |     "file": "main.kcl", | ||||||
|     "title": "Tire", |     "pathFromProjectDirectoryToFirstFile": "walkie-talkie/main.kcl", | ||||||
|     "description": "A tire is a critical component of a vehicle that provides the necessary traction and grip between the car and the road. It supports the vehicle's weight and absorbs shocks from road irregularities." |     "multipleFiles": true, | ||||||
|  |     "title": "Walkie Talkie", | ||||||
|  |     "description": "A portable, handheld two-way radio device that allows users to communicate wirelessly over short to medium distances. It operates on specific radio frequencies and features a push-to-talk button for transmitting messages, making it ideal for quick and reliable communication in outdoor, work, or emergency settings." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "washer.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "washer/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Washer", |     "title": "Washer", | ||||||
|     "description": "A small, typically disk-shaped component with a hole in the middle, used in a wide range of applications, primarily in conjunction with fasteners like bolts and screws. Washers distribute the load of a fastener across a broader area. This is especially important when the fastening surface is soft or uneven, as it helps to prevent damage to the surface and ensures the load is evenly distributed, reducing the risk of the fastener becoming loose over time." |     "description": "A small, typically disk-shaped component with a hole in the middle, used in a wide range of applications, primarily in conjunction with fasteners like bolts and screws. Washers distribute the load of a fastener across a broader area. This is especially important when the fastening surface is soft or uneven, as it helps to prevent damage to the surface and ensures the load is evenly distributed, reducing the risk of the fastener becoming loose over time." | ||||||
|   }, |  | ||||||
|   { |  | ||||||
|     "file": "wheel-rotor.kcl", |  | ||||||
|     "title": "Wheel rotor", |  | ||||||
|     "description": "A component of a disc brake system. It provides a surface for brake pads to press against, generating the friction needed to slow or stop the vehicle." |  | ||||||
|   } |   } | ||||||
| ] | ] | ||||||
| @ -31,7 +31,6 @@ import { | |||||||
|   settingsLoader, |   settingsLoader, | ||||||
|   telemetryLoader, |   telemetryLoader, | ||||||
| } from 'lib/routeLoaders' | } from 'lib/routeLoaders' | ||||||
| import { CommandBarProvider } from 'components/CommandBar/CommandBarProvider' |  | ||||||
| import SettingsAuthProvider from 'components/SettingsAuthProvider' | import SettingsAuthProvider from 'components/SettingsAuthProvider' | ||||||
| import LspProvider from 'components/LspProvider' | import LspProvider from 'components/LspProvider' | ||||||
| import { KclContextProvider } from 'lang/KclProvider' | import { KclContextProvider } from 'lang/KclProvider' | ||||||
| @ -58,23 +57,21 @@ const router = createRouter([ | |||||||
|     /* Make sure auth is the outermost provider or else we will have |     /* Make sure auth is the outermost provider or else we will have | ||||||
|      * inefficient re-renders, use the react profiler to see. */ |      * inefficient re-renders, use the react profiler to see. */ | ||||||
|     element: ( |     element: ( | ||||||
|       <CommandBarProvider> |       <RouteProvider> | ||||||
|         <RouteProvider> |         <SettingsAuthProvider> | ||||||
|           <SettingsAuthProvider> |           <LspProvider> | ||||||
|             <LspProvider> |             <ProjectsContextProvider> | ||||||
|               <ProjectsContextProvider> |               <KclContextProvider> | ||||||
|                 <KclContextProvider> |                 <AppStateProvider> | ||||||
|                   <AppStateProvider> |                   <MachineManagerProvider> | ||||||
|                     <MachineManagerProvider> |                     <Outlet /> | ||||||
|                       <Outlet /> |                   </MachineManagerProvider> | ||||||
|                     </MachineManagerProvider> |                 </AppStateProvider> | ||||||
|                   </AppStateProvider> |               </KclContextProvider> | ||||||
|                 </KclContextProvider> |             </ProjectsContextProvider> | ||||||
|               </ProjectsContextProvider> |           </LspProvider> | ||||||
|             </LspProvider> |         </SettingsAuthProvider> | ||||||
|           </SettingsAuthProvider> |       </RouteProvider> | ||||||
|         </RouteProvider> |  | ||||||
|       </CommandBarProvider> |  | ||||||
|     ), |     ), | ||||||
|     errorElement: <ErrorPage />, |     errorElement: <ErrorPage />, | ||||||
|     children: [ |     children: [ | ||||||
|  | |||||||
							
								
								
									
										221
									
								
								src/Toolbar.tsx
									
									
									
									
									
								
							
							
						
						| @ -1,8 +1,7 @@ | |||||||
| import { useRef, useMemo, memo } from 'react' | import { useRef, useMemo, memo, useCallback, useState } from 'react' | ||||||
| import { isCursorInSketchCommandRange } from 'lang/util' | import { isCursorInSketchCommandRange } from 'lang/util' | ||||||
| import { engineCommandManager, kclManager } from 'lib/singletons' | import { engineCommandManager, kclManager } from 'lib/singletons' | ||||||
| import { useModelingContext } from 'hooks/useModelingContext' | import { useModelingContext } from 'hooks/useModelingContext' | ||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' |  | ||||||
| import { useNetworkContext } from 'hooks/useNetworkContext' | import { useNetworkContext } from 'hooks/useNetworkContext' | ||||||
| import { NetworkHealthState } from 'hooks/useNetworkStatus' | import { NetworkHealthState } from 'hooks/useNetworkStatus' | ||||||
| import { ActionButton } from 'components/ActionButton' | import { ActionButton } from 'components/ActionButton' | ||||||
| @ -22,20 +21,19 @@ import { | |||||||
| } from 'lib/toolbar' | } from 'lib/toolbar' | ||||||
| import { isDesktop } from 'lib/isDesktop' | import { isDesktop } from 'lib/isDesktop' | ||||||
| import { openExternalBrowserIfDesktop } from 'lib/openWindow' | import { openExternalBrowserIfDesktop } from 'lib/openWindow' | ||||||
|  | import { commandBarActor } from 'machines/commandBarMachine' | ||||||
|  |  | ||||||
| export function Toolbar({ | export function Toolbar({ | ||||||
|   className = '', |   className = '', | ||||||
|   ...props |   ...props | ||||||
| }: React.HTMLAttributes<HTMLElement>) { | }: React.HTMLAttributes<HTMLElement>) { | ||||||
|   const { state, send, context } = useModelingContext() |   const { state, send, context } = useModelingContext() | ||||||
|   const { commandBarSend } = useCommandsContext() |  | ||||||
|   const iconClassName = |   const iconClassName = | ||||||
|     'group-disabled:text-chalkboard-50 !text-inherit dark:group-enabled:group-hover:!text-inherit' |     'group-disabled:text-chalkboard-50 !text-inherit dark:group-enabled:group-hover:!text-inherit' | ||||||
|   const bgClassName = '!bg-transparent' |   const bgClassName = '!bg-transparent' | ||||||
|   const buttonBgClassName = |   const buttonBgClassName = | ||||||
|     'bg-chalkboard-transparent dark:bg-transparent disabled:bg-transparent dark:disabled:bg-transparent enabled:hover:bg-chalkboard-10 dark:enabled:hover:bg-chalkboard-100 pressed:!bg-primary pressed:enabled:hover:!text-chalkboard-10' |     'bg-chalkboard-transparent dark:bg-transparent disabled:bg-transparent dark:disabled:bg-transparent enabled:hover:bg-chalkboard-10 dark:enabled:hover:bg-chalkboard-100 pressed:!bg-primary pressed:enabled:hover:!text-chalkboard-10' | ||||||
|   const buttonBorderClassName = |   const buttonBorderClassName = '!border-transparent' | ||||||
|     '!border-transparent hover:!border-chalkboard-20 dark:enabled:hover:!border-primary pressed:!border-primary ui-open:!border-primary' |  | ||||||
|  |  | ||||||
|   const sketchPathId = useMemo(() => { |   const sketchPathId = useMemo(() => { | ||||||
|     if (!isSingleCursorInPipe(context.selectionRanges, kclManager.ast)) |     if (!isSingleCursorInPipe(context.selectionRanges, kclManager.ast)) | ||||||
| @ -50,6 +48,7 @@ export function Toolbar({ | |||||||
|   const { overallState } = useNetworkContext() |   const { overallState } = useNetworkContext() | ||||||
|   const { isExecuting } = useKclContext() |   const { isExecuting } = useKclContext() | ||||||
|   const { isStreamReady } = useAppState() |   const { isStreamReady } = useAppState() | ||||||
|  |   const [showRichContent, setShowRichContent] = useState(false) | ||||||
|  |  | ||||||
|   const disableAllButtons = |   const disableAllButtons = | ||||||
|     (overallState !== NetworkHealthState.Ok && |     (overallState !== NetworkHealthState.Ok && | ||||||
| @ -71,12 +70,45 @@ export function Toolbar({ | |||||||
|     () => ({ |     () => ({ | ||||||
|       modelingState: state, |       modelingState: state, | ||||||
|       modelingSend: send, |       modelingSend: send, | ||||||
|       commandBarSend, |  | ||||||
|       sketchPathId, |       sketchPathId, | ||||||
|     }), |     }), | ||||||
|     [state, send, commandBarSend, sketchPathId] |     [state, send, commandBarActor.send, sketchPathId] | ||||||
|   ) |   ) | ||||||
|  |  | ||||||
|  |   const tooltipContentClassName = !showRichContent | ||||||
|  |     ? '' | ||||||
|  |     : '!text-left text-wrap !text-xs !p-0 !pb-2 flex gap-2 !max-w-none !w-72 flex-col items-stretch' | ||||||
|  |   const richContentTimeout = useRef<number | null>(null) | ||||||
|  |   const richContentClearTimeout = useRef<number | null>(null) | ||||||
|  |   // On mouse enter, show rich content after a 1s delay | ||||||
|  |   const handleMouseEnter = useCallback(() => { | ||||||
|  |     // Cancel the clear timeout if it's already set | ||||||
|  |     if (richContentClearTimeout.current) { | ||||||
|  |       clearTimeout(richContentClearTimeout.current) | ||||||
|  |     } | ||||||
|  |     // Start our own timeout to show the rich content | ||||||
|  |     richContentTimeout.current = window.setTimeout(() => { | ||||||
|  |       setShowRichContent(true) | ||||||
|  |       if (richContentClearTimeout.current) { | ||||||
|  |         clearTimeout(richContentClearTimeout.current) | ||||||
|  |       } | ||||||
|  |     }, 1000) | ||||||
|  |   }, [setShowRichContent]) | ||||||
|  |   // On mouse leave, clear the timeout and hide rich content | ||||||
|  |   const handleMouseLeave = useCallback(() => { | ||||||
|  |     // Clear the timeout to show rich content | ||||||
|  |     if (richContentTimeout.current) { | ||||||
|  |       clearTimeout(richContentTimeout.current) | ||||||
|  |     } | ||||||
|  |     // Start a timeout to hide the rich content | ||||||
|  |     richContentClearTimeout.current = window.setTimeout(() => { | ||||||
|  |       setShowRichContent(false) | ||||||
|  |       if (richContentClearTimeout.current) { | ||||||
|  |         clearTimeout(richContentClearTimeout.current) | ||||||
|  |       } | ||||||
|  |     }, 500) | ||||||
|  |   }, [setShowRichContent]) | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
|    * Resolve all the callbacks and values for the current mode, |    * Resolve all the callbacks and values for the current mode, | ||||||
|    * so we don't need to worry about the other modes |    * so we don't need to worry about the other modes | ||||||
| @ -174,43 +206,64 @@ export function Toolbar({ | |||||||
|                   status: itemConfig.status, |                   status: itemConfig.status, | ||||||
|                 }))} |                 }))} | ||||||
|               > |               > | ||||||
|                 <ActionButton |                 <div | ||||||
|                   Element="button" |                   className="contents" | ||||||
|                   id={maybeIconConfig[0].id} |                   // Mouse events do not fire on disabled buttons | ||||||
|                   data-testid={maybeIconConfig[0].id} |                   onMouseEnter={handleMouseEnter} | ||||||
|                   iconStart={{ |                   onMouseLeave={handleMouseLeave} | ||||||
|                     icon: maybeIconConfig[0].icon, |  | ||||||
|                     className: iconClassName, |  | ||||||
|                     bgClassName: bgClassName, |  | ||||||
|                   }} |  | ||||||
|                   className={ |  | ||||||
|                     '!border-transparent !px-0 pressed:!text-chalkboard-10 pressed:enabled:hovered:!text-chalkboard-10 ' + |  | ||||||
|                     buttonBgClassName |  | ||||||
|                   } |  | ||||||
|                   aria-pressed={maybeIconConfig[0].isActive} |  | ||||||
|                   disabled={ |  | ||||||
|                     disableAllButtons || |  | ||||||
|                     maybeIconConfig[0].status !== 'available' || |  | ||||||
|                     maybeIconConfig[0].disabled |  | ||||||
|                   } |  | ||||||
|                   name={maybeIconConfig[0].title} |  | ||||||
|                   // aria-description is still in ARIA 1.3 draft. |  | ||||||
|                   // eslint-disable-next-line jsx-a11y/aria-props |  | ||||||
|                   aria-description={maybeIconConfig[0].description} |  | ||||||
|                   onClick={() => |  | ||||||
|                     maybeIconConfig[0].onClick(configCallbackProps) |  | ||||||
|                   } |  | ||||||
|                 > |                 > | ||||||
|                   <span |                   <ActionButton | ||||||
|                     className={!maybeIconConfig[0].showTitle ? 'sr-only' : ''} |                     Element="button" | ||||||
|  |                     id={maybeIconConfig[0].id} | ||||||
|  |                     data-testid={maybeIconConfig[0].id} | ||||||
|  |                     iconStart={{ | ||||||
|  |                       icon: maybeIconConfig[0].icon, | ||||||
|  |                       className: iconClassName, | ||||||
|  |                       bgClassName: bgClassName, | ||||||
|  |                     }} | ||||||
|  |                     className={ | ||||||
|  |                       '!border-transparent !px-0 pressed:!text-chalkboard-10 pressed:enabled:hovered:!text-chalkboard-10 ' + | ||||||
|  |                       buttonBgClassName | ||||||
|  |                     } | ||||||
|  |                     aria-pressed={maybeIconConfig[0].isActive} | ||||||
|  |                     disabled={ | ||||||
|  |                       disableAllButtons || | ||||||
|  |                       maybeIconConfig[0].status !== 'available' || | ||||||
|  |                       maybeIconConfig[0].disabled | ||||||
|  |                     } | ||||||
|  |                     name={maybeIconConfig[0].title} | ||||||
|  |                     // aria-description is still in ARIA 1.3 draft. | ||||||
|  |                     // eslint-disable-next-line jsx-a11y/aria-props | ||||||
|  |                     aria-description={maybeIconConfig[0].description} | ||||||
|  |                     onClick={() => | ||||||
|  |                       maybeIconConfig[0].onClick(configCallbackProps) | ||||||
|  |                     } | ||||||
|                   > |                   > | ||||||
|                     {maybeIconConfig[0].title} |                     <span | ||||||
|                   </span> |                       className={!maybeIconConfig[0].showTitle ? 'sr-only' : ''} | ||||||
|                 </ActionButton> |                     > | ||||||
|                 <ToolbarItemTooltip |                       {maybeIconConfig[0].title} | ||||||
|                   itemConfig={maybeIconConfig[0]} |                     </span> | ||||||
|                   configCallbackProps={configCallbackProps} |                     <ToolbarItemTooltip | ||||||
|                 /> |                       itemConfig={maybeIconConfig[0]} | ||||||
|  |                       configCallbackProps={configCallbackProps} | ||||||
|  |                       wrapperClassName="ui-open:!hidden" | ||||||
|  |                       contentClassName={tooltipContentClassName} | ||||||
|  |                     > | ||||||
|  |                       {showRichContent ? ( | ||||||
|  |                         <ToolbarItemTooltipRichContent | ||||||
|  |                           itemConfig={maybeIconConfig[0]} | ||||||
|  |                         /> | ||||||
|  |                       ) : ( | ||||||
|  |                         <ToolbarItemTooltipShortContent | ||||||
|  |                           status={maybeIconConfig[0].status} | ||||||
|  |                           title={maybeIconConfig[0].title} | ||||||
|  |                           hotkey={maybeIconConfig[0].hotkey} | ||||||
|  |                         /> | ||||||
|  |                       )} | ||||||
|  |                     </ToolbarItemTooltip> | ||||||
|  |                   </ActionButton> | ||||||
|  |                 </div> | ||||||
|               </ActionButtonDropdown> |               </ActionButtonDropdown> | ||||||
|             ) |             ) | ||||||
|           } |           } | ||||||
| @ -218,7 +271,13 @@ export function Toolbar({ | |||||||
|  |  | ||||||
|           // A single button |           // A single button | ||||||
|           return ( |           return ( | ||||||
|             <div className="relative" key={itemConfig.id}> |             <div | ||||||
|  |               className="relative" | ||||||
|  |               key={itemConfig.id} | ||||||
|  |               // Mouse events do not fire on disabled buttons | ||||||
|  |               onMouseEnter={handleMouseEnter} | ||||||
|  |               onMouseLeave={handleMouseLeave} | ||||||
|  |             > | ||||||
|               <ActionButton |               <ActionButton | ||||||
|                 Element="button" |                 Element="button" | ||||||
|                 key={itemConfig.id} |                 key={itemConfig.id} | ||||||
| @ -255,7 +314,18 @@ export function Toolbar({ | |||||||
|               <ToolbarItemTooltip |               <ToolbarItemTooltip | ||||||
|                 itemConfig={itemConfig} |                 itemConfig={itemConfig} | ||||||
|                 configCallbackProps={configCallbackProps} |                 configCallbackProps={configCallbackProps} | ||||||
|               /> |                 contentClassName={tooltipContentClassName} | ||||||
|  |               > | ||||||
|  |                 {showRichContent ? ( | ||||||
|  |                   <ToolbarItemTooltipRichContent itemConfig={itemConfig} /> | ||||||
|  |                 ) : ( | ||||||
|  |                   <ToolbarItemTooltipShortContent | ||||||
|  |                     status={itemConfig.status} | ||||||
|  |                     title={itemConfig.title} | ||||||
|  |                     hotkey={itemConfig.hotkey} | ||||||
|  |                   /> | ||||||
|  |                 )} | ||||||
|  |               </ToolbarItemTooltip> | ||||||
|             </div> |             </div> | ||||||
|           ) |           ) | ||||||
|         })} |         })} | ||||||
| @ -269,6 +339,12 @@ export function Toolbar({ | |||||||
|   ) |   ) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | interface ToolbarItemContentsProps extends React.PropsWithChildren { | ||||||
|  |   itemConfig: ToolbarItemResolved | ||||||
|  |   configCallbackProps: ToolbarItemCallbackProps | ||||||
|  |   wrapperClassName?: string | ||||||
|  |   contentClassName?: string | ||||||
|  | } | ||||||
| /** | /** | ||||||
|  * The single button and dropdown button share content, so we extract it here |  * The single button and dropdown button share content, so we extract it here | ||||||
|  * It contains a tooltip with the title, description, and links |  * It contains a tooltip with the title, description, and links | ||||||
| @ -277,12 +353,10 @@ export function Toolbar({ | |||||||
| const ToolbarItemTooltip = memo(function ToolbarItemContents({ | const ToolbarItemTooltip = memo(function ToolbarItemContents({ | ||||||
|   itemConfig, |   itemConfig, | ||||||
|   configCallbackProps, |   configCallbackProps, | ||||||
| }: { |   wrapperClassName = '', | ||||||
|   itemConfig: ToolbarItemResolved |   contentClassName = '', | ||||||
|   configCallbackProps: ToolbarItemCallbackProps |   children, | ||||||
| }) { | }: ToolbarItemContentsProps) { | ||||||
|   const { state } = useModelingContext() |  | ||||||
|  |  | ||||||
|   useHotkeys( |   useHotkeys( | ||||||
|     itemConfig.hotkey || '', |     itemConfig.hotkey || '', | ||||||
|     () => { |     () => { | ||||||
| @ -305,11 +379,50 @@ const ToolbarItemTooltip = memo(function ToolbarItemContents({ | |||||||
|           ? ({ '-webkit-app-region': 'no-drag' } as React.CSSProperties) |           ? ({ '-webkit-app-region': 'no-drag' } as React.CSSProperties) | ||||||
|           : {} |           : {} | ||||||
|       } |       } | ||||||
|  |       hoverOnly | ||||||
|       position="bottom" |       position="bottom" | ||||||
|       wrapperClassName="!p-4 !pointer-events-auto" |       wrapperClassName={'!p-4 !pointer-events-auto ' + wrapperClassName} | ||||||
|       contentClassName="!text-left text-wrap !text-xs !p-0 !pb-2 flex gap-2 !max-w-none !w-72 flex-col items-stretch" |       contentClassName={contentClassName} | ||||||
|  |       delay={0} | ||||||
|     > |     > | ||||||
|  |       {children} | ||||||
|  |     </Tooltip> | ||||||
|  |   ) | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | const ToolbarItemTooltipShortContent = ({ | ||||||
|  |   status, | ||||||
|  |   title, | ||||||
|  |   hotkey, | ||||||
|  | }: { | ||||||
|  |   status: string | ||||||
|  |   title: string | ||||||
|  |   hotkey?: string | string[] | ||||||
|  | }) => ( | ||||||
|  |   <span | ||||||
|  |     className={`text-sm ${ | ||||||
|  |       status !== 'available' ? 'text-chalkboard-70 dark:text-chalkboard-40' : '' | ||||||
|  |     }`} | ||||||
|  |   > | ||||||
|  |     {title} | ||||||
|  |     {hotkey && ( | ||||||
|  |       <kbd className="inline-block ml-2 flex-none hotkey">{hotkey}</kbd> | ||||||
|  |     )} | ||||||
|  |   </span> | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | const ToolbarItemTooltipRichContent = ({ | ||||||
|  |   itemConfig, | ||||||
|  | }: { | ||||||
|  |   itemConfig: ToolbarItemResolved | ||||||
|  | }) => { | ||||||
|  |   const { state } = useModelingContext() | ||||||
|  |   return ( | ||||||
|  |     <> | ||||||
|       <div className="rounded-top flex items-center gap-2 pt-3 pb-2 px-2 bg-chalkboard-20/50 dark:bg-chalkboard-80/50"> |       <div className="rounded-top flex items-center gap-2 pt-3 pb-2 px-2 bg-chalkboard-20/50 dark:bg-chalkboard-80/50"> | ||||||
|  |         {itemConfig.icon && ( | ||||||
|  |           <CustomIcon className="w-5 h-5" name={itemConfig.icon} /> | ||||||
|  |         )} | ||||||
|         <span |         <span | ||||||
|           className={`text-sm flex-1 ${ |           className={`text-sm flex-1 ${ | ||||||
|             itemConfig.status !== 'available' |             itemConfig.status !== 'available' | ||||||
| @ -378,6 +491,6 @@ const ToolbarItemTooltip = memo(function ToolbarItemContents({ | |||||||
|           </ul> |           </ul> | ||||||
|         </> |         </> | ||||||
|       )} |       )} | ||||||
|     </Tooltip> |     </> | ||||||
|   ) |   ) | ||||||
| }) | } | ||||||
|  | |||||||
| @ -108,6 +108,8 @@ export class CameraControls { | |||||||
|   interactionGuards: MouseGuard = cameraMouseDragGuards.Zoo |   interactionGuards: MouseGuard = cameraMouseDragGuards.Zoo | ||||||
|   isFovAnimationInProgress = false |   isFovAnimationInProgress = false | ||||||
|   perspectiveFovBeforeOrtho = 45 |   perspectiveFovBeforeOrtho = 45 | ||||||
|  |   // NOTE: Duplicated state across Provider and singleton. Mapped from settingsMachine | ||||||
|  |   _setting_allowOrbitInSketchMode = false | ||||||
|   get isPerspective() { |   get isPerspective() { | ||||||
|     return this.camera instanceof PerspectiveCamera |     return this.camera instanceof PerspectiveCamera | ||||||
|   } |   } | ||||||
|  | |||||||
| @ -25,13 +25,13 @@ import { | |||||||
|   CallExpression, |   CallExpression, | ||||||
|   PathToNode, |   PathToNode, | ||||||
|   Program, |   Program, | ||||||
|   SourceRange, |  | ||||||
|   Expr, |   Expr, | ||||||
|   parse, |   parse, | ||||||
|   recast, |   recast, | ||||||
|   defaultSourceRange, |   defaultSourceRange, | ||||||
|   resultIsOk, |   resultIsOk, | ||||||
|   ProgramMemory, |   ProgramMemory, | ||||||
|  |   topLevelRange, | ||||||
| } from 'lang/wasm' | } from 'lang/wasm' | ||||||
| import { CustomIcon, CustomIconName } from 'components/CustomIcon' | import { CustomIcon, CustomIconName } from 'components/CustomIcon' | ||||||
| import { ConstrainInfo } from 'lang/std/stdTypes' | import { ConstrainInfo } from 'lang/std/stdTypes' | ||||||
| @ -46,8 +46,8 @@ import { | |||||||
| } from 'lang/modifyAst' | } from 'lang/modifyAst' | ||||||
| import { ActionButton } from 'components/ActionButton' | import { ActionButton } from 'components/ActionButton' | ||||||
| import { err, reportRejection, trap } from 'lib/trap' | import { err, reportRejection, trap } from 'lib/trap' | ||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' |  | ||||||
| import { Node } from 'wasm-lib/kcl/bindings/Node' | import { Node } from 'wasm-lib/kcl/bindings/Node' | ||||||
|  | import { commandBarActor } from 'machines/commandBarMachine' | ||||||
|  |  | ||||||
| function useShouldHideScene(): { hideClient: boolean; hideServer: boolean } { | function useShouldHideScene(): { hideClient: boolean; hideServer: boolean } { | ||||||
|   const [isCamMoving, setIsCamMoving] = useState(false) |   const [isCamMoving, setIsCamMoving] = useState(false) | ||||||
| @ -510,7 +510,6 @@ const ConstraintSymbol = ({ | |||||||
|   constrainInfo: ConstrainInfo |   constrainInfo: ConstrainInfo | ||||||
|   verticalPosition: 'top' | 'bottom' |   verticalPosition: 'top' | 'bottom' | ||||||
| }) => { | }) => { | ||||||
|   const { commandBarSend } = useCommandsContext() |  | ||||||
|   const { context } = useModelingContext() |   const { context } = useModelingContext() | ||||||
|   const varNameMap: { |   const varNameMap: { | ||||||
|     [key in ConstrainInfo['type']]: { |     [key in ConstrainInfo['type']]: { | ||||||
| @ -600,8 +599,8 @@ const ConstraintSymbol = ({ | |||||||
|   if (err(_node)) return |   if (err(_node)) return | ||||||
|   const node = _node.node |   const node = _node.node | ||||||
|  |  | ||||||
|   const range: SourceRange = node |   const range = node | ||||||
|     ? [node.start, node.end, true] |     ? topLevelRange(node.start, node.end) | ||||||
|     : defaultSourceRange() |     : defaultSourceRange() | ||||||
|  |  | ||||||
|   if (_type === 'intersectionTag') return null |   if (_type === 'intersectionTag') return null | ||||||
| @ -630,7 +629,7 @@ const ConstraintSymbol = ({ | |||||||
|         // disabled={implicitDesc} TODO why does this change styles that are hard to override? |         // disabled={implicitDesc} TODO why does this change styles that are hard to override? | ||||||
|         onClick={toSync(async () => { |         onClick={toSync(async () => { | ||||||
|           if (!isConstrained) { |           if (!isConstrained) { | ||||||
|             commandBarSend({ |             commandBarActor.send({ | ||||||
|               type: 'Find and select command', |               type: 'Find and select command', | ||||||
|               data: { |               data: { | ||||||
|                 name: 'Constrain with named value', |                 name: 'Constrain with named value', | ||||||
| @ -756,7 +755,6 @@ export const CamDebugSettings = () => { | |||||||
|     sceneInfra.camControls.reactCameraProperties |     sceneInfra.camControls.reactCameraProperties | ||||||
|   ) |   ) | ||||||
|   const [fov, setFov] = useState(12) |   const [fov, setFov] = useState(12) | ||||||
|   const { commandBarSend } = useCommandsContext() |  | ||||||
|  |  | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     sceneInfra.camControls.setReactCameraPropertiesCallback(setCamSettings) |     sceneInfra.camControls.setReactCameraPropertiesCallback(setCamSettings) | ||||||
| @ -775,7 +773,7 @@ export const CamDebugSettings = () => { | |||||||
|         type="checkbox" |         type="checkbox" | ||||||
|         checked={camSettings.type === 'perspective'} |         checked={camSettings.type === 'perspective'} | ||||||
|         onChange={() => |         onChange={() => | ||||||
|           commandBarSend({ |           commandBarActor.send({ | ||||||
|             type: 'Find and select command', |             type: 'Find and select command', | ||||||
|             data: { |             data: { | ||||||
|               groupId: 'settings', |               groupId: 'settings', | ||||||
|  | |||||||
| @ -1,5 +1,6 @@ | |||||||
| import { | import { | ||||||
|   BoxGeometry, |   BoxGeometry, | ||||||
|  |   Color, | ||||||
|   DoubleSide, |   DoubleSide, | ||||||
|   Group, |   Group, | ||||||
|   Intersection, |   Intersection, | ||||||
| @ -58,7 +59,9 @@ import { | |||||||
|   sourceRangeFromRust, |   sourceRangeFromRust, | ||||||
|   resultIsOk, |   resultIsOk, | ||||||
|   SourceRange, |   SourceRange, | ||||||
|  |   topLevelRange, | ||||||
| } from 'lang/wasm' | } from 'lang/wasm' | ||||||
|  | import { calculate_circle_from_3_points } from '../wasm-lib/pkg/wasm_lib' | ||||||
| import { | import { | ||||||
|   engineCommandManager, |   engineCommandManager, | ||||||
|   kclManager, |   kclManager, | ||||||
| @ -70,7 +73,7 @@ import { getNodeFromPath, getNodePathFromSourceRange } from 'lang/queryAst' | |||||||
| import { executeAst, ToolTip } from 'lang/langHelpers' | import { executeAst, ToolTip } from 'lang/langHelpers' | ||||||
| import { | import { | ||||||
|   createProfileStartHandle, |   createProfileStartHandle, | ||||||
|   createArcGeometry, |   createCircleGeometry, | ||||||
|   SegmentUtils, |   SegmentUtils, | ||||||
|   segmentUtils, |   segmentUtils, | ||||||
| } from './segments' | } from './segments' | ||||||
| @ -109,6 +112,8 @@ import { CSS2DObject } from 'three/examples/jsm/renderers/CSS2DRenderer' | |||||||
| import { Point3d } from 'wasm-lib/kcl/bindings/Point3d' | import { Point3d } from 'wasm-lib/kcl/bindings/Point3d' | ||||||
| import { SegmentInputs } from 'lang/std/stdTypes' | import { SegmentInputs } from 'lang/std/stdTypes' | ||||||
| import { Node } from 'wasm-lib/kcl/bindings/Node' | import { Node } from 'wasm-lib/kcl/bindings/Node' | ||||||
|  | import { LabeledArg } from 'wasm-lib/kcl/bindings/LabeledArg' | ||||||
|  | import { Literal } from 'wasm-lib/kcl/bindings/Literal' | ||||||
| import { radToDeg } from 'three/src/math/MathUtils' | import { radToDeg } from 'three/src/math/MathUtils' | ||||||
| import { getArtifactFromRange, codeRefFromRange } from 'lang/std/artifactGraph' | import { getArtifactFromRange, codeRefFromRange } from 'lang/std/artifactGraph' | ||||||
|  |  | ||||||
| @ -624,7 +629,7 @@ export class SceneEntities { | |||||||
|  |  | ||||||
|       const startRange = _node1.node.start |       const startRange = _node1.node.start | ||||||
|       const endRange = _node1.node.end |       const endRange = _node1.node.end | ||||||
|       const sourceRange: SourceRange = [startRange, endRange, true] |       const sourceRange = topLevelRange(startRange, endRange) | ||||||
|       const selection: Selections = computeSelectionFromSourceRangeAndAST( |       const selection: Selections = computeSelectionFromSourceRangeAndAST( | ||||||
|         sourceRange, |         sourceRange, | ||||||
|         maybeModdedAst |         maybeModdedAst | ||||||
| @ -1261,110 +1266,98 @@ export class SceneEntities { | |||||||
|     const groupOfDrafts = new Group() |     const groupOfDrafts = new Group() | ||||||
|     groupOfDrafts.name = 'circle-3-point-group' |     groupOfDrafts.name = 'circle-3-point-group' | ||||||
|     groupOfDrafts.position.copy(sketchOrigin) |     groupOfDrafts.position.copy(sketchOrigin) | ||||||
|  |  | ||||||
|     // lee: I'm keeping this here as a developer gotchya: |     // lee: I'm keeping this here as a developer gotchya: | ||||||
|     // Do not reorient your surfaces to the intersection plane. Your points are |     // If you use 3D points, do not rotate anything. | ||||||
|     // already in 3D space, not 2D. If you intersect say XZ, you want the points |     // If you use 2D points (easier to deal with, generally do this!), then | ||||||
|     // to continue to live at the 3D intersection point, not be rotated to end |     // rotate the group just like this! Remember to rotate other groups too! | ||||||
|     // up elsewhere! |     groupOfDrafts.setRotationFromQuaternion(orientation) | ||||||
|     // groupOfDrafts.setRotationFromQuaternion(orientation) |  | ||||||
|     this.scene.add(groupOfDrafts) |     this.scene.add(groupOfDrafts) | ||||||
|  |  | ||||||
|     const DRAFT_POINT_RADIUS = 6 |     // How large the points on the circle will render as | ||||||
|  |     const DRAFT_POINT_RADIUS = 10 // px | ||||||
|  |  | ||||||
|     const createPoint = (center: Vector3): number => { |     // The target of our dragging | ||||||
|  |     let target: Object3D | undefined = undefined | ||||||
|  |  | ||||||
|  |     // The KCL this will generate. | ||||||
|  |     const kclCircle3Point = parse(`circleThreePoint( | ||||||
|  |       p1 = [0.0, 0.0], | ||||||
|  |       p2 = [0.0, 0.0], | ||||||
|  |       p3 = [0.0, 0.0], | ||||||
|  |     )`) | ||||||
|  |  | ||||||
|  |     const createPoint = ( | ||||||
|  |       center: Vector3, | ||||||
|  |       opts?: { noInteraction?: boolean } | ||||||
|  |     ): Mesh => { | ||||||
|       const geometry = new SphereGeometry(DRAFT_POINT_RADIUS) |       const geometry = new SphereGeometry(DRAFT_POINT_RADIUS) | ||||||
|       const color = getThemeColorForThreeJs(sceneInfra._theme) |       const color = getThemeColorForThreeJs(sceneInfra._theme) | ||||||
|       const material = new MeshBasicMaterial({ color }) |  | ||||||
|  |       const material = new MeshBasicMaterial({ | ||||||
|  |         color: opts?.noInteraction | ||||||
|  |           ? sceneInfra._theme === 'light' | ||||||
|  |             ? new Color(color).multiplyScalar(0.15) | ||||||
|  |             : new Color(0x010101).multiplyScalar(2000) | ||||||
|  |           : color, | ||||||
|  |       }) | ||||||
|  |  | ||||||
|       const mesh = new Mesh(geometry, material) |       const mesh = new Mesh(geometry, material) | ||||||
|       mesh.userData = { type: CIRCLE_3_POINT_DRAFT_POINT } |       mesh.userData = { | ||||||
|  |         type: opts?.noInteraction ? 'ghost' : CIRCLE_3_POINT_DRAFT_POINT, | ||||||
|  |       } | ||||||
|  |       mesh.renderOrder = 1000 | ||||||
|       mesh.layers.set(SKETCH_LAYER) |       mesh.layers.set(SKETCH_LAYER) | ||||||
|       mesh.position.copy(center) |       mesh.position.copy(center) | ||||||
|       mesh.scale.set(scale, scale, scale) |       mesh.scale.set(scale, scale, scale) | ||||||
|       mesh.renderOrder = 100 |       mesh.renderOrder = 100 | ||||||
|  |  | ||||||
|       groupOfDrafts.add(mesh) |       return mesh | ||||||
|  |  | ||||||
|       return mesh.id |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const circle3Point = ( |     const createCircle3PointGraphic = async ( | ||||||
|       points: Vector2[] |       points: Vector2[], | ||||||
|     ): undefined | { center: Vector3; radius: number } => { |       center: Vector2, | ||||||
|       // A 3-point circle is undefined if it doesn't have 3 points :) |       radius: number | ||||||
|       if (points.length !== 3) return undefined |     ) => { | ||||||
|  |       if ( | ||||||
|       // y = (i/j)(x-h) + b |         Number.isNaN(radius) || | ||||||
|       // i and j variables for the slopes |         Number.isNaN(center.x) || | ||||||
|       const i = [points[1].x - points[0].x, points[2].x - points[1].x] |         Number.isNaN(center.y) | ||||||
|       const j = [points[1].y - points[0].y, points[2].y - points[1].y] |       ) | ||||||
|  |         return | ||||||
|       // Our / threejs coordinate system affects this a lot. If you take this |  | ||||||
|       // code into a different code base, you may have to adjust a/b to being |  | ||||||
|       // -1/a/b, b/a, etc! In this case, a/-b did the trick. |  | ||||||
|       const m = [i[0] / -j[0], i[1] / -j[1]] |  | ||||||
|  |  | ||||||
|       const h = [ |  | ||||||
|         (points[0].x + points[1].x) / 2, |  | ||||||
|         (points[1].x + points[2].x) / 2, |  | ||||||
|       ] |  | ||||||
|       const b = [ |  | ||||||
|         (points[0].y + points[1].y) / 2, |  | ||||||
|         (points[1].y + points[2].y) / 2, |  | ||||||
|       ] |  | ||||||
|  |  | ||||||
|       // Algebraically derived |  | ||||||
|       const x = (-m[0] * h[0] + b[0] - b[1] + m[1] * h[1]) / (m[1] - m[0]) |  | ||||||
|       const y = m[0] * (x - h[0]) + b[0] |  | ||||||
|  |  | ||||||
|       const center = new Vector3(x, y, 0) |  | ||||||
|       const radius = Math.sqrt((points[1].x - x) ** 2 + (points[1].y - y) ** 2) |  | ||||||
|  |  | ||||||
|       return { |  | ||||||
|         center, |  | ||||||
|         radius, |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // TO BE SHORT LIVED: unused function to draw the circle and lines. |  | ||||||
|     // @ts-ignore |  | ||||||
|     // eslint-disable-next-line |  | ||||||
|     const createCircle3Point = (points: Vector2[]) => { |  | ||||||
|       const circleParams = circle3Point(points) |  | ||||||
|  |  | ||||||
|       // A circle cannot be created for these points. |  | ||||||
|       if (!circleParams) return |  | ||||||
|  |  | ||||||
|       const color = getThemeColorForThreeJs(sceneInfra._theme) |       const color = getThemeColorForThreeJs(sceneInfra._theme) | ||||||
|       const geometryCircle = createArcGeometry({ |       const lineCircle = createCircleGeometry({ | ||||||
|         center: [circleParams.center.x, circleParams.center.y], |         center: [center.x, center.y], | ||||||
|         radius: circleParams.radius, |         radius, | ||||||
|         startAngle: 0, |         color, | ||||||
|         endAngle: Math.PI * 2, |         isDashed: false, | ||||||
|         ccw: true, |         scale: 1, | ||||||
|         isDashed: true, |  | ||||||
|         scale, |  | ||||||
|       }) |       }) | ||||||
|       const materialCircle = new MeshBasicMaterial({ color }) |       lineCircle.userData = { type: CIRCLE_3_POINT_DRAFT_CIRCLE } | ||||||
|  |       lineCircle.layers.set(SKETCH_LAYER) | ||||||
|  |       // devnote: it's a mistake to use these with EllipseCurve :) | ||||||
|  |       // lineCircle.position.set(center.x, center.y, 0) | ||||||
|  |       // lineCircle.scale.set(scale, scale, scale) | ||||||
|  |  | ||||||
|       if (groupCircle) groupOfDrafts.remove(groupCircle) |       if (groupCircle) groupOfDrafts.remove(groupCircle) | ||||||
|       groupCircle = new Group() |       groupCircle = new Group() | ||||||
|       groupCircle.renderOrder = 1 |       groupCircle.renderOrder = 1 | ||||||
|  |       groupCircle.add(lineCircle) | ||||||
|  |  | ||||||
|       const meshCircle = new Mesh(geometryCircle, materialCircle) |       const pointMesh = createPoint(new Vector3(center.x, center.y, 0), { | ||||||
|       meshCircle.userData = { type: CIRCLE_3_POINT_DRAFT_CIRCLE } |         noInteraction: true, | ||||||
|       meshCircle.layers.set(SKETCH_LAYER) |       }) | ||||||
|       meshCircle.position.set(circleParams.center.x, circleParams.center.y, 0) |       groupCircle.add(pointMesh) | ||||||
|       meshCircle.scale.set(scale, scale, scale) |  | ||||||
|       groupCircle.add(meshCircle) |  | ||||||
|  |  | ||||||
|       const geometryPolyLine = new BufferGeometry().setFromPoints([ |       const geometryPolyLine = new BufferGeometry().setFromPoints([ | ||||||
|         ...points, |         ...points.map((p) => new Vector3(p.x, p.y, 0)), | ||||||
|         points[0], |         new Vector3(points[0].x, points[0].y, 0), | ||||||
|       ]) |       ]) | ||||||
|       const materialPolyLine = new LineDashedMaterial({ |       const materialPolyLine = new LineDashedMaterial({ | ||||||
|         color, |         color, | ||||||
|         scale, |         scale: 1 / scale, | ||||||
|         dashSize: 6, |         dashSize: 6, | ||||||
|         gapSize: 6, |         gapSize: 6, | ||||||
|       }) |       }) | ||||||
| @ -1375,13 +1368,146 @@ export class SceneEntities { | |||||||
|       groupOfDrafts.add(groupCircle) |       groupOfDrafts.add(groupCircle) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // The target of our dragging |     const insertCircle3PointKclIntoAstSnapshot = ( | ||||||
|     let target: Object3D | undefined = undefined |       points: Vector2[] | ||||||
|  |     ): Program => { | ||||||
|  |       if (err(kclCircle3Point) || kclCircle3Point.program === null) | ||||||
|  |         return kclManager.ast | ||||||
|  |       if (kclCircle3Point.program.body[0].type !== 'ExpressionStatement') | ||||||
|  |         return kclManager.ast | ||||||
|  |       if ( | ||||||
|  |         kclCircle3Point.program.body[0].expression.type !== 'CallExpressionKw' | ||||||
|  |       ) | ||||||
|  |         return kclManager.ast | ||||||
|  |  | ||||||
|  |       const arg = (x: LabeledArg): Literal[] | undefined => { | ||||||
|  |         if ( | ||||||
|  |           'arg' in x && | ||||||
|  |           'elements' in x.arg && | ||||||
|  |           x.arg.type === 'ArrayExpression' | ||||||
|  |         ) { | ||||||
|  |           if (x.arg.elements.every((x) => x.type === 'Literal')) { | ||||||
|  |             return x.arg.elements | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |         return undefined | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       const kclCircle3PointArgs = | ||||||
|  |         kclCircle3Point.program.body[0].expression.arguments | ||||||
|  |  | ||||||
|  |       const arg0 = arg(kclCircle3PointArgs[0]) | ||||||
|  |       if (!arg0) return kclManager.ast | ||||||
|  |       arg0[0].value = { value: points[0].x, suffix: 'None' } | ||||||
|  |       arg0[0].raw = points[0].x.toString() | ||||||
|  |       arg0[1].value = { value: points[0].y, suffix: 'None' } | ||||||
|  |       arg0[1].raw = points[0].y.toString() | ||||||
|  |  | ||||||
|  |       const arg1 = arg(kclCircle3PointArgs[1]) | ||||||
|  |       if (!arg1) return kclManager.ast | ||||||
|  |       arg1[0].value = { value: points[1].x, suffix: 'None' } | ||||||
|  |       arg1[0].raw = points[1].x.toString() | ||||||
|  |       arg1[1].value = { value: points[1].y, suffix: 'None' } | ||||||
|  |       arg1[1].raw = points[1].y.toString() | ||||||
|  |  | ||||||
|  |       const arg2 = arg(kclCircle3PointArgs[2]) | ||||||
|  |       if (!arg2) return kclManager.ast | ||||||
|  |       arg2[0].value = { value: points[2].x, suffix: 'None' } | ||||||
|  |       arg2[0].raw = points[2].x.toString() | ||||||
|  |       arg2[1].value = { value: points[2].y, suffix: 'None' } | ||||||
|  |       arg2[1].raw = points[2].y.toString() | ||||||
|  |  | ||||||
|  |       const astSnapshot = structuredClone(kclManager.ast) | ||||||
|  |       const startSketchOnASTNode = getNodeFromPath<VariableDeclaration>( | ||||||
|  |         astSnapshot, | ||||||
|  |         startSketchOnASTNodePath, | ||||||
|  |         'VariableDeclaration' | ||||||
|  |       ) | ||||||
|  |       if (err(startSketchOnASTNode)) return astSnapshot | ||||||
|  |  | ||||||
|  |       // It's possible we're already dealing with a PipeExpression. | ||||||
|  |       // Modify the current one. | ||||||
|  |       if ( | ||||||
|  |         startSketchOnASTNode.node.declaration.init.type === 'PipeExpression' && | ||||||
|  |         startSketchOnASTNode.node.declaration.init.body[1].type === | ||||||
|  |           'CallExpressionKw' && | ||||||
|  |         startSketchOnASTNode.node.declaration.init.body.length >= 2 | ||||||
|  |       ) { | ||||||
|  |         startSketchOnASTNode.node.declaration.init.body[1].arguments = | ||||||
|  |           kclCircle3Point.program.body[0].expression.arguments | ||||||
|  |       } else { | ||||||
|  |         // Clone a new node based on the old, and replace the old with the new. | ||||||
|  |         const clonedStartSketchOnASTNode = structuredClone(startSketchOnASTNode) | ||||||
|  |         startSketchOnASTNode.node.declaration.init = createPipeExpression([ | ||||||
|  |           clonedStartSketchOnASTNode.node.declaration.init, | ||||||
|  |           kclCircle3Point.program.body[0].expression, | ||||||
|  |         ]) | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       // Return the `Program` | ||||||
|  |       return astSnapshot | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const updateCircle3Point = async (opts?: { execute?: true }) => { | ||||||
|  |       const points_ = Array.from(points.values()) | ||||||
|  |       const circleParams = calculate_circle_from_3_points( | ||||||
|  |         points_[0].x, | ||||||
|  |         points_[0].y, | ||||||
|  |         points_[1].x, | ||||||
|  |         points_[1].y, | ||||||
|  |         points_[2].x, | ||||||
|  |         points_[2].y | ||||||
|  |       ) | ||||||
|  |  | ||||||
|  |       if (Number.isNaN(circleParams.radius)) return | ||||||
|  |  | ||||||
|  |       await createCircle3PointGraphic( | ||||||
|  |         points_, | ||||||
|  |         new Vector2(circleParams.center_x, circleParams.center_y), | ||||||
|  |         circleParams.radius | ||||||
|  |       ) | ||||||
|  |       const astWithNewCode = insertCircle3PointKclIntoAstSnapshot(points_) | ||||||
|  |       const codeAsString = recast(astWithNewCode) | ||||||
|  |       if (err(codeAsString)) return | ||||||
|  |       codeManager.updateCodeStateEditor(codeAsString) | ||||||
|  |     } | ||||||
|  |  | ||||||
|     const cleanupFn = () => { |     const cleanupFn = () => { | ||||||
|       this.scene.remove(groupOfDrafts) |       this.scene.remove(groupOfDrafts) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     // The AST node we extracted earlier may already have a circleThreePoint! | ||||||
|  |     // Use the points in the AST as starting points. | ||||||
|  |     const astSnapshot = structuredClone(kclManager.ast) | ||||||
|  |     const maybeVariableDeclaration = getNodeFromPath<VariableDeclaration>( | ||||||
|  |       astSnapshot, | ||||||
|  |       startSketchOnASTNodePath, | ||||||
|  |       'VariableDeclaration' | ||||||
|  |     ) | ||||||
|  |     if (err(maybeVariableDeclaration)) | ||||||
|  |       return () => { | ||||||
|  |         done() | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |     const maybeCallExpressionKw = maybeVariableDeclaration.node.declaration.init | ||||||
|  |     if ( | ||||||
|  |       maybeCallExpressionKw.type === 'PipeExpression' && | ||||||
|  |       maybeCallExpressionKw.body[1].type === 'CallExpressionKw' && | ||||||
|  |       maybeCallExpressionKw.body[1]?.callee.name === 'circleThreePoint' | ||||||
|  |     ) { | ||||||
|  |       maybeCallExpressionKw?.body[1].arguments | ||||||
|  |         .map( | ||||||
|  |           ({ arg }: any) => | ||||||
|  |             new Vector2(arg.elements[0].value, arg.elements[1].value) | ||||||
|  |         ) | ||||||
|  |         .forEach((point: Vector2) => { | ||||||
|  |           const pointMesh = createPoint(new Vector3(point.x, point.y, 0)) | ||||||
|  |           groupOfDrafts.add(pointMesh) | ||||||
|  |           points.set(pointMesh.id, point) | ||||||
|  |         }) | ||||||
|  |       void updateCircle3Point() | ||||||
|  |     } | ||||||
|  |  | ||||||
|     sceneInfra.setCallbacks({ |     sceneInfra.setCallbacks({ | ||||||
|       async onDrag(args) { |       async onDrag(args) { | ||||||
|         const draftPointsIntersected = args.intersects.filter( |         const draftPointsIntersected = args.intersects.filter( | ||||||
| @ -1397,8 +1523,18 @@ export class SceneEntities { | |||||||
|         // The user was off their mark! Missed the object to select. |         // The user was off their mark! Missed the object to select. | ||||||
|         if (!target) return |         if (!target) return | ||||||
|  |  | ||||||
|         target.position.copy(args.intersectionPoint.threeD) |         target.position.copy( | ||||||
|  |           new Vector3( | ||||||
|  |             args.intersectionPoint.twoD.x, | ||||||
|  |             args.intersectionPoint.twoD.y, | ||||||
|  |             0 | ||||||
|  |           ) | ||||||
|  |         ) | ||||||
|         points.set(target.id, args.intersectionPoint.twoD) |         points.set(target.id, args.intersectionPoint.twoD) | ||||||
|  |  | ||||||
|  |         if (points.size <= 2) return | ||||||
|  |  | ||||||
|  |         await updateCircle3Point() | ||||||
|       }, |       }, | ||||||
|       async onDragEnd(_args) { |       async onDragEnd(_args) { | ||||||
|         target = undefined |         target = undefined | ||||||
| @ -1407,45 +1543,19 @@ export class SceneEntities { | |||||||
|         if (points.size >= 3) return |         if (points.size >= 3) return | ||||||
|         if (!args.intersectionPoint) return |         if (!args.intersectionPoint) return | ||||||
|  |  | ||||||
|         const id = createPoint(args.intersectionPoint.threeD) |         const pointMesh = createPoint( | ||||||
|         points.set(id, args.intersectionPoint.twoD) |           new Vector3( | ||||||
|  |             args.intersectionPoint.twoD.x, | ||||||
|         if (points.size < 2) return |             args.intersectionPoint.twoD.y, | ||||||
|  |             0 | ||||||
|         // We've now got 3 points, let's create our circle! |           ) | ||||||
|         const astSnapshot = structuredClone(kclManager.ast) |  | ||||||
|         let nodeQueryResult |  | ||||||
|         nodeQueryResult = getNodeFromPath<VariableDeclaration>( |  | ||||||
|           astSnapshot, |  | ||||||
|           startSketchOnASTNodePath, |  | ||||||
|           'VariableDeclaration' |  | ||||||
|         ) |         ) | ||||||
|         if (err(nodeQueryResult)) return Promise.reject(nodeQueryResult) |         groupOfDrafts.add(pointMesh) | ||||||
|         const startSketchOnASTNode = nodeQueryResult |         points.set(pointMesh.id, args.intersectionPoint.twoD) | ||||||
|  |  | ||||||
|         const circleParams = circle3Point(Array.from(points.values())) |         if (points.size <= 2) return | ||||||
|  |  | ||||||
|         if (!circleParams) return |         await updateCircle3Point() | ||||||
|  |  | ||||||
|         const kclCircle3Point = parse(`circle({ |  | ||||||
|             center = [${circleParams.center.x}, ${circleParams.center.y}], |  | ||||||
|             radius = ${circleParams.radius}, |  | ||||||
|           }, %)`) |  | ||||||
|  |  | ||||||
|         if (err(kclCircle3Point) || kclCircle3Point.program === null) return |  | ||||||
|         if (kclCircle3Point.program.body[0].type !== 'ExpressionStatement') |  | ||||||
|           return |  | ||||||
|  |  | ||||||
|         const clonedStartSketchOnASTNode = structuredClone(startSketchOnASTNode) |  | ||||||
|         startSketchOnASTNode.node.declaration.init = createPipeExpression([ |  | ||||||
|           clonedStartSketchOnASTNode.node.declaration.init, |  | ||||||
|           kclCircle3Point.program.body[0].expression, |  | ||||||
|         ]) |  | ||||||
|  |  | ||||||
|         await kclManager.executeAstMock(astSnapshot) |  | ||||||
|         await codeManager.updateEditorWithAstAndWriteToFile(astSnapshot) |  | ||||||
|  |  | ||||||
|         done() |  | ||||||
|       }, |       }, | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
| @ -1903,7 +2013,7 @@ export class SceneEntities { | |||||||
|         kclManager.programMemory, |         kclManager.programMemory, | ||||||
|         { |         { | ||||||
|           type: 'sourceRange', |           type: 'sourceRange', | ||||||
|           sourceRange: [node.start, node.end, true], |           sourceRange: topLevelRange(node.start, node.end), | ||||||
|         }, |         }, | ||||||
|         getChangeSketchInput() |         getChangeSketchInput() | ||||||
|       ) |       ) | ||||||
| @ -1941,8 +2051,8 @@ export class SceneEntities { | |||||||
|       ) |       ) | ||||||
|       if (!(sk instanceof Reason)) { |       if (!(sk instanceof Reason)) { | ||||||
|         sketch = sk |         sketch = sk | ||||||
|       } else if ((maybeSketch as Solid).sketch) { |       } else if (maybeSketch && (maybeSketch.value as Solid)?.sketch) { | ||||||
|         sketch = (maybeSketch as Solid).sketch |         sketch = (maybeSketch.value as Solid).sketch | ||||||
|       } |       } | ||||||
|       if (!sketch) return |       if (!sketch) return | ||||||
|  |  | ||||||
| @ -2154,7 +2264,7 @@ export class SceneEntities { | |||||||
|           ) |           ) | ||||||
|           if (trap(_node, { suppress: true })) return |           if (trap(_node, { suppress: true })) return | ||||||
|           const node = _node.node |           const node = _node.node | ||||||
|           editorManager.setHighlightRange([[node.start, node.end, true]]) |           editorManager.setHighlightRange([topLevelRange(node.start, node.end)]) | ||||||
|           const yellow = 0xffff00 |           const yellow = 0xffff00 | ||||||
|           colorSegment(selected, yellow) |           colorSegment(selected, yellow) | ||||||
|           const extraSegmentGroup = parent.getObjectByName(EXTRA_SEGMENT_HANDLE) |           const extraSegmentGroup = parent.getObjectByName(EXTRA_SEGMENT_HANDLE) | ||||||
| @ -2431,7 +2541,7 @@ export function sketchFromPathToNode({ | |||||||
|   const varDec = _varDec.node |   const varDec = _varDec.node | ||||||
|   const result = programMemory.get(varDec?.id?.name || '') |   const result = programMemory.get(varDec?.id?.name || '') | ||||||
|   if (result?.type === 'Solid') { |   if (result?.type === 'Solid') { | ||||||
|     return result.sketch |     return result.value.sketch | ||||||
|   } |   } | ||||||
|   const sg = sketchFromKclValue(result, varDec?.id?.name) |   const sg = sketchFromKclValue(result, varDec?.id?.name) | ||||||
|   if (err(sg)) { |   if (err(sg)) { | ||||||
|  | |||||||
| @ -9,6 +9,9 @@ import { | |||||||
|   ExtrudeGeometry, |   ExtrudeGeometry, | ||||||
|   Group, |   Group, | ||||||
|   LineCurve3, |   LineCurve3, | ||||||
|  |   LineBasicMaterial, | ||||||
|  |   LineDashedMaterial, | ||||||
|  |   Line, | ||||||
|   Mesh, |   Mesh, | ||||||
|   MeshBasicMaterial, |   MeshBasicMaterial, | ||||||
|   NormalBufferAttributes, |   NormalBufferAttributes, | ||||||
| @ -58,6 +61,7 @@ import { SegmentInputs } from 'lang/std/stdTypes' | |||||||
| import { err } from 'lib/trap' | import { err } from 'lib/trap' | ||||||
| import { editorManager, sceneInfra } from 'lib/singletons' | import { editorManager, sceneInfra } from 'lib/singletons' | ||||||
| import { Selections } from 'lib/selections' | import { Selections } from 'lib/selections' | ||||||
|  | import { commandBarActor } from 'machines/commandBarMachine' | ||||||
|  |  | ||||||
| interface CreateSegmentArgs { | interface CreateSegmentArgs { | ||||||
|   input: SegmentInputs |   input: SegmentInputs | ||||||
| @ -844,7 +848,7 @@ function createLengthIndicator({ | |||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     // Command Bar |     // Command Bar | ||||||
|     editorManager.commandBarSend({ |     commandBarActor.send({ | ||||||
|       type: 'Find and select command', |       type: 'Find and select command', | ||||||
|       data: { |       data: { | ||||||
|         name: 'Constrain length', |         name: 'Constrain length', | ||||||
| @ -1003,6 +1007,49 @@ export function createArcGeometry({ | |||||||
|   return geo |   return geo | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // (lee) The above is much more complex than necessary. | ||||||
|  | // I've derived the new code from: | ||||||
|  | // https://threejs.org/docs/#api/en/extras/curves/EllipseCurve | ||||||
|  | // I'm not sure why it wasn't done like this in the first place? | ||||||
|  | // I don't touch the code above because it may break something else. | ||||||
|  | export function createCircleGeometry({ | ||||||
|  |   center, | ||||||
|  |   radius, | ||||||
|  |   color, | ||||||
|  |   isDashed = false, | ||||||
|  |   scale = 1, | ||||||
|  | }: { | ||||||
|  |   center: Coords2d | ||||||
|  |   radius: number | ||||||
|  |   color: number | ||||||
|  |   isDashed?: boolean | ||||||
|  |   scale?: number | ||||||
|  | }): Line { | ||||||
|  |   const circle = new EllipseCurve( | ||||||
|  |     center[0], | ||||||
|  |     center[1], | ||||||
|  |     radius, | ||||||
|  |     radius, | ||||||
|  |     0, | ||||||
|  |     Math.PI * 2, | ||||||
|  |     true, | ||||||
|  |     scale | ||||||
|  |   ) | ||||||
|  |   const points = circle.getPoints(75) // just enough points to not see edges. | ||||||
|  |   const geometry = new BufferGeometry().setFromPoints(points) | ||||||
|  |   const material = !isDashed | ||||||
|  |     ? new LineBasicMaterial({ color }) | ||||||
|  |     : new LineDashedMaterial({ | ||||||
|  |         color, | ||||||
|  |         scale, | ||||||
|  |         dashSize: 6, | ||||||
|  |         gapSize: 6, | ||||||
|  |       }) | ||||||
|  |   const line = new Line(geometry, material) | ||||||
|  |   line.computeLineDistances() | ||||||
|  |   return line | ||||||
|  | } | ||||||
|  |  | ||||||
| export function dashedStraight( | export function dashedStraight( | ||||||
|   from: Coords2d, |   from: Coords2d, | ||||||
|   to: Coords2d, |   to: Coords2d, | ||||||
|  | |||||||
| @ -1,9 +1,11 @@ | |||||||
| import { Popover } from '@headlessui/react' | import { Popover } from '@headlessui/react' | ||||||
| import { ActionButtonProps } from './ActionButton' | import { ActionButtonProps } from './ActionButton' | ||||||
| import { CustomIcon } from './CustomIcon' | import { CustomIcon } from './CustomIcon' | ||||||
|  | import Tooltip from './Tooltip' | ||||||
|  |  | ||||||
| type ActionButtonSplitProps = ActionButtonProps & { Element: 'button' } & { | type ActionButtonSplitProps = ActionButtonProps & { Element: 'button' } & { | ||||||
|   name?: string |   name?: string | ||||||
|  |   dropdownTooltipText?: string | ||||||
|   splitMenuItems: { |   splitMenuItems: { | ||||||
|     id: string |     id: string | ||||||
|     label: string |     label: string | ||||||
| @ -17,6 +19,7 @@ type ActionButtonSplitProps = ActionButtonProps & { Element: 'button' } & { | |||||||
| export function ActionButtonDropdown({ | export function ActionButtonDropdown({ | ||||||
|   splitMenuItems, |   splitMenuItems, | ||||||
|   className, |   className, | ||||||
|  |   dropdownTooltipText = 'More tools', | ||||||
|   children, |   children, | ||||||
|   ...props |   ...props | ||||||
| }: ActionButtonSplitProps) { | }: ActionButtonSplitProps) { | ||||||
| @ -26,7 +29,14 @@ export function ActionButtonDropdown({ | |||||||
|       {({ close }) => ( |       {({ close }) => ( | ||||||
|         <> |         <> | ||||||
|           {children} |           {children} | ||||||
|           <Popover.Button className="border-transparent dark:border-transparent p-0 m-0 rounded-none !outline-none ui-open:border-primary ui-open:bg-primary"> |           <Popover.Button | ||||||
|  |             className={ | ||||||
|  |               '!border-transparent dark:!border-transparent ' + | ||||||
|  |               'bg-chalkboard-transparent dark:bg-transparent disabled:bg-transparent dark:disabled:bg-transparent ' + | ||||||
|  |               'enabled:hover:bg-chalkboard-10 dark:enabled:hover:bg-chalkboard-100 ' + | ||||||
|  |               'pressed:!bg-primary pressed:enabled:hover:!text-chalkboard-10 p-0 m-0 rounded-none !outline-none ui-open:border-primary ui-open:bg-primary' | ||||||
|  |             } | ||||||
|  |           > | ||||||
|             <CustomIcon |             <CustomIcon | ||||||
|               name="caretDown" |               name="caretDown" | ||||||
|               className={ |               className={ | ||||||
| @ -37,6 +47,14 @@ export function ActionButtonDropdown({ | |||||||
|             <span className="sr-only"> |             <span className="sr-only"> | ||||||
|               {props.name ? props.name + ': ' : ''}open menu |               {props.name ? props.name + ': ' : ''}open menu | ||||||
|             </span> |             </span> | ||||||
|  |             <Tooltip | ||||||
|  |               delay={0} | ||||||
|  |               position="bottom" | ||||||
|  |               hoverOnly | ||||||
|  |               wrapperClassName="ui-open:!hidden" | ||||||
|  |             > | ||||||
|  |               {dropdownTooltipText} | ||||||
|  |             </Tooltip> | ||||||
|           </Popover.Button> |           </Popover.Button> | ||||||
|           <Popover.Panel |           <Popover.Panel | ||||||
|             as="ul" |             as="ul" | ||||||
|  | |||||||
| @ -5,7 +5,7 @@ import { useEffect, useRef, useState } from 'react' | |||||||
| import { trap } from 'lib/trap' | import { trap } from 'lib/trap' | ||||||
| import { codeToIdSelections } from 'lib/selections' | import { codeToIdSelections } from 'lib/selections' | ||||||
| import { codeRefFromRange } from 'lang/std/artifactGraph' | import { codeRefFromRange } from 'lang/std/artifactGraph' | ||||||
| import { defaultSourceRange } from 'lang/wasm' | import { defaultSourceRange, SourceRange, topLevelRange } from 'lang/wasm' | ||||||
|  |  | ||||||
| export function AstExplorer() { | export function AstExplorer() { | ||||||
|   const { context } = useModelingContext() |   const { context } = useModelingContext() | ||||||
| @ -118,19 +118,19 @@ function DisplayObj({ | |||||||
|         hasCursor ? 'bg-violet-100/80 dark:bg-violet-100/25' : '' |         hasCursor ? 'bg-violet-100/80 dark:bg-violet-100/25' : '' | ||||||
|       }`} |       }`} | ||||||
|       onMouseEnter={(e) => { |       onMouseEnter={(e) => { | ||||||
|         editorManager.setHighlightRange([[obj?.start || 0, obj.end, true]]) |         editorManager.setHighlightRange([ | ||||||
|  |           topLevelRange(obj?.start || 0, obj.end), | ||||||
|  |         ]) | ||||||
|         e.stopPropagation() |         e.stopPropagation() | ||||||
|       }} |       }} | ||||||
|       onMouseMove={(e) => { |       onMouseMove={(e) => { | ||||||
|         e.stopPropagation() |         e.stopPropagation() | ||||||
|         editorManager.setHighlightRange([[obj?.start || 0, obj.end, true]]) |         editorManager.setHighlightRange([ | ||||||
|  |           topLevelRange(obj?.start || 0, obj.end), | ||||||
|  |         ]) | ||||||
|       }} |       }} | ||||||
|       onClick={(e) => { |       onClick={(e) => { | ||||||
|         const range: [number, number, boolean] = [ |         const range = topLevelRange(obj?.start || 0, obj.end || 0) | ||||||
|           obj?.start || 0, |  | ||||||
|           obj.end || 0, |  | ||||||
|           true, |  | ||||||
|         ] |  | ||||||
|         const idInfo = codeToIdSelections([ |         const idInfo = codeToIdSelections([ | ||||||
|           { codeRef: codeRefFromRange(range, kclManager.ast) }, |           { codeRef: codeRefFromRange(range, kclManager.ast) }, | ||||||
|         ])[0] |         ])[0] | ||||||
|  | |||||||
| @ -1,8 +1,8 @@ | |||||||
| import { Combobox } from '@headlessui/react' | import { Combobox } from '@headlessui/react' | ||||||
| import { useSelector } from '@xstate/react' | import { useSelector } from '@xstate/react' | ||||||
| import Fuse from 'fuse.js' | import Fuse from 'fuse.js' | ||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' |  | ||||||
| import { CommandArgument, CommandArgumentOption } from 'lib/commandTypes' | import { CommandArgument, CommandArgumentOption } from 'lib/commandTypes' | ||||||
|  | import { commandBarActor, useCommandBarState } from 'machines/commandBarMachine' | ||||||
| import { useEffect, useMemo, useRef, useState } from 'react' | import { useEffect, useMemo, useRef, useState } from 'react' | ||||||
| import { AnyStateMachine, StateFrom } from 'xstate' | import { AnyStateMachine, StateFrom } from 'xstate' | ||||||
|  |  | ||||||
| @ -23,7 +23,7 @@ function CommandArgOptionInput({ | |||||||
|   placeholder?: string |   placeholder?: string | ||||||
| }) { | }) { | ||||||
|   const actorContext = useSelector(arg.machineActor, contextSelector) |   const actorContext = useSelector(arg.machineActor, contextSelector) | ||||||
|   const { commandBarSend, commandBarState } = useCommandsContext() |   const commandBarState = useCommandBarState() | ||||||
|   const resolvedOptions = useMemo( |   const resolvedOptions = useMemo( | ||||||
|     () => |     () => | ||||||
|       typeof arg.options === 'function' |       typeof arg.options === 'function' | ||||||
| @ -134,6 +134,7 @@ function CommandArgOptionInput({ | |||||||
|           </label> |           </label> | ||||||
|           <Combobox.Input |           <Combobox.Input | ||||||
|             id="option-input" |             id="option-input" | ||||||
|  |             data-testid="cmd-bar-arg-value" | ||||||
|             ref={inputRef} |             ref={inputRef} | ||||||
|             onChange={(event) => |             onChange={(event) => | ||||||
|               !event.target.disabled && setQuery(event.target.value) |               !event.target.disabled && setQuery(event.target.value) | ||||||
| @ -141,7 +142,7 @@ function CommandArgOptionInput({ | |||||||
|             className="flex-grow px-2 py-1 border-b border-b-chalkboard-100 dark:border-b-chalkboard-80 !bg-transparent focus:outline-none" |             className="flex-grow px-2 py-1 border-b border-b-chalkboard-100 dark:border-b-chalkboard-80 !bg-transparent focus:outline-none" | ||||||
|             onKeyDown={(event) => { |             onKeyDown={(event) => { | ||||||
|               if (event.metaKey && event.key === 'k') |               if (event.metaKey && event.key === 'k') | ||||||
|                 commandBarSend({ type: 'Close' }) |                 commandBarActor.send({ type: 'Close' }) | ||||||
|               if (event.key === 'Backspace' && !event.currentTarget.value) { |               if (event.key === 'Backspace' && !event.currentTarget.value) { | ||||||
|                 stepBack() |                 stepBack() | ||||||
|               } |               } | ||||||
|  | |||||||
| @ -1,6 +1,5 @@ | |||||||
| import { Dialog, Popover, Transition } from '@headlessui/react' | import { Dialog, Popover, Transition } from '@headlessui/react' | ||||||
| import { Fragment, useEffect } from 'react' | import { Fragment, useEffect } from 'react' | ||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' |  | ||||||
| import CommandBarArgument from './CommandBarArgument' | import CommandBarArgument from './CommandBarArgument' | ||||||
| import CommandComboBox from '../CommandComboBox' | import CommandComboBox from '../CommandComboBox' | ||||||
| import CommandBarReview from './CommandBarReview' | import CommandBarReview from './CommandBarReview' | ||||||
| @ -8,12 +7,13 @@ import { useLocation } from 'react-router-dom' | |||||||
| import useHotkeyWrapper from 'lib/hotkeyWrapper' | import useHotkeyWrapper from 'lib/hotkeyWrapper' | ||||||
| import { CustomIcon } from 'components/CustomIcon' | import { CustomIcon } from 'components/CustomIcon' | ||||||
| import Tooltip from 'components/Tooltip' | import Tooltip from 'components/Tooltip' | ||||||
|  | import { commandBarActor, useCommandBarState } from 'machines/commandBarMachine' | ||||||
|  |  | ||||||
| export const COMMAND_PALETTE_HOTKEY = 'mod+k' | export const COMMAND_PALETTE_HOTKEY = 'mod+k' | ||||||
|  |  | ||||||
| export const CommandBar = () => { | export const CommandBar = () => { | ||||||
|   const { pathname } = useLocation() |   const { pathname } = useLocation() | ||||||
|   const { commandBarState, commandBarSend } = useCommandsContext() |   const commandBarState = useCommandBarState() | ||||||
|   const { |   const { | ||||||
|     context: { selectedCommand, currentArgument, commands }, |     context: { selectedCommand, currentArgument, commands }, | ||||||
|   } = commandBarState |   } = commandBarState | ||||||
| @ -22,16 +22,17 @@ export const CommandBar = () => { | |||||||
|  |  | ||||||
|   // Close the command bar when navigating |   // Close the command bar when navigating | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     commandBarSend({ type: 'Close' }) |     if (commandBarState.matches('Closed')) return | ||||||
|  |     commandBarActor.send({ type: 'Close' }) | ||||||
|   }, [pathname]) |   }, [pathname]) | ||||||
|  |  | ||||||
|   // Hook up keyboard shortcuts |   // Hook up keyboard shortcuts | ||||||
|   useHotkeyWrapper([COMMAND_PALETTE_HOTKEY], () => { |   useHotkeyWrapper([COMMAND_PALETTE_HOTKEY], () => { | ||||||
|     if (commandBarState.context.commands.length === 0) return |     if (commandBarState.context.commands.length === 0) return | ||||||
|     if (commandBarState.matches('Closed')) { |     if (commandBarState.matches('Closed')) { | ||||||
|       commandBarSend({ type: 'Open' }) |       commandBarActor.send({ type: 'Open' }) | ||||||
|     } else { |     } else { | ||||||
|       commandBarSend({ type: 'Close' }) |       commandBarActor.send({ type: 'Close' }) | ||||||
|     } |     } | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
| @ -51,14 +52,14 @@ export const CommandBar = () => { | |||||||
|           ...entries[entries.length - 1][1], |           ...entries[entries.length - 1][1], | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         commandBarSend({ |         commandBarActor.send({ | ||||||
|           type: 'Edit argument', |           type: 'Edit argument', | ||||||
|           data: { |           data: { | ||||||
|             arg: currentArg, |             arg: currentArg, | ||||||
|           }, |           }, | ||||||
|         }) |         }) | ||||||
|       } else { |       } else { | ||||||
|         commandBarSend({ type: 'Deselect command' }) |         commandBarActor.send({ type: 'Deselect command' }) | ||||||
|       } |       } | ||||||
|     } else { |     } else { | ||||||
|       const entries = Object.entries(selectedCommand?.args || {}) |       const entries = Object.entries(selectedCommand?.args || {}) | ||||||
| @ -67,9 +68,9 @@ export const CommandBar = () => { | |||||||
|       ) |       ) | ||||||
|  |  | ||||||
|       if (index === 0) { |       if (index === 0) { | ||||||
|         commandBarSend({ type: 'Deselect command' }) |         commandBarActor.send({ type: 'Deselect command' }) | ||||||
|       } else { |       } else { | ||||||
|         commandBarSend({ |         commandBarActor.send({ | ||||||
|           type: 'Change current argument', |           type: 'Change current argument', | ||||||
|           data: { |           data: { | ||||||
|             arg: { name: entries[index - 1][0], ...entries[index - 1][1] }, |             arg: { name: entries[index - 1][0], ...entries[index - 1][1] }, | ||||||
| @ -84,14 +85,14 @@ export const CommandBar = () => { | |||||||
|       show={!commandBarState.matches('Closed') || false} |       show={!commandBarState.matches('Closed') || false} | ||||||
|       afterLeave={() => { |       afterLeave={() => { | ||||||
|         if (selectedCommand?.onCancel) selectedCommand.onCancel() |         if (selectedCommand?.onCancel) selectedCommand.onCancel() | ||||||
|         commandBarSend({ type: 'Clear' }) |         commandBarActor.send({ type: 'Clear' }) | ||||||
|       }} |       }} | ||||||
|       as={Fragment} |       as={Fragment} | ||||||
|     > |     > | ||||||
|       <WrapperComponent |       <WrapperComponent | ||||||
|         open={!commandBarState.matches('Closed') || isSelectionArgument} |         open={!commandBarState.matches('Closed') || isSelectionArgument} | ||||||
|         onClose={() => { |         onClose={() => { | ||||||
|           commandBarSend({ type: 'Close' }) |           commandBarActor.send({ type: 'Close' }) | ||||||
|         }} |         }} | ||||||
|         className={ |         className={ | ||||||
|           'fixed inset-0 z-50 overflow-y-auto pb-4 pt-1 ' + |           'fixed inset-0 z-50 overflow-y-auto pb-4 pt-1 ' + | ||||||
| @ -121,7 +122,7 @@ export const CommandBar = () => { | |||||||
|               ) |               ) | ||||||
|             )} |             )} | ||||||
|             <button |             <button | ||||||
|               onClick={() => commandBarSend({ type: 'Close' })} |               onClick={() => commandBarActor.send({ type: 'Close' })} | ||||||
|               className="group block !absolute left-auto right-full top-[-3px] m-2.5 p-0 border-none bg-transparent hover:bg-transparent" |               className="group block !absolute left-auto right-full top-[-3px] m-2.5 p-0 border-none bg-transparent hover:bg-transparent" | ||||||
|             > |             > | ||||||
|               <CustomIcon |               <CustomIcon | ||||||
|  | |||||||
| @ -2,13 +2,13 @@ import CommandArgOptionInput from './CommandArgOptionInput' | |||||||
| import CommandBarBasicInput from './CommandBarBasicInput' | import CommandBarBasicInput from './CommandBarBasicInput' | ||||||
| import CommandBarSelectionInput from './CommandBarSelectionInput' | import CommandBarSelectionInput from './CommandBarSelectionInput' | ||||||
| import { CommandArgument } from 'lib/commandTypes' | import { CommandArgument } from 'lib/commandTypes' | ||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' |  | ||||||
| import CommandBarHeader from './CommandBarHeader' | import CommandBarHeader from './CommandBarHeader' | ||||||
| import CommandBarKclInput from './CommandBarKclInput' | import CommandBarKclInput from './CommandBarKclInput' | ||||||
| import CommandBarTextareaInput from './CommandBarTextareaInput' | import CommandBarTextareaInput from './CommandBarTextareaInput' | ||||||
|  | import { commandBarActor, useCommandBarState } from 'machines/commandBarMachine' | ||||||
|  |  | ||||||
| function CommandBarArgument({ stepBack }: { stepBack: () => void }) { | function CommandBarArgument({ stepBack }: { stepBack: () => void }) { | ||||||
|   const { commandBarState, commandBarSend } = useCommandsContext() |   const commandBarState = useCommandBarState() | ||||||
|   const { |   const { | ||||||
|     context: { currentArgument }, |     context: { currentArgument }, | ||||||
|   } = commandBarState |   } = commandBarState | ||||||
| @ -16,7 +16,7 @@ function CommandBarArgument({ stepBack }: { stepBack: () => void }) { | |||||||
|   function onSubmit(data: unknown) { |   function onSubmit(data: unknown) { | ||||||
|     if (!currentArgument) return |     if (!currentArgument) return | ||||||
|  |  | ||||||
|     commandBarSend({ |     commandBarActor.send({ | ||||||
|       type: 'Submit argument', |       type: 'Submit argument', | ||||||
|       data: { |       data: { | ||||||
|         [currentArgument.name]: data, |         [currentArgument.name]: data, | ||||||
|  | |||||||
| @ -1,5 +1,5 @@ | |||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' |  | ||||||
| import { CommandArgument } from 'lib/commandTypes' | import { CommandArgument } from 'lib/commandTypes' | ||||||
|  | import { commandBarActor, useCommandBarState } from 'machines/commandBarMachine' | ||||||
| import { useEffect, useRef } from 'react' | import { useEffect, useRef } from 'react' | ||||||
| import { useHotkeys } from 'react-hotkeys-hook' | import { useHotkeys } from 'react-hotkeys-hook' | ||||||
|  |  | ||||||
| @ -15,8 +15,8 @@ function CommandBarBasicInput({ | |||||||
|   stepBack: () => void |   stepBack: () => void | ||||||
|   onSubmit: (event: unknown) => void |   onSubmit: (event: unknown) => void | ||||||
| }) { | }) { | ||||||
|   const { commandBarSend, commandBarState } = useCommandsContext() |   const commandBarState = useCommandBarState() | ||||||
|   useHotkeys('mod + k, mod + /', () => commandBarSend({ type: 'Close' })) |   useHotkeys('mod + k, mod + /', () => commandBarActor.send({ type: 'Close' })) | ||||||
|   const inputRef = useRef<HTMLInputElement>(null) |   const inputRef = useRef<HTMLInputElement>(null) | ||||||
|  |  | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|  | |||||||
| @ -1,4 +1,3 @@ | |||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' |  | ||||||
| import { CustomIcon } from '../CustomIcon' | import { CustomIcon } from '../CustomIcon' | ||||||
| import React, { useState } from 'react' | import React, { useState } from 'react' | ||||||
| import { ActionButton } from '../ActionButton' | import { ActionButton } from '../ActionButton' | ||||||
| @ -7,9 +6,10 @@ import { useHotkeys } from 'react-hotkeys-hook' | |||||||
| import { KclCommandValue, KclExpressionWithVariable } from 'lib/commandTypes' | import { KclCommandValue, KclExpressionWithVariable } from 'lib/commandTypes' | ||||||
| import Tooltip from 'components/Tooltip' | import Tooltip from 'components/Tooltip' | ||||||
| import { roundOff } from 'lib/utils' | import { roundOff } from 'lib/utils' | ||||||
|  | import { commandBarActor, useCommandBarState } from 'machines/commandBarMachine' | ||||||
|  |  | ||||||
| function CommandBarHeader({ children }: React.PropsWithChildren<{}>) { | function CommandBarHeader({ children }: React.PropsWithChildren<{}>) { | ||||||
|   const { commandBarState, commandBarSend } = useCommandsContext() |   const commandBarState = useCommandBarState() | ||||||
|   const { |   const { | ||||||
|     context: { selectedCommand, currentArgument, argumentsToSubmit }, |     context: { selectedCommand, currentArgument, argumentsToSubmit }, | ||||||
|   } = commandBarState |   } = commandBarState | ||||||
| @ -49,7 +49,7 @@ function CommandBarHeader({ children }: React.PropsWithChildren<{}>) { | |||||||
|         ] |         ] | ||||||
|         const arg = selectedCommand?.args[argName] |         const arg = selectedCommand?.args[argName] | ||||||
|         if (!argName || !arg) return |         if (!argName || !arg) return | ||||||
|         commandBarSend({ |         commandBarActor.send({ | ||||||
|           type: 'Change current argument', |           type: 'Change current argument', | ||||||
|           data: { arg: { ...arg, name: argName } }, |           data: { arg: { ...arg, name: argName } }, | ||||||
|         }) |         }) | ||||||
| @ -100,7 +100,7 @@ function CommandBarHeader({ children }: React.PropsWithChildren<{}>) { | |||||||
|                     } |                     } | ||||||
|                     disabled={!isReviewing && currentArgument?.name === argName} |                     disabled={!isReviewing && currentArgument?.name === argName} | ||||||
|                     onClick={() => { |                     onClick={() => { | ||||||
|                       commandBarSend({ |                       commandBarActor.send({ | ||||||
|                         type: isReviewing |                         type: isReviewing | ||||||
|                           ? 'Edit argument' |                           ? 'Edit argument' | ||||||
|                           : 'Change current argument', |                           : 'Change current argument', | ||||||
|  | |||||||
| @ -7,7 +7,6 @@ import { | |||||||
| } from '@codemirror/autocomplete' | } from '@codemirror/autocomplete' | ||||||
| import { EditorView, keymap, ViewUpdate } from '@codemirror/view' | import { EditorView, keymap, ViewUpdate } from '@codemirror/view' | ||||||
| import { CustomIcon } from 'components/CustomIcon' | import { CustomIcon } from 'components/CustomIcon' | ||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' |  | ||||||
| import { useSettingsAuthContext } from 'hooks/useSettingsAuthContext' | import { useSettingsAuthContext } from 'hooks/useSettingsAuthContext' | ||||||
| import { CommandArgument, KclCommandValue } from 'lib/commandTypes' | import { CommandArgument, KclCommandValue } from 'lib/commandTypes' | ||||||
| import { getSystemTheme } from 'lib/theme' | import { getSystemTheme } from 'lib/theme' | ||||||
| @ -20,6 +19,7 @@ import styles from './CommandBarKclInput.module.css' | |||||||
| import { createIdentifier, createVariableDeclaration } from 'lang/modifyAst' | import { createIdentifier, createVariableDeclaration } from 'lang/modifyAst' | ||||||
| import { useCodeMirror } from 'components/ModelingSidebar/ModelingPanes/CodeEditor' | import { useCodeMirror } from 'components/ModelingSidebar/ModelingPanes/CodeEditor' | ||||||
| import { useSelector } from '@xstate/react' | import { useSelector } from '@xstate/react' | ||||||
|  | import { commandBarActor, useCommandBarState } from 'machines/commandBarMachine' | ||||||
|  |  | ||||||
| const machineContextSelector = (snapshot?: { | const machineContextSelector = (snapshot?: { | ||||||
|   context: Record<string, unknown> |   context: Record<string, unknown> | ||||||
| @ -37,7 +37,7 @@ function CommandBarKclInput({ | |||||||
|   stepBack: () => void |   stepBack: () => void | ||||||
|   onSubmit: (event: unknown) => void |   onSubmit: (event: unknown) => void | ||||||
| }) { | }) { | ||||||
|   const { commandBarSend, commandBarState } = useCommandsContext() |   const commandBarState = useCommandBarState() | ||||||
|   const previouslySetValue = commandBarState.context.argumentsToSubmit[ |   const previouslySetValue = commandBarState.context.argumentsToSubmit[ | ||||||
|     arg.name |     arg.name | ||||||
|   ] as KclCommandValue | undefined |   ] as KclCommandValue | undefined | ||||||
| @ -82,7 +82,7 @@ function CommandBarKclInput({ | |||||||
|       false |       false | ||||||
|   ) |   ) | ||||||
|   const [canSubmit, setCanSubmit] = useState(true) |   const [canSubmit, setCanSubmit] = useState(true) | ||||||
|   useHotkeys('mod + k, mod + /', () => commandBarSend({ type: 'Close' })) |   useHotkeys('mod + k, mod + /', () => commandBarActor.send({ type: 'Close' })) | ||||||
|   const editorRef = useRef<HTMLDivElement>(null) |   const editorRef = useRef<HTMLDivElement>(null) | ||||||
|  |  | ||||||
|   const { |   const { | ||||||
|  | |||||||
| @ -1,43 +0,0 @@ | |||||||
| import { createActorContext } from '@xstate/react' |  | ||||||
| import { editorManager } from 'lib/singletons' |  | ||||||
| import { commandBarMachine } from 'machines/commandBarMachine' |  | ||||||
| import { useEffect } from 'react' |  | ||||||
|  |  | ||||||
| export const CommandsContext = createActorContext( |  | ||||||
|   commandBarMachine.provide({ |  | ||||||
|     guards: { |  | ||||||
|       'Command has no arguments': ({ context }) => { |  | ||||||
|         return ( |  | ||||||
|           !context.selectedCommand?.args || |  | ||||||
|           Object.keys(context.selectedCommand?.args).length === 0 |  | ||||||
|         ) |  | ||||||
|       }, |  | ||||||
|       'All arguments are skippable': ({ context }) => { |  | ||||||
|         return Object.values(context.selectedCommand!.args!).every( |  | ||||||
|           (argConfig) => argConfig.skip |  | ||||||
|         ) |  | ||||||
|       }, |  | ||||||
|     }, |  | ||||||
|   }) |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| export const CommandBarProvider = ({ |  | ||||||
|   children, |  | ||||||
| }: { |  | ||||||
|   children: React.ReactNode |  | ||||||
| }) => { |  | ||||||
|   return ( |  | ||||||
|     <CommandsContext.Provider> |  | ||||||
|       <CommandBarProviderInner>{children}</CommandBarProviderInner> |  | ||||||
|     </CommandsContext.Provider> |  | ||||||
|   ) |  | ||||||
| } |  | ||||||
| function CommandBarProviderInner({ children }: { children: React.ReactNode }) { |  | ||||||
|   const commandBarActor = CommandsContext.useActorRef() |  | ||||||
|  |  | ||||||
|   useEffect(() => { |  | ||||||
|     editorManager.setCommandBarSend(commandBarActor.send) |  | ||||||
|   }) |  | ||||||
|  |  | ||||||
|   return children |  | ||||||
| } |  | ||||||
| @ -1,9 +1,9 @@ | |||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' | import { commandBarActor, useCommandBarState } from 'machines/commandBarMachine' | ||||||
| import CommandBarHeader from './CommandBarHeader' | import CommandBarHeader from './CommandBarHeader' | ||||||
| import { useHotkeys } from 'react-hotkeys-hook' | import { useHotkeys } from 'react-hotkeys-hook' | ||||||
|  |  | ||||||
| function CommandBarReview({ stepBack }: { stepBack: () => void }) { | function CommandBarReview({ stepBack }: { stepBack: () => void }) { | ||||||
|   const { commandBarState, commandBarSend } = useCommandsContext() |   const commandBarState = useCommandBarState() | ||||||
|   const { |   const { | ||||||
|     context: { argumentsToSubmit, selectedCommand }, |     context: { argumentsToSubmit, selectedCommand }, | ||||||
|   } = commandBarState |   } = commandBarState | ||||||
| @ -33,7 +33,7 @@ function CommandBarReview({ stepBack }: { stepBack: () => void }) { | |||||||
|           parseInt(b.keys[0], 10) - 1 |           parseInt(b.keys[0], 10) - 1 | ||||||
|         ] |         ] | ||||||
|         const arg = selectedCommand?.args[argName] |         const arg = selectedCommand?.args[argName] | ||||||
|         commandBarSend({ |         commandBarActor.send({ | ||||||
|           type: 'Edit argument', |           type: 'Edit argument', | ||||||
|           data: { arg: { ...arg, name: argName } }, |           data: { arg: { ...arg, name: argName } }, | ||||||
|         }) |         }) | ||||||
| @ -50,7 +50,7 @@ function CommandBarReview({ stepBack }: { stepBack: () => void }) { | |||||||
|  |  | ||||||
|   function submitCommand(e: React.FormEvent<HTMLFormElement>) { |   function submitCommand(e: React.FormEvent<HTMLFormElement>) { | ||||||
|     e.preventDefault() |     e.preventDefault() | ||||||
|     commandBarSend({ |     commandBarActor.send({ | ||||||
|       type: 'Submit command', |       type: 'Submit command', | ||||||
|       output: argumentsToSubmit, |       output: argumentsToSubmit, | ||||||
|     }) |     }) | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| import { useSelector } from '@xstate/react' | import { useSelector } from '@xstate/react' | ||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' |  | ||||||
| import { Artifact } from 'lang/std/artifactGraph' | import { Artifact } from 'lang/std/artifactGraph' | ||||||
| import { CommandArgument } from 'lib/commandTypes' | import { CommandArgument } from 'lib/commandTypes' | ||||||
| import { | import { | ||||||
| @ -10,6 +9,7 @@ import { | |||||||
| import { kclManager } from 'lib/singletons' | import { kclManager } from 'lib/singletons' | ||||||
| import { reportRejection } from 'lib/trap' | import { reportRejection } from 'lib/trap' | ||||||
| import { toSync } from 'lib/utils' | import { toSync } from 'lib/utils' | ||||||
|  | import { commandBarActor, useCommandBarState } from 'machines/commandBarMachine' | ||||||
| import { modelingMachine } from 'machines/modelingMachine' | import { modelingMachine } from 'machines/modelingMachine' | ||||||
| import { useEffect, useMemo, useRef, useState } from 'react' | import { useEffect, useMemo, useRef, useState } from 'react' | ||||||
| import { StateFrom } from 'xstate' | import { StateFrom } from 'xstate' | ||||||
| @ -17,7 +17,7 @@ import { StateFrom } from 'xstate' | |||||||
| const semanticEntityNames: { | const semanticEntityNames: { | ||||||
|   [key: string]: Array<Artifact['type'] | 'defaultPlane'> |   [key: string]: Array<Artifact['type'] | 'defaultPlane'> | ||||||
| } = { | } = { | ||||||
|   face: ['wall', 'cap', 'solid2D'], |   face: ['wall', 'cap', 'solid2d'], | ||||||
|   edge: ['segment', 'sweepEdge', 'edgeCutEdge'], |   edge: ['segment', 'sweepEdge', 'edgeCutEdge'], | ||||||
|   point: [], |   point: [], | ||||||
|   plane: ['defaultPlane'], |   plane: ['defaultPlane'], | ||||||
| @ -49,7 +49,7 @@ function CommandBarSelectionInput({ | |||||||
|   onSubmit: (data: unknown) => void |   onSubmit: (data: unknown) => void | ||||||
| }) { | }) { | ||||||
|   const inputRef = useRef<HTMLInputElement>(null) |   const inputRef = useRef<HTMLInputElement>(null) | ||||||
|   const { commandBarState, commandBarSend } = useCommandsContext() |   const commandBarState = useCommandBarState() | ||||||
|   const [hasSubmitted, setHasSubmitted] = useState(false) |   const [hasSubmitted, setHasSubmitted] = useState(false) | ||||||
|   const selection = useSelector(arg.machineActor, selectionSelector) |   const selection = useSelector(arg.machineActor, selectionSelector) | ||||||
|   const selectionsByType = useMemo(() => { |   const selectionsByType = useMemo(() => { | ||||||
| @ -145,7 +145,7 @@ function CommandBarSelectionInput({ | |||||||
|             if (event.key === 'Backspace') { |             if (event.key === 'Backspace') { | ||||||
|               stepBack() |               stepBack() | ||||||
|             } else if (event.key === 'Escape') { |             } else if (event.key === 'Escape') { | ||||||
|               commandBarSend({ type: 'Close' }) |               commandBarActor.send({ type: 'Close' }) | ||||||
|             } |             } | ||||||
|           }} |           }} | ||||||
|           onChange={handleChange} |           onChange={handleChange} | ||||||
|  | |||||||
| @ -1,5 +1,5 @@ | |||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' |  | ||||||
| import { CommandArgument } from 'lib/commandTypes' | import { CommandArgument } from 'lib/commandTypes' | ||||||
|  | import { commandBarActor, useCommandBarState } from 'machines/commandBarMachine' | ||||||
| import { RefObject, useEffect, useRef } from 'react' | import { RefObject, useEffect, useRef } from 'react' | ||||||
| import { useHotkeys } from 'react-hotkeys-hook' | import { useHotkeys } from 'react-hotkeys-hook' | ||||||
|  |  | ||||||
| @ -15,8 +15,8 @@ function CommandBarTextareaInput({ | |||||||
|   stepBack: () => void |   stepBack: () => void | ||||||
|   onSubmit: (event: unknown) => void |   onSubmit: (event: unknown) => void | ||||||
| }) { | }) { | ||||||
|   const { commandBarSend, commandBarState } = useCommandsContext() |   const commandBarState = useCommandBarState() | ||||||
|   useHotkeys('mod + k, mod + /', () => commandBarSend({ type: 'Close' })) |   useHotkeys('mod + k, mod + /', () => commandBarActor.send({ type: 'Close' })) | ||||||
|   const formRef = useRef<HTMLFormElement>(null) |   const formRef = useRef<HTMLFormElement>(null) | ||||||
|   const inputRef = useRef<HTMLTextAreaElement>(null) |   const inputRef = useRef<HTMLTextAreaElement>(null) | ||||||
|   useTextareaAutoGrow(inputRef) |   useTextareaAutoGrow(inputRef) | ||||||
|  | |||||||
| @ -1,16 +1,15 @@ | |||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' |  | ||||||
| import usePlatform from 'hooks/usePlatform' | import usePlatform from 'hooks/usePlatform' | ||||||
| import { hotkeyDisplay } from 'lib/hotkeyWrapper' | import { hotkeyDisplay } from 'lib/hotkeyWrapper' | ||||||
| import { COMMAND_PALETTE_HOTKEY } from './CommandBar/CommandBar' | import { COMMAND_PALETTE_HOTKEY } from './CommandBar/CommandBar' | ||||||
|  | import { commandBarActor } from 'machines/commandBarMachine' | ||||||
|  |  | ||||||
| export function CommandBarOpenButton() { | export function CommandBarOpenButton() { | ||||||
|   const { commandBarSend } = useCommandsContext() |  | ||||||
|   const platform = usePlatform() |   const platform = usePlatform() | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <button |     <button | ||||||
|       className="group rounded-full flex items-center justify-center gap-2 px-2 py-1 bg-primary/10 dark:bg-chalkboard-90 dark:backdrop-blur-sm border-primary hover:border-primary dark:border-chalkboard-50 dark:hover:border-inherit text-primary dark:text-inherit" |       className="group rounded-full flex items-center justify-center gap-2 px-2 py-1 bg-primary/10 dark:bg-chalkboard-90 dark:backdrop-blur-sm border-primary hover:border-primary dark:border-chalkboard-50 dark:hover:border-inherit text-primary dark:text-inherit" | ||||||
|       onClick={() => commandBarSend({ type: 'Open' })} |       onClick={() => commandBarActor.send({ type: 'Open' })} | ||||||
|       data-testid="command-bar-open-button" |       data-testid="command-bar-open-button" | ||||||
|     > |     > | ||||||
|       <span>Commands</span> |       <span>Commands</span> | ||||||
|  | |||||||
| @ -1,9 +1,11 @@ | |||||||
| import { Combobox } from '@headlessui/react' | import { Combobox } from '@headlessui/react' | ||||||
| import Fuse from 'fuse.js' | import Fuse from 'fuse.js' | ||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' |  | ||||||
| import { Command } from 'lib/commandTypes' | import { Command } from 'lib/commandTypes' | ||||||
| import { useEffect, useState } from 'react' | import { useEffect, useState } from 'react' | ||||||
| import { CustomIcon } from './CustomIcon' | import { CustomIcon } from './CustomIcon' | ||||||
|  | import { getActorNextEvents } from 'lib/utils' | ||||||
|  | import { sortCommands } from 'lib/commandUtils' | ||||||
|  | import { commandBarActor } from 'machines/commandBarMachine' | ||||||
|  |  | ||||||
| function CommandComboBox({ | function CommandComboBox({ | ||||||
|   options, |   options, | ||||||
| @ -12,14 +14,21 @@ function CommandComboBox({ | |||||||
|   options: Command[] |   options: Command[] | ||||||
|   placeholder?: string |   placeholder?: string | ||||||
| }) { | }) { | ||||||
|   const { commandBarSend } = useCommandsContext() |  | ||||||
|   const [query, setQuery] = useState('') |   const [query, setQuery] = useState('') | ||||||
|   const [filteredOptions, setFilteredOptions] = useState<typeof options>() |   const [filteredOptions, setFilteredOptions] = useState<typeof options>() | ||||||
|  |  | ||||||
|   const defaultOption = |   const defaultOption = | ||||||
|     options.find((o) => 'isCurrent' in o && o.isCurrent) || null |     options.find((o) => 'isCurrent' in o && o.isCurrent) || null | ||||||
|  |   // sort disabled commands to the bottom | ||||||
|  |   const sortedOptions = options | ||||||
|  |     .map((command) => ({ | ||||||
|  |       command, | ||||||
|  |       disabled: optionIsDisabled(command), | ||||||
|  |     })) | ||||||
|  |     .sort(sortCommands) | ||||||
|  |     .map(({ command }) => command) | ||||||
|  |  | ||||||
|   const fuse = new Fuse(options, { |   const fuse = new Fuse(sortedOptions, { | ||||||
|     keys: ['displayName', 'name', 'description'], |     keys: ['displayName', 'name', 'description'], | ||||||
|     threshold: 0.3, |     threshold: 0.3, | ||||||
|     ignoreLocation: true, |     ignoreLocation: true, | ||||||
| @ -27,11 +36,11 @@ function CommandComboBox({ | |||||||
|  |  | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     const results = fuse.search(query).map((result) => result.item) |     const results = fuse.search(query).map((result) => result.item) | ||||||
|     setFilteredOptions(query.length > 0 ? results : options) |     setFilteredOptions(query.length > 0 ? results : sortedOptions) | ||||||
|   }, [query]) |   }, [query]) | ||||||
|  |  | ||||||
|   function handleSelection(command: Command) { |   function handleSelection(command: Command) { | ||||||
|     commandBarSend({ type: 'Select command', data: { command } }) |     commandBarActor.send({ type: 'Select command', data: { command } }) | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
| @ -42,6 +51,7 @@ function CommandComboBox({ | |||||||
|           className="w-5 h-5 bg-primary/10 dark:bg-primary text-primary dark:text-inherit" |           className="w-5 h-5 bg-primary/10 dark:bg-primary text-primary dark:text-inherit" | ||||||
|         /> |         /> | ||||||
|         <Combobox.Input |         <Combobox.Input | ||||||
|  |           data-testid="cmd-bar-search" | ||||||
|           onChange={(event) => setQuery(event.target.value)} |           onChange={(event) => setQuery(event.target.value)} | ||||||
|           className="w-full bg-transparent focus:outline-none selection:bg-primary/20 dark:selection:bg-primary/40 dark:focus:outline-none" |           className="w-full bg-transparent focus:outline-none selection:bg-primary/20 dark:selection:bg-primary/40 dark:focus:outline-none" | ||||||
|           onKeyDown={(event) => { |           onKeyDown={(event) => { | ||||||
| @ -50,7 +60,7 @@ function CommandComboBox({ | |||||||
|               (event.key === 'Backspace' && !event.currentTarget.value) |               (event.key === 'Backspace' && !event.currentTarget.value) | ||||||
|             ) { |             ) { | ||||||
|               event.preventDefault() |               event.preventDefault() | ||||||
|               commandBarSend({ type: 'Close' }) |               commandBarActor.send({ type: 'Close' }) | ||||||
|             } |             } | ||||||
|           }} |           }} | ||||||
|           placeholder={ |           placeholder={ | ||||||
| @ -73,7 +83,9 @@ function CommandComboBox({ | |||||||
|           <Combobox.Option |           <Combobox.Option | ||||||
|             key={option.groupId + option.name + (option.displayName || '')} |             key={option.groupId + option.name + (option.displayName || '')} | ||||||
|             value={option} |             value={option} | ||||||
|             className="flex items-center gap-4 px-4 py-1.5 first:mt-2 last:mb-2 ui-active:bg-primary/10 dark:ui-active:bg-chalkboard-90" |             className="flex items-center gap-4 px-4 py-1.5 first:mt-2 last:mb-2 ui-active:bg-primary/10 dark:ui-active:bg-chalkboard-90 ui-disabled:!text-chalkboard-50" | ||||||
|  |             disabled={optionIsDisabled(option)} | ||||||
|  |             data-testid={`cmd-bar-option`} | ||||||
|           > |           > | ||||||
|             {'icon' in option && option.icon && ( |             {'icon' in option && option.icon && ( | ||||||
|               <CustomIcon name={option.icon} className="w-5 h-5" /> |               <CustomIcon name={option.icon} className="w-5 h-5" /> | ||||||
| @ -96,3 +108,11 @@ function CommandComboBox({ | |||||||
| } | } | ||||||
|  |  | ||||||
| export default CommandComboBox | export default CommandComboBox | ||||||
|  |  | ||||||
|  | function optionIsDisabled(option: Command): boolean { | ||||||
|  |   return ( | ||||||
|  |     'machineActor' in option && | ||||||
|  |     option.machineActor !== undefined && | ||||||
|  |     !getActorNextEvents(option.machineActor.getSnapshot()).includes(option.name) | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | |||||||
| @ -538,6 +538,16 @@ const CustomIconMap = { | |||||||
|       /> |       /> | ||||||
|     </svg> |     </svg> | ||||||
|   ), |   ), | ||||||
|  |   helix: ( | ||||||
|  |     <svg viewBox="0 0 20 20" fill="none" xmlns="http://www.w3.org/2000/svg"> | ||||||
|  |       <path | ||||||
|  |         fillRule="evenodd" | ||||||
|  |         clipRule="evenodd" | ||||||
|  |         d="M12.3796 6.35525C10.6758 5.64945 8.44129 5.27796 6.92519 5.64172C6.15726 5.82597 5.7318 6.05228 5.55779 6.21295C5.76304 6.32354 6.2288 6.43945 7.03653 6.43302C7.87009 6.42638 8.9975 6.29045 10.4229 5.9501L10.6551 6.92275C9.17724 7.27564 7.9725 7.42559 7.04449 7.43298C6.14216 7.44017 5.42343 7.31395 4.98579 7.03617C4.75792 6.89153 4.53857 6.65945 4.50435 6.32695C4.47054 5.99852 4.63374 5.72683 4.81912 5.53684C5.17998 5.16702 5.83926 4.87389 6.69188 4.66932C8.48928 4.23806 10.9508 4.68095 12.7623 5.43139C13.669 5.80697 14.4784 6.28567 14.9739 6.82869C15.2234 7.10197 15.4238 7.42493 15.4827 7.78937C15.5448 8.1741 15.4392 8.54567 15.1831 8.86785C14.9896 9.11133 14.6502 9.31092 14.327 9.47089C14.1575 9.55477 13.9707 9.63785 13.7736 9.71907C14.257 9.99254 14.6732 10.2984 14.9739 10.6279C15.2234 10.9011 15.4238 11.2241 15.4827 11.5885C15.5448 11.9733 15.4392 12.3448 15.1831 12.667C14.9896 12.9105 14.6502 13.1101 14.327 13.2701C14.1575 13.3539 13.9707 13.437 13.7735 13.5182C14.3755 13.8587 14.8991 14.2636 15.2067 14.7211L14.3767 15.2789C14.1912 15.0029 13.8109 14.6842 13.2483 14.3702C13.0112 14.2378 12.7496 14.1107 12.4694 13.9913C11.8027 14.2087 11.1417 14.3953 10.6642 14.5188L10.6552 14.5212L10.6551 14.5211C9.17724 14.874 7.9725 15.0239 7.04449 15.0313C6.14216 15.0385 5.42343 14.9123 4.98579 14.6345C4.75792 14.4899 4.53857 14.2578 4.50435 13.9253C4.47054 13.5969 4.63374 13.3252 4.81912 13.1352C5.17998 12.7653 5.83926 12.4722 6.69188 12.2677C8.12302 11.9243 9.96538 12.1368 11.5511 12.6039C11.5872 12.6145 11.6233 12.6253 11.6593 12.6363L10.0638 13.2745C8.93153 13.0645 7.80454 13.0291 6.92519 13.2401C6.15727 13.4243 5.73181 13.6506 5.5578 13.8113C5.76305 13.9219 6.2288 14.0378 7.03653 14.0313C7.8692 14.0247 8.99509 13.8891 10.4183 13.5495C10.5419 13.5175 10.678 13.4812 10.8233 13.4412C10.8184 13.4399 10.8134 13.4387 10.8085 13.4374L12.6 12.9L12.5922 12.8948C12.6584 12.8718 12.7243 12.8485 12.7894 12.825C13.2047 12.6754 13.5845 12.5217 13.8834 12.3738C14.2059 12.2142 14.359 12.0967 14.4003 12.0448C14.4964 11.9239 14.509 11.832 14.4955 11.748C14.4786 11.6437 14.4094 11.4927 14.2353 11.302C13.8963 10.9305 13.2766 10.536 12.4694 10.1921C11.8027 10.4096 11.1417 10.5962 10.6642 10.7197L10.6552 10.722L10.6551 10.7219C9.17724 11.0748 7.9725 11.2248 7.04449 11.2322C6.14216 11.2393 5.42343 11.1131 4.98579 10.8353C4.75792 10.6907 4.53857 10.4586 4.50435 10.1261C4.47054 9.79768 4.63374 9.526 4.81912 9.33601C5.17998 8.96618 5.83926 8.67306 6.69188 8.46848C8.14467 8.11991 10.0313 8.34242 11.6579 8.83682L10.0624 9.47503C8.9375 9.26666 7.80922 9.22878 6.92519 9.44089C6.15726 9.62514 5.7318 9.85144 5.55779 10.0121C5.76304 10.1227 6.2288 10.2386 7.03653 10.2322C7.86921 10.2255 8.9951 10.0899 10.4183 9.75035C10.542 9.71834 10.6781 9.68201 10.8235 9.64197L10.8072 9.63784L12.6 9.1L12.593 9.09536C12.659 9.0724 12.7245 9.04921 12.7894 9.02583C13.2047 8.87627 13.5845 8.72256 13.8834 8.57464C14.2059 8.41505 14.359 8.29757 14.4003 8.24564C14.4964 8.1247 14.509 8.0328 14.4955 7.94882C14.4786 7.84455 14.4094 7.69357 14.2353 7.50279C13.8839 7.11769 13.2307 6.7078 12.3796 6.35525ZM5.47539 9.95537C5.47546 9.95536 5.47623 9.95615 5.47745 9.95779C5.47592 9.9562 5.47531 9.95538 5.47539 9.95537ZM5.49369 10.0846C5.49289 10.0866 5.49232 10.0876 5.49223 10.0876C5.49215 10.0877 5.49255 10.0866 5.49369 10.0846ZM5.47539 13.7545C5.47546 13.7545 5.47623 13.7553 5.47745 13.757C5.47592 13.7554 5.47531 13.7546 5.47539 13.7545ZM5.49369 13.8838C5.49289 13.8858 5.49232 13.8868 5.49223 13.8868C5.49215 13.8868 5.49255 13.8858 5.49369 13.8838ZM5.47539 6.1562C5.47546 6.15619 5.47623 6.15698 5.47745 6.15862C5.47592 6.15704 5.47531 6.15622 5.47539 6.1562ZM5.49369 6.28544C5.49289 6.28746 5.49232 6.28848 5.49223 6.28848C5.49215 6.28849 5.49255 6.28748 5.49369 6.28544Z" | ||||||
|  |         fill="currentColor" | ||||||
|  |       /> | ||||||
|  |     </svg> | ||||||
|  |   ), | ||||||
|   hole: ( |   hole: ( | ||||||
|     <svg |     <svg | ||||||
|       viewBox="0 0 20 20" |       viewBox="0 0 20 20" | ||||||
|  | |||||||
| @ -1,24 +1,21 @@ | |||||||
| import { useMemo } from 'react' | import { useMemo } from 'react' | ||||||
| import { engineCommandManager } from 'lib/singletons' | import { engineCommandManager } from 'lib/singletons' | ||||||
| import { | import { expandPlane, PlaneArtifactRich } from 'lang/std/artifactGraph' | ||||||
|   ArtifactGraph, | import { ArtifactGraph } from 'lang/wasm' | ||||||
|   expandPlane, |  | ||||||
|   PlaneArtifactRich, |  | ||||||
| } from 'lang/std/artifactGraph' |  | ||||||
| import { DebugDisplayArray, GenericObj } from './DebugDisplayObj' | import { DebugDisplayArray, GenericObj } from './DebugDisplayObj' | ||||||
| 
 | 
 | ||||||
| export function DebugFeatureTree() { | export function DebugArtifactGraph() { | ||||||
|   const featureTree = useMemo(() => { |   const artifactGraphTree = useMemo(() => { | ||||||
|     return computeTree(engineCommandManager.artifactGraph) |     return computeTree(engineCommandManager.artifactGraph) | ||||||
|   }, [engineCommandManager.artifactGraph]) |   }, [engineCommandManager.artifactGraph]) | ||||||
| 
 | 
 | ||||||
|   const filterKeys: string[] = ['__meta', 'codeRef', 'pathToNode'] |   const filterKeys: string[] = ['__meta', 'codeRef', 'pathToNode'] | ||||||
|   return ( |   return ( | ||||||
|     <details data-testid="debug-feature-tree" className="relative"> |     <details data-testid="debug-feature-tree" className="relative"> | ||||||
|       <summary>Feature Tree</summary> |       <summary>Artifact Graph</summary> | ||||||
|       {featureTree.length > 0 ? ( |       {artifactGraphTree.length > 0 ? ( | ||||||
|         <pre className="text-xs"> |         <pre className="text-xs"> | ||||||
|           <DebugDisplayArray arr={featureTree} filterKeys={filterKeys} /> |           <DebugDisplayArray arr={artifactGraphTree} filterKeys={filterKeys} /> | ||||||
|         </pre> |         </pre> | ||||||
|       ) : ( |       ) : ( | ||||||
|         <p>(Empty)</p> |         <p>(Empty)</p> | ||||||
| @ -12,7 +12,6 @@ import { | |||||||
|   StateFrom, |   StateFrom, | ||||||
|   fromPromise, |   fromPromise, | ||||||
| } from 'xstate' | } from 'xstate' | ||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' |  | ||||||
| import { fileMachine } from 'machines/fileMachine' | import { fileMachine } from 'machines/fileMachine' | ||||||
| import { isDesktop } from 'lib/isDesktop' | import { isDesktop } from 'lib/isDesktop' | ||||||
| import { | import { | ||||||
| @ -30,6 +29,7 @@ import { | |||||||
| } from 'lib/getKclSamplesManifest' | } from 'lib/getKclSamplesManifest' | ||||||
| import { useSettingsAuthContext } from 'hooks/useSettingsAuthContext' | import { useSettingsAuthContext } from 'hooks/useSettingsAuthContext' | ||||||
| import { markOnce } from 'lib/performance' | import { markOnce } from 'lib/performance' | ||||||
|  | import { commandBarActor } from 'machines/commandBarMachine' | ||||||
|  |  | ||||||
| type MachineContext<T extends AnyStateMachine> = { | type MachineContext<T extends AnyStateMachine> = { | ||||||
|   state: StateFrom<T> |   state: StateFrom<T> | ||||||
| @ -47,7 +47,6 @@ export const FileMachineProvider = ({ | |||||||
|   children: React.ReactNode |   children: React.ReactNode | ||||||
| }) => { | }) => { | ||||||
|   const navigate = useNavigate() |   const navigate = useNavigate() | ||||||
|   const { commandBarSend } = useCommandsContext() |  | ||||||
|   const { settings } = useSettingsAuthContext() |   const { settings } = useSettingsAuthContext() | ||||||
|   const { project, file } = useRouteLoaderData(PATHS.FILE) as IndexLoaderData |   const { project, file } = useRouteLoaderData(PATHS.FILE) as IndexLoaderData | ||||||
|   const [kclSamples, setKclSamples] = React.useState<KclSamplesManifestItem[]>( |   const [kclSamples, setKclSamples] = React.useState<KclSamplesManifestItem[]>( | ||||||
| @ -57,7 +56,9 @@ export const FileMachineProvider = ({ | |||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     markOnce('code/didLoadFile') |     markOnce('code/didLoadFile') | ||||||
|     async function fetchKclSamples() { |     async function fetchKclSamples() { | ||||||
|       setKclSamples(await getKclSamplesManifest()) |       const manifest = await getKclSamplesManifest() | ||||||
|  |       const filteredFiles = manifest.filter((file) => !file.multipleFiles) | ||||||
|  |       setKclSamples(filteredFiles) | ||||||
|     } |     } | ||||||
|     fetchKclSamples().catch(reportError) |     fetchKclSamples().catch(reportError) | ||||||
|   }, []) |   }, []) | ||||||
| @ -88,7 +89,7 @@ export const FileMachineProvider = ({ | |||||||
|         navigateToFile: ({ context, event }) => { |         navigateToFile: ({ context, event }) => { | ||||||
|           if (event.type !== 'xstate.done.actor.create-and-open-file') return |           if (event.type !== 'xstate.done.actor.create-and-open-file') return | ||||||
|           if (event.output && 'name' in event.output) { |           if (event.output && 'name' in event.output) { | ||||||
|             commandBarSend({ type: 'Close' }) |             commandBarActor.send({ type: 'Close' }) | ||||||
|             navigate( |             navigate( | ||||||
|               `..${PATHS.FILE}/${encodeURIComponent( |               `..${PATHS.FILE}/${encodeURIComponent( | ||||||
|                 context.selectedDirectory + |                 context.selectedDirectory + | ||||||
| @ -324,7 +325,7 @@ export const FileMachineProvider = ({ | |||||||
|           } |           } | ||||||
|         }, |         }, | ||||||
|         kclSamples.map((sample) => ({ |         kclSamples.map((sample) => ({ | ||||||
|           value: sample.file, |           value: sample.pathFromProjectDirectoryToFirstFile, | ||||||
|           name: sample.title, |           name: sample.title, | ||||||
|         })) |         })) | ||||||
|       ).filter( |       ).filter( | ||||||
| @ -334,15 +335,18 @@ export const FileMachineProvider = ({ | |||||||
|   ) |   ) | ||||||
|  |  | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     commandBarSend({ type: 'Add commands', data: { commands: kclCommandMemo } }) |     commandBarActor.send({ | ||||||
|  |       type: 'Add commands', | ||||||
|  |       data: { commands: kclCommandMemo }, | ||||||
|  |     }) | ||||||
|  |  | ||||||
|     return () => { |     return () => { | ||||||
|       commandBarSend({ |       commandBarActor.send({ | ||||||
|         type: 'Remove commands', |         type: 'Remove commands', | ||||||
|         data: { commands: kclCommandMemo }, |         data: { commands: kclCommandMemo }, | ||||||
|       }) |       }) | ||||||
|     } |     } | ||||||
|   }, [commandBarSend, kclCommandMemo]) |   }, [commandBarActor.send, kclCommandMemo]) | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <FileContext.Provider |     <FileContext.Provider | ||||||
|  | |||||||
| @ -1,11 +1,11 @@ | |||||||
| import { createContext, useEffect, useState } from 'react' | import { createContext, useEffect, useState } from 'react' | ||||||
|  |  | ||||||
| import { engineCommandManager } from 'lib/singletons' | import { engineCommandManager } from 'lib/singletons' | ||||||
| import { CommandsContext } from 'components/CommandBar/CommandBarProvider' |  | ||||||
| import { isDesktop } from 'lib/isDesktop' | import { isDesktop } from 'lib/isDesktop' | ||||||
| import { components } from 'lib/machine-api' | import { components } from 'lib/machine-api' | ||||||
| import { reportRejection } from 'lib/trap' | import { reportRejection } from 'lib/trap' | ||||||
| import { toSync } from 'lib/utils' | import { toSync } from 'lib/utils' | ||||||
|  | import { commandBarActor } from 'machines/commandBarMachine' | ||||||
|  |  | ||||||
| export type MachinesListing = Array< | export type MachinesListing = Array< | ||||||
|   components['schemas']['MachineInfoResponse'] |   components['schemas']['MachineInfoResponse'] | ||||||
| @ -42,8 +42,6 @@ export const MachineManagerProvider = ({ | |||||||
|     components['schemas']['MachineInfoResponse'] | null |     components['schemas']['MachineInfoResponse'] | null | ||||||
|   >(null) |   >(null) | ||||||
|  |  | ||||||
|   const commandBarActor = CommandsContext.useActorRef() |  | ||||||
|  |  | ||||||
|   // Get the reason message for why there are no machines. |   // Get the reason message for why there are no machines. | ||||||
|   const noMachinesReason = (): string | undefined => { |   const noMachinesReason = (): string | undefined => { | ||||||
|     if (machines.length > 0) { |     if (machines.length > 0) { | ||||||
|  | |||||||
| @ -1,4 +1,4 @@ | |||||||
| import { useMachine } from '@xstate/react' | import { useMachine, useSelector } from '@xstate/react' | ||||||
| import React, { | import React, { | ||||||
|   createContext, |   createContext, | ||||||
|   useEffect, |   useEffect, | ||||||
| @ -11,6 +11,7 @@ import { | |||||||
|   AnyStateMachine, |   AnyStateMachine, | ||||||
|   ContextFrom, |   ContextFrom, | ||||||
|   Prop, |   Prop, | ||||||
|  |   SnapshotFrom, | ||||||
|   StateFrom, |   StateFrom, | ||||||
|   assign, |   assign, | ||||||
|   fromPromise, |   fromPromise, | ||||||
| @ -78,7 +79,6 @@ import toast from 'react-hot-toast' | |||||||
| import { useLoaderData, useNavigate, useSearchParams } from 'react-router-dom' | import { useLoaderData, useNavigate, useSearchParams } from 'react-router-dom' | ||||||
| import { letEngineAnimateAndSyncCamAfter } from 'clientSideScene/CameraControls' | import { letEngineAnimateAndSyncCamAfter } from 'clientSideScene/CameraControls' | ||||||
| import { err, reportRejection, trap } from 'lib/trap' | import { err, reportRejection, trap } from 'lib/trap' | ||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' |  | ||||||
| import { | import { | ||||||
|   ExportIntent, |   ExportIntent, | ||||||
|   EngineConnectionStateType, |   EngineConnectionStateType, | ||||||
| @ -91,6 +91,7 @@ import { IndexLoaderData } from 'lib/types' | |||||||
| import { Node } from 'wasm-lib/kcl/bindings/Node' | import { Node } from 'wasm-lib/kcl/bindings/Node' | ||||||
| import { promptToEditFlow } from 'lib/promptToEdit' | import { promptToEditFlow } from 'lib/promptToEdit' | ||||||
| import { kclEditorActor } from 'machines/kclEditorMachine' | import { kclEditorActor } from 'machines/kclEditorMachine' | ||||||
|  | import { commandBarActor } from 'machines/commandBarMachine' | ||||||
|  |  | ||||||
| type MachineContext<T extends AnyStateMachine> = { | type MachineContext<T extends AnyStateMachine> = { | ||||||
|   state: StateFrom<T> |   state: StateFrom<T> | ||||||
| @ -102,6 +103,10 @@ export const ModelingMachineContext = createContext( | |||||||
|   {} as MachineContext<typeof modelingMachine> |   {} as MachineContext<typeof modelingMachine> | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | const commandBarIsClosedSelector = ( | ||||||
|  |   state: SnapshotFrom<typeof commandBarActor> | ||||||
|  | ) => state.matches('Closed') | ||||||
|  |  | ||||||
| export const ModelingMachineProvider = ({ | export const ModelingMachineProvider = ({ | ||||||
|   children, |   children, | ||||||
| }: { | }: { | ||||||
| @ -111,7 +116,7 @@ export const ModelingMachineProvider = ({ | |||||||
|     auth, |     auth, | ||||||
|     settings: { |     settings: { | ||||||
|       context: { |       context: { | ||||||
|         app: { theme, enableSSAO }, |         app: { theme, enableSSAO, allowOrbitInSketchMode }, | ||||||
|         modeling: { |         modeling: { | ||||||
|           defaultUnit, |           defaultUnit, | ||||||
|           cameraProjection, |           cameraProjection, | ||||||
| @ -121,6 +126,7 @@ export const ModelingMachineProvider = ({ | |||||||
|       }, |       }, | ||||||
|     }, |     }, | ||||||
|   } = useSettingsAuthContext() |   } = useSettingsAuthContext() | ||||||
|  |   const previousAllowOrbitInSketchMode = useRef(allowOrbitInSketchMode.current) | ||||||
|   const navigate = useNavigate() |   const navigate = useNavigate() | ||||||
|   const { context, send: fileMachineSend } = useFileContext() |   const { context, send: fileMachineSend } = useFileContext() | ||||||
|   const { file } = useLoaderData() as IndexLoaderData |   const { file } = useLoaderData() as IndexLoaderData | ||||||
| @ -131,8 +137,10 @@ export const ModelingMachineProvider = ({ | |||||||
|   let [searchParams] = useSearchParams() |   let [searchParams] = useSearchParams() | ||||||
|   const pool = searchParams.get('pool') |   const pool = searchParams.get('pool') | ||||||
|  |  | ||||||
|   const { commandBarState, commandBarSend } = useCommandsContext() |   const isCommandBarClosed = useSelector( | ||||||
|  |     commandBarActor, | ||||||
|  |     commandBarIsClosedSelector | ||||||
|  |   ) | ||||||
|   // Settings machine setup |   // Settings machine setup | ||||||
|   // const retrievedSettings = useRef( |   // const retrievedSettings = useRef( | ||||||
|   // localStorage?.getItem(MODELING_PERSIST_KEY) || '{}' |   // localStorage?.getItem(MODELING_PERSIST_KEY) || '{}' | ||||||
| @ -387,7 +395,16 @@ export const ModelingMachineProvider = ({ | |||||||
|             } |             } | ||||||
|  |  | ||||||
|             if (setSelections.selectionType === 'completeSelection') { |             if (setSelections.selectionType === 'completeSelection') { | ||||||
|               editorManager.selectRange(setSelections.selection) |               const codeMirrorSelection = editorManager.createEditorSelection( | ||||||
|  |                 setSelections.selection | ||||||
|  |               ) | ||||||
|  |               kclEditorActor.send({ | ||||||
|  |                 type: 'setLastSelectionEvent', | ||||||
|  |                 data: { | ||||||
|  |                   codeMirrorSelection, | ||||||
|  |                   scrollIntoView: false, | ||||||
|  |                 }, | ||||||
|  |               }) | ||||||
|               if (!sketchDetails) |               if (!sketchDetails) | ||||||
|                 return { |                 return { | ||||||
|                   selectionRanges: setSelections.selection, |                   selectionRanges: setSelections.selection, | ||||||
| @ -528,7 +545,6 @@ export const ModelingMachineProvider = ({ | |||||||
|             trimmedPrompt, |             trimmedPrompt, | ||||||
|             fileMachineSend, |             fileMachineSend, | ||||||
|             navigate, |             navigate, | ||||||
|             commandBarSend, |  | ||||||
|             context, |             context, | ||||||
|             token, |             token, | ||||||
|             settings: { |             settings: { | ||||||
| @ -542,7 +558,7 @@ export const ModelingMachineProvider = ({ | |||||||
|         'has valid selection for deletion': ({ |         'has valid selection for deletion': ({ | ||||||
|           context: { selectionRanges }, |           context: { selectionRanges }, | ||||||
|         }) => { |         }) => { | ||||||
|           if (!commandBarState.matches('Closed')) return false |           if (!isCommandBarClosed) return false | ||||||
|           if (selectionRanges.graphSelections.length <= 0) return false |           if (selectionRanges.graphSelections.length <= 0) return false | ||||||
|           return true |           return true | ||||||
|         }, |         }, | ||||||
| @ -634,7 +650,8 @@ export const ModelingMachineProvider = ({ | |||||||
|             input.plane |             input.plane | ||||||
|           ) |           ) | ||||||
|           await kclManager.updateAst(modifiedAst, false) |           await kclManager.updateAst(modifiedAst, false) | ||||||
|           sceneInfra.camControls.enableRotate = false |           sceneInfra.camControls.enableRotate = | ||||||
|  |             sceneInfra.camControls._setting_allowOrbitInSketchMode | ||||||
|           sceneInfra.camControls.syncDirection = 'clientToEngine' |           sceneInfra.camControls.syncDirection = 'clientToEngine' | ||||||
|  |  | ||||||
|           await letEngineAnimateAndSyncCamAfter( |           await letEngineAnimateAndSyncCamAfter( | ||||||
| @ -647,6 +664,7 @@ export const ModelingMachineProvider = ({ | |||||||
|             zAxis: input.zAxis, |             zAxis: input.zAxis, | ||||||
|             yAxis: input.yAxis, |             yAxis: input.yAxis, | ||||||
|             origin: [0, 0, 0], |             origin: [0, 0, 0], | ||||||
|  |             animateTargetId: input.planeId, | ||||||
|           } |           } | ||||||
|         }), |         }), | ||||||
|         'animate-to-sketch': fromPromise( |         'animate-to-sketch': fromPromise( | ||||||
| @ -671,6 +689,7 @@ export const ModelingMachineProvider = ({ | |||||||
|               origin: info.sketchDetails.origin.map( |               origin: info.sketchDetails.origin.map( | ||||||
|                 (a) => a / sceneInfra._baseUnitMultiplier |                 (a) => a / sceneInfra._baseUnitMultiplier | ||||||
|               ) as [number, number, number], |               ) as [number, number, number], | ||||||
|  |               animateTargetId: info?.sketchDetails?.faceId || '', | ||||||
|             } |             } | ||||||
|           } |           } | ||||||
|         ), |         ), | ||||||
| @ -1188,6 +1207,41 @@ export const ModelingMachineProvider = ({ | |||||||
|     } |     } | ||||||
|   }, [engineCommandManager.engineConnection, modelingSend]) |   }, [engineCommandManager.engineConnection, modelingSend]) | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     // Only trigger this if the state actually changes, if it stays the same do not reload the camera | ||||||
|  |     if ( | ||||||
|  |       previousAllowOrbitInSketchMode.current === allowOrbitInSketchMode.current | ||||||
|  |     ) { | ||||||
|  |       //no op | ||||||
|  |       previousAllowOrbitInSketchMode.current = allowOrbitInSketchMode.current | ||||||
|  |       return | ||||||
|  |     } | ||||||
|  |     const inSketchMode = modelingState.matches('Sketch') | ||||||
|  |  | ||||||
|  |     // If you are in sketch mode and you disable the orbit, return back to the normal view to the target | ||||||
|  |     if (!allowOrbitInSketchMode.current) { | ||||||
|  |       const targetId = modelingState.context.sketchDetails?.animateTargetId | ||||||
|  |       if (inSketchMode && targetId) { | ||||||
|  |         letEngineAnimateAndSyncCamAfter(engineCommandManager, targetId) | ||||||
|  |           .then(() => {}) | ||||||
|  |           .catch((e) => { | ||||||
|  |             console.error( | ||||||
|  |               'failed to sync engine and client scene after disabling allow orbit in sketch mode' | ||||||
|  |             ) | ||||||
|  |             console.error(e) | ||||||
|  |           }) | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // While you are in sketch mode you should be able to control the enable rotate | ||||||
|  |     // Once you exit it goes back to normal | ||||||
|  |     if (inSketchMode) { | ||||||
|  |       sceneInfra.camControls.enableRotate = allowOrbitInSketchMode.current | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     previousAllowOrbitInSketchMode.current = allowOrbitInSketchMode.current | ||||||
|  |   }, [allowOrbitInSketchMode]) | ||||||
|  |  | ||||||
|   // Allow using the delete key to delete solids |   // Allow using the delete key to delete solids | ||||||
|   useHotkeys(['backspace', 'delete', 'del'], () => { |   useHotkeys(['backspace', 'delete', 'del'], () => { | ||||||
|     modelingSend({ type: 'Delete selection' }) |     modelingSend({ type: 'Delete selection' }) | ||||||
|  | |||||||
| @ -1,4 +1,4 @@ | |||||||
| import { DebugFeatureTree } from 'components/DebugFeatureTree' | import { DebugArtifactGraph } from 'components/DebugArtifactGraph' | ||||||
| import { AstExplorer } from '../../AstExplorer' | import { AstExplorer } from '../../AstExplorer' | ||||||
| import { EngineCommands } from '../../EngineCommands' | import { EngineCommands } from '../../EngineCommands' | ||||||
| import { CamDebugSettings } from 'clientSideScene/ClientSideSceneComp' | import { CamDebugSettings } from 'clientSideScene/ClientSideSceneComp' | ||||||
| @ -14,7 +14,7 @@ export const DebugPane = () => { | |||||||
|           <EngineCommands /> |           <EngineCommands /> | ||||||
|           <CamDebugSettings /> |           <CamDebugSettings /> | ||||||
|           <AstExplorer /> |           <AstExplorer /> | ||||||
|           <DebugFeatureTree /> |           <DebugArtifactGraph /> | ||||||
|         </div> |         </div> | ||||||
|       </section> |       </section> | ||||||
|     </div> |     </div> | ||||||
|  | |||||||
| @ -3,6 +3,7 @@ | |||||||
|   @apply font-mono !no-underline text-xs font-bold select-none text-chalkboard-90; |   @apply font-mono !no-underline text-xs font-bold select-none text-chalkboard-90; | ||||||
|   @apply ui-active:bg-primary/10 ui-active:text-primary ui-active:text-inherit; |   @apply ui-active:bg-primary/10 ui-active:text-primary ui-active:text-inherit; | ||||||
|   @apply transition-colors ease-out; |   @apply transition-colors ease-out; | ||||||
|  |   @apply m-0; | ||||||
| } | } | ||||||
|  |  | ||||||
| :global(.dark) .button { | :global(.dark) .button { | ||||||
|  | |||||||
| @ -9,12 +9,11 @@ import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' | |||||||
| import { kclManager } from 'lib/singletons' | import { kclManager } from 'lib/singletons' | ||||||
| import { openExternalBrowserIfDesktop } from 'lib/openWindow' | import { openExternalBrowserIfDesktop } from 'lib/openWindow' | ||||||
| import { reportRejection } from 'lib/trap' | import { reportRejection } from 'lib/trap' | ||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' | import { commandBarActor } from 'machines/commandBarMachine' | ||||||
|  |  | ||||||
| export const KclEditorMenu = ({ children }: PropsWithChildren) => { | export const KclEditorMenu = ({ children }: PropsWithChildren) => { | ||||||
|   const { enable: convertToVarEnabled, handleClick: handleConvertToVarClick } = |   const { enable: convertToVarEnabled, handleClick: handleConvertToVarClick } = | ||||||
|     useConvertToVariable() |     useConvertToVariable() | ||||||
|   const { commandBarSend } = useCommandsContext() |  | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <Menu> |     <Menu> | ||||||
| @ -85,7 +84,7 @@ export const KclEditorMenu = ({ children }: PropsWithChildren) => { | |||||||
|           <Menu.Item> |           <Menu.Item> | ||||||
|             <button |             <button | ||||||
|               onClick={() => { |               onClick={() => { | ||||||
|                 commandBarSend({ |                 commandBarActor.send({ | ||||||
|                   type: 'Find and select command', |                   type: 'Find and select command', | ||||||
|                   data: { |                   data: { | ||||||
|                     groupId: 'code', |                     groupId: 'code', | ||||||
|  | |||||||
| @ -95,9 +95,11 @@ export const processMemory = (programMemory: ProgramMemory) => { | |||||||
|     ) { |     ) { | ||||||
|       const sk = sketchFromKclValueOptional(val, key) |       const sk = sketchFromKclValueOptional(val, key) | ||||||
|       if (val.type === 'Solid') { |       if (val.type === 'Solid') { | ||||||
|         processedMemory[key] = val.value.map(({ ...rest }: ExtrudeSurface) => { |         processedMemory[key] = val.value.value.map( | ||||||
|           return rest |           ({ ...rest }: ExtrudeSurface) => { | ||||||
|         }) |             return rest | ||||||
|  |           } | ||||||
|  |         ) | ||||||
|       } else if (!(sk instanceof Reason)) { |       } else if (!(sk instanceof Reason)) { | ||||||
|         processedMemory[key] = sk.paths.map(({ __geoMeta, ...rest }: Path) => { |         processedMemory[key] = sk.paths.map(({ __geoMeta, ...rest }: Path) => { | ||||||
|           return rest |           return rest | ||||||
|  | |||||||
| @ -15,12 +15,12 @@ import { ModelingPane } from './ModelingPane' | |||||||
| import { isDesktop } from 'lib/isDesktop' | import { isDesktop } from 'lib/isDesktop' | ||||||
| import { useModelingContext } from 'hooks/useModelingContext' | import { useModelingContext } from 'hooks/useModelingContext' | ||||||
| import { CustomIconName } from 'components/CustomIcon' | import { CustomIconName } from 'components/CustomIcon' | ||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' |  | ||||||
| import { IconDefinition } from '@fortawesome/free-solid-svg-icons' | import { IconDefinition } from '@fortawesome/free-solid-svg-icons' | ||||||
| import { useKclContext } from 'lang/KclProvider' | import { useKclContext } from 'lang/KclProvider' | ||||||
| import { MachineManagerContext } from 'components/MachineManagerProvider' | import { MachineManagerContext } from 'components/MachineManagerProvider' | ||||||
| import { onboardingPaths } from 'routes/Onboarding/paths' | import { onboardingPaths } from 'routes/Onboarding/paths' | ||||||
| import { SIDEBAR_BUTTON_SUFFIX } from 'lib/constants' | import { SIDEBAR_BUTTON_SUFFIX } from 'lib/constants' | ||||||
|  | import { commandBarActor } from 'machines/commandBarMachine' | ||||||
|  |  | ||||||
| interface ModelingSidebarProps { | interface ModelingSidebarProps { | ||||||
|   paneOpacity: '' | 'opacity-20' | 'opacity-40' |   paneOpacity: '' | 'opacity-20' | 'opacity-40' | ||||||
| @ -37,7 +37,6 @@ function getPlatformString(): 'web' | 'desktop' { | |||||||
|  |  | ||||||
| export function ModelingSidebar({ paneOpacity }: ModelingSidebarProps) { | export function ModelingSidebar({ paneOpacity }: ModelingSidebarProps) { | ||||||
|   const machineManager = useContext(MachineManagerContext) |   const machineManager = useContext(MachineManagerContext) | ||||||
|   const { commandBarSend } = useCommandsContext() |  | ||||||
|   const kclContext = useKclContext() |   const kclContext = useKclContext() | ||||||
|   const { settings } = useSettingsAuthContext() |   const { settings } = useSettingsAuthContext() | ||||||
|   const onboardingStatus = settings.context.app.onboardingStatus |   const onboardingStatus = settings.context.app.onboardingStatus | ||||||
| @ -66,7 +65,7 @@ export function ModelingSidebar({ paneOpacity }: ModelingSidebarProps) { | |||||||
|       icon: 'floppyDiskArrow', |       icon: 'floppyDiskArrow', | ||||||
|       keybinding: 'Ctrl + Shift + E', |       keybinding: 'Ctrl + Shift + E', | ||||||
|       action: () => |       action: () => | ||||||
|         commandBarSend({ |         commandBarActor.send({ | ||||||
|           type: 'Find and select command', |           type: 'Find and select command', | ||||||
|           data: { name: 'Export', groupId: 'modeling' }, |           data: { name: 'Export', groupId: 'modeling' }, | ||||||
|         }), |         }), | ||||||
| @ -79,7 +78,7 @@ export function ModelingSidebar({ paneOpacity }: ModelingSidebarProps) { | |||||||
|       keybinding: 'Ctrl + Shift + M', |       keybinding: 'Ctrl + Shift + M', | ||||||
|       // eslint-disable-next-line @typescript-eslint/no-misused-promises |       // eslint-disable-next-line @typescript-eslint/no-misused-promises | ||||||
|       action: async () => { |       action: async () => { | ||||||
|         commandBarSend({ |         commandBarActor.send({ | ||||||
|           type: 'Find and select command', |           type: 'Find and select command', | ||||||
|           data: { name: 'Make', groupId: 'modeling' }, |           data: { name: 'Make', groupId: 'modeling' }, | ||||||
|         }) |         }) | ||||||
|  | |||||||
| @ -1,7 +1,6 @@ | |||||||
| import { fireEvent, render, screen } from '@testing-library/react' | import { fireEvent, render, screen } from '@testing-library/react' | ||||||
| import { BrowserRouter } from 'react-router-dom' | import { BrowserRouter } from 'react-router-dom' | ||||||
| import { SettingsAuthProviderJest } from './SettingsAuthProvider' | import { SettingsAuthProviderJest } from './SettingsAuthProvider' | ||||||
| import { CommandBarProvider } from './CommandBar/CommandBarProvider' |  | ||||||
| import { | import { | ||||||
|   NETWORK_HEALTH_TEXT, |   NETWORK_HEALTH_TEXT, | ||||||
|   NetworkHealthIndicator, |   NetworkHealthIndicator, | ||||||
| @ -12,9 +11,7 @@ function TestWrap({ children }: { children: React.ReactNode }) { | |||||||
|   // wrap in router and xState context |   // wrap in router and xState context | ||||||
|   return ( |   return ( | ||||||
|     <BrowserRouter> |     <BrowserRouter> | ||||||
|       <CommandBarProvider> |       <SettingsAuthProviderJest>{children}</SettingsAuthProviderJest> | ||||||
|         <SettingsAuthProviderJest>{children}</SettingsAuthProviderJest> |  | ||||||
|       </CommandBarProvider> |  | ||||||
|     </BrowserRouter> |     </BrowserRouter> | ||||||
|   ) |   ) | ||||||
| } | } | ||||||
|  | |||||||
| @ -2,7 +2,6 @@ import { render, screen } from '@testing-library/react' | |||||||
| import { BrowserRouter } from 'react-router-dom' | import { BrowserRouter } from 'react-router-dom' | ||||||
| import ProjectSidebarMenu from './ProjectSidebarMenu' | import ProjectSidebarMenu from './ProjectSidebarMenu' | ||||||
| import { SettingsAuthProviderJest } from './SettingsAuthProvider' | import { SettingsAuthProviderJest } from './SettingsAuthProvider' | ||||||
| import { CommandBarProvider } from './CommandBar/CommandBarProvider' |  | ||||||
| import { Project } from 'lib/project' | import { Project } from 'lib/project' | ||||||
|  |  | ||||||
| const now = new Date() | const now = new Date() | ||||||
| @ -33,11 +32,9 @@ describe('ProjectSidebarMenu tests', () => { | |||||||
|   test('Disables popover menu by default', () => { |   test('Disables popover menu by default', () => { | ||||||
|     render( |     render( | ||||||
|       <BrowserRouter> |       <BrowserRouter> | ||||||
|         <CommandBarProvider> |         <SettingsAuthProviderJest> | ||||||
|           <SettingsAuthProviderJest> |           <ProjectSidebarMenu project={projectWellFormed} /> | ||||||
|             <ProjectSidebarMenu project={projectWellFormed} /> |         </SettingsAuthProviderJest> | ||||||
|           </SettingsAuthProviderJest> |  | ||||||
|         </CommandBarProvider> |  | ||||||
|       </BrowserRouter> |       </BrowserRouter> | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|  | |||||||
| @ -7,7 +7,6 @@ import { Link, useLocation, useNavigate } from 'react-router-dom' | |||||||
| import { Fragment, useMemo, useContext } from 'react' | import { Fragment, useMemo, useContext } from 'react' | ||||||
| import { Logo } from './Logo' | import { Logo } from './Logo' | ||||||
| import { APP_NAME } from 'lib/constants' | import { APP_NAME } from 'lib/constants' | ||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' |  | ||||||
| import { CustomIcon } from './CustomIcon' | import { CustomIcon } from './CustomIcon' | ||||||
| import { useLspContext } from './LspProvider' | import { useLspContext } from './LspProvider' | ||||||
| import { engineCommandManager, kclManager } from 'lib/singletons' | import { engineCommandManager, kclManager } from 'lib/singletons' | ||||||
| @ -15,6 +14,9 @@ import { MachineManagerContext } from 'components/MachineManagerProvider' | |||||||
| import usePlatform from 'hooks/usePlatform' | import usePlatform from 'hooks/usePlatform' | ||||||
| import { useAbsoluteFilePath } from 'hooks/useAbsoluteFilePath' | import { useAbsoluteFilePath } from 'hooks/useAbsoluteFilePath' | ||||||
| import Tooltip from './Tooltip' | import Tooltip from './Tooltip' | ||||||
|  | import { SnapshotFrom } from 'xstate' | ||||||
|  | import { commandBarActor } from 'machines/commandBarMachine' | ||||||
|  | import { useSelector } from '@xstate/react' | ||||||
|  |  | ||||||
| const ProjectSidebarMenu = ({ | const ProjectSidebarMenu = ({ | ||||||
|   project, |   project, | ||||||
| @ -84,6 +86,9 @@ function AppLogoLink({ | |||||||
|   ) |   ) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | const commandsSelector = (state: SnapshotFrom<typeof commandBarActor>) => | ||||||
|  |   state.context.commands | ||||||
|  |  | ||||||
| function ProjectMenuPopover({ | function ProjectMenuPopover({ | ||||||
|   project, |   project, | ||||||
|   file, |   file, | ||||||
| @ -96,16 +101,14 @@ function ProjectMenuPopover({ | |||||||
|   const navigate = useNavigate() |   const navigate = useNavigate() | ||||||
|   const filePath = useAbsoluteFilePath() |   const filePath = useAbsoluteFilePath() | ||||||
|   const machineManager = useContext(MachineManagerContext) |   const machineManager = useContext(MachineManagerContext) | ||||||
|  |   const commands = useSelector(commandBarActor, commandsSelector) | ||||||
|  |  | ||||||
|   const { commandBarState, commandBarSend } = useCommandsContext() |  | ||||||
|   const { onProjectClose } = useLspContext() |   const { onProjectClose } = useLspContext() | ||||||
|   const exportCommandInfo = { name: 'Export', groupId: 'modeling' } |   const exportCommandInfo = { name: 'Export', groupId: 'modeling' } | ||||||
|   const makeCommandInfo = { name: 'Make', groupId: 'modeling' } |   const makeCommandInfo = { name: 'Make', groupId: 'modeling' } | ||||||
|   const findCommand = (obj: { name: string; groupId: string }) => |   const findCommand = (obj: { name: string; groupId: string }) => | ||||||
|     Boolean( |     Boolean( | ||||||
|       commandBarState.context.commands.find( |       commands.find((c) => c.name === obj.name && c.groupId === obj.groupId) | ||||||
|         (c) => c.name === obj.name && c.groupId === obj.groupId |  | ||||||
|       ) |  | ||||||
|     ) |     ) | ||||||
|   const machineCount = machineManager.machines.length |   const machineCount = machineManager.machines.length | ||||||
|  |  | ||||||
| @ -150,7 +153,7 @@ function ProjectMenuPopover({ | |||||||
|           ), |           ), | ||||||
|           disabled: !findCommand(exportCommandInfo), |           disabled: !findCommand(exportCommandInfo), | ||||||
|           onClick: () => |           onClick: () => | ||||||
|             commandBarSend({ |             commandBarActor.send({ | ||||||
|               type: 'Find and select command', |               type: 'Find and select command', | ||||||
|               data: exportCommandInfo, |               data: exportCommandInfo, | ||||||
|             }), |             }), | ||||||
| @ -175,7 +178,7 @@ function ProjectMenuPopover({ | |||||||
|           ), |           ), | ||||||
|           disabled: !findCommand(makeCommandInfo) || machineCount === 0, |           disabled: !findCommand(makeCommandInfo) || machineCount === 0, | ||||||
|           onClick: () => { |           onClick: () => { | ||||||
|             commandBarSend({ |             commandBarActor.send({ | ||||||
|               type: 'Find and select command', |               type: 'Find and select command', | ||||||
|               data: makeCommandInfo, |               data: makeCommandInfo, | ||||||
|             }) |             }) | ||||||
| @ -200,7 +203,7 @@ function ProjectMenuPopover({ | |||||||
|     [ |     [ | ||||||
|       platform, |       platform, | ||||||
|       findCommand, |       findCommand, | ||||||
|       commandBarSend, |       commandBarActor.send, | ||||||
|       engineCommandManager, |       engineCommandManager, | ||||||
|       onProjectClose, |       onProjectClose, | ||||||
|       isDesktop, |       isDesktop, | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| import { useMachine } from '@xstate/react' | import { useMachine } from '@xstate/react' | ||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' |  | ||||||
| import { useFileSystemWatcher } from 'hooks/useFileSystemWatcher' | import { useFileSystemWatcher } from 'hooks/useFileSystemWatcher' | ||||||
| import { useProjectsLoader } from 'hooks/useProjectsLoader' | import { useProjectsLoader } from 'hooks/useProjectsLoader' | ||||||
| import { projectsMachine } from 'machines/projectsMachine' | import { projectsMachine } from 'machines/projectsMachine' | ||||||
| @ -18,11 +17,13 @@ import { | |||||||
|   getNextProjectIndex, |   getNextProjectIndex, | ||||||
|   interpolateProjectNameWithIndex, |   interpolateProjectNameWithIndex, | ||||||
|   doesProjectNameNeedInterpolated, |   doesProjectNameNeedInterpolated, | ||||||
|  |   getUniqueProjectName, | ||||||
| } from 'lib/desktopFS' | } from 'lib/desktopFS' | ||||||
| import { useSettingsAuthContext } from 'hooks/useSettingsAuthContext' | import { useSettingsAuthContext } from 'hooks/useSettingsAuthContext' | ||||||
| import useStateMachineCommands from 'hooks/useStateMachineCommands' | import useStateMachineCommands from 'hooks/useStateMachineCommands' | ||||||
| import { projectsCommandBarConfig } from 'lib/commandBarConfigs/projectsCommandConfig' | import { projectsCommandBarConfig } from 'lib/commandBarConfigs/projectsCommandConfig' | ||||||
| import { isDesktop } from 'lib/isDesktop' | import { isDesktop } from 'lib/isDesktop' | ||||||
|  | import { commandBarActor } from 'machines/commandBarMachine' | ||||||
|  |  | ||||||
| type MachineContext<T extends AnyStateMachine> = { | type MachineContext<T extends AnyStateMachine> = { | ||||||
|   state?: StateFrom<T> |   state?: StateFrom<T> | ||||||
| @ -72,7 +73,6 @@ const ProjectsContextDesktop = ({ | |||||||
| }) => { | }) => { | ||||||
|   const navigate = useNavigate() |   const navigate = useNavigate() | ||||||
|   const location = useLocation() |   const location = useLocation() | ||||||
|   const { commandBarSend } = useCommandsContext() |  | ||||||
|   const { onProjectOpen } = useLspContext() |   const { onProjectOpen } = useLspContext() | ||||||
|   const { |   const { | ||||||
|     settings: { context: settings }, |     settings: { context: settings }, | ||||||
| @ -125,7 +125,7 @@ const ProjectsContextDesktop = ({ | |||||||
|               }, |               }, | ||||||
|               null |               null | ||||||
|             ) |             ) | ||||||
|             commandBarSend({ type: 'Close' }) |             commandBarActor.send({ type: 'Close' }) | ||||||
|             const newPathName = `${PATHS.FILE}/${encodeURIComponent( |             const newPathName = `${PATHS.FILE}/${encodeURIComponent( | ||||||
|               projectPath |               projectPath | ||||||
|             )}` |             )}` | ||||||
| @ -195,16 +195,12 @@ const ProjectsContextDesktop = ({ | |||||||
|               : settings.projects.defaultProjectName.current |               : settings.projects.defaultProjectName.current | ||||||
|           ).trim() |           ).trim() | ||||||
|  |  | ||||||
|           if (doesProjectNameNeedInterpolated(name)) { |           const uniqueName = getUniqueProjectName(name, input.projects) | ||||||
|             const nextIndex = getNextProjectIndex(name, input.projects) |           await createNewProjectDirectory(uniqueName) | ||||||
|             name = interpolateProjectNameWithIndex(name, nextIndex) |  | ||||||
|           } |  | ||||||
|  |  | ||||||
|           await createNewProjectDirectory(name) |  | ||||||
|  |  | ||||||
|           return { |           return { | ||||||
|             message: `Successfully created "${name}"`, |             message: `Successfully created "${uniqueName}"`, | ||||||
|             name, |             name: uniqueName, | ||||||
|           } |           } | ||||||
|         }), |         }), | ||||||
|         renameProject: fromPromise(async ({ input }) => { |         renameProject: fromPromise(async ({ input }) => { | ||||||
|  | |||||||
| @ -29,7 +29,6 @@ import { | |||||||
|   createSettingsCommand, |   createSettingsCommand, | ||||||
|   settingsWithCommandConfigs, |   settingsWithCommandConfigs, | ||||||
| } from 'lib/commandBarConfigs/settingsCommandConfig' | } from 'lib/commandBarConfigs/settingsCommandConfig' | ||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' |  | ||||||
| import { Command } from 'lib/commandTypes' | import { Command } from 'lib/commandTypes' | ||||||
| import { BaseUnit } from 'lib/settings/settingsTypes' | import { BaseUnit } from 'lib/settings/settingsTypes' | ||||||
| import { | import { | ||||||
| @ -42,6 +41,7 @@ import { isDesktop } from 'lib/isDesktop' | |||||||
| import { useFileSystemWatcher } from 'hooks/useFileSystemWatcher' | import { useFileSystemWatcher } from 'hooks/useFileSystemWatcher' | ||||||
| import { codeManager } from 'lib/singletons' | import { codeManager } from 'lib/singletons' | ||||||
| import { createRouteCommands } from 'lib/commandBarConfigs/routeCommandConfig' | import { createRouteCommands } from 'lib/commandBarConfigs/routeCommandConfig' | ||||||
|  | import { commandBarActor } from 'machines/commandBarMachine' | ||||||
|  |  | ||||||
| type MachineContext<T extends AnyStateMachine> = { | type MachineContext<T extends AnyStateMachine> = { | ||||||
|   state: StateFrom<T> |   state: StateFrom<T> | ||||||
| @ -109,7 +109,6 @@ export const SettingsAuthProviderBase = ({ | |||||||
| }) => { | }) => { | ||||||
|   const location = useLocation() |   const location = useLocation() | ||||||
|   const navigate = useNavigate() |   const navigate = useNavigate() | ||||||
|   const { commandBarSend } = useCommandsContext() |  | ||||||
|   const [settingsPath, setSettingsPath] = useState<string | undefined>( |   const [settingsPath, setSettingsPath] = useState<string | undefined>( | ||||||
|     undefined |     undefined | ||||||
|   ) |   ) | ||||||
| @ -137,6 +136,11 @@ export const SettingsAuthProviderBase = ({ | |||||||
|           sceneInfra.theme = opposingTheme |           sceneInfra.theme = opposingTheme | ||||||
|           sceneEntitiesManager.updateSegmentBaseColor(opposingTheme) |           sceneEntitiesManager.updateSegmentBaseColor(opposingTheme) | ||||||
|         }, |         }, | ||||||
|  |         setAllowOrbitInSketchMode: ({ context }) => { | ||||||
|  |           sceneInfra.camControls._setting_allowOrbitInSketchMode = | ||||||
|  |             context.app.allowOrbitInSketchMode.current | ||||||
|  |           // ModelingMachineProvider will do a use effect to trigger the camera engine sync | ||||||
|  |         }, | ||||||
|         toastSuccess: ({ event }) => { |         toastSuccess: ({ event }) => { | ||||||
|           if (!('data' in event)) return |           if (!('data' in event)) return | ||||||
|           const eventParts = event.type.replace(/^set./, '').split('.') as [ |           const eventParts = event.type.replace(/^set./, '').split('.') as [ | ||||||
| @ -273,10 +277,10 @@ export const SettingsAuthProviderBase = ({ | |||||||
|       ) |       ) | ||||||
|       .filter((c) => c !== null) as Command[] |       .filter((c) => c !== null) as Command[] | ||||||
|  |  | ||||||
|     commandBarSend({ type: 'Add commands', data: { commands: commands } }) |     commandBarActor.send({ type: 'Add commands', data: { commands: commands } }) | ||||||
|  |  | ||||||
|     return () => { |     return () => { | ||||||
|       commandBarSend({ |       commandBarActor.send({ | ||||||
|         type: 'Remove commands', |         type: 'Remove commands', | ||||||
|         data: { commands }, |         data: { commands }, | ||||||
|       }) |       }) | ||||||
| @ -285,7 +289,7 @@ export const SettingsAuthProviderBase = ({ | |||||||
|     settingsState, |     settingsState, | ||||||
|     settingsSend, |     settingsSend, | ||||||
|     settingsActor, |     settingsActor, | ||||||
|     commandBarSend, |     commandBarActor.send, | ||||||
|     settingsWithCommandConfigs, |     settingsWithCommandConfigs, | ||||||
|   ]) |   ]) | ||||||
|  |  | ||||||
| @ -298,7 +302,7 @@ export const SettingsAuthProviderBase = ({ | |||||||
|       encodeURIComponent(loadedProject?.file?.path || BROWSER_PATH) |       encodeURIComponent(loadedProject?.file?.path || BROWSER_PATH) | ||||||
|     const { RouteTelemetryCommand, RouteHomeCommand, RouteSettingsCommand } = |     const { RouteTelemetryCommand, RouteHomeCommand, RouteSettingsCommand } = | ||||||
|       createRouteCommands(navigate, location, filePath) |       createRouteCommands(navigate, location, filePath) | ||||||
|     commandBarSend({ |     commandBarActor.send({ | ||||||
|       type: 'Remove commands', |       type: 'Remove commands', | ||||||
|       data: { |       data: { | ||||||
|         commands: [ |         commands: [ | ||||||
| @ -309,12 +313,12 @@ export const SettingsAuthProviderBase = ({ | |||||||
|       }, |       }, | ||||||
|     }) |     }) | ||||||
|     if (location.pathname === PATHS.HOME) { |     if (location.pathname === PATHS.HOME) { | ||||||
|       commandBarSend({ |       commandBarActor.send({ | ||||||
|         type: 'Add commands', |         type: 'Add commands', | ||||||
|         data: { commands: [RouteTelemetryCommand, RouteSettingsCommand] }, |         data: { commands: [RouteTelemetryCommand, RouteSettingsCommand] }, | ||||||
|       }) |       }) | ||||||
|     } else if (location.pathname.includes(PATHS.FILE)) { |     } else if (location.pathname.includes(PATHS.FILE)) { | ||||||
|       commandBarSend({ |       commandBarActor.send({ | ||||||
|         type: 'Add commands', |         type: 'Add commands', | ||||||
|         data: { |         data: { | ||||||
|           commands: [ |           commands: [ | ||||||
|  | |||||||
| @ -17,10 +17,11 @@ import { | |||||||
| import { useRouteLoaderData } from 'react-router-dom' | import { useRouteLoaderData } from 'react-router-dom' | ||||||
| import { PATHS } from 'lib/paths' | import { PATHS } from 'lib/paths' | ||||||
| import { IndexLoaderData } from 'lib/types' | import { IndexLoaderData } from 'lib/types' | ||||||
| import { useCommandsContext } from 'hooks/useCommandsContext' |  | ||||||
| import { err, reportRejection } from 'lib/trap' | import { err, reportRejection } from 'lib/trap' | ||||||
| import { getArtifactOfTypes } from 'lang/std/artifactGraph' | import { getArtifactOfTypes } from 'lang/std/artifactGraph' | ||||||
| import { ViewControlContextMenu } from './ViewControlMenu' | import { ViewControlContextMenu } from './ViewControlMenu' | ||||||
|  | import { commandBarActor, useCommandBarState } from 'machines/commandBarMachine' | ||||||
|  | import { useSelector } from '@xstate/react' | ||||||
|  |  | ||||||
| enum StreamState { | enum StreamState { | ||||||
|   Playing = 'playing', |   Playing = 'playing', | ||||||
| @ -35,7 +36,7 @@ export const Stream = () => { | |||||||
|   const videoRef = useRef<HTMLVideoElement>(null) |   const videoRef = useRef<HTMLVideoElement>(null) | ||||||
|   const { settings } = useSettingsAuthContext() |   const { settings } = useSettingsAuthContext() | ||||||
|   const { state, send } = useModelingContext() |   const { state, send } = useModelingContext() | ||||||
|   const { commandBarState } = useCommandsContext() |   const commandBarState = useCommandBarState() | ||||||
|   const { mediaStream } = useAppStream() |   const { mediaStream } = useAppStream() | ||||||
|   const { overallState, immediateState } = useNetworkContext() |   const { overallState, immediateState } = useNetworkContext() | ||||||
|   const [streamState, setStreamState] = useState(StreamState.Unset) |   const [streamState, setStreamState] = useState(StreamState.Unset) | ||||||
| @ -301,7 +302,7 @@ export const Stream = () => { | |||||||
|           return |           return | ||||||
|         } |         } | ||||||
|         const path = getArtifactOfTypes( |         const path = getArtifactOfTypes( | ||||||
|           { key: entity_id, types: ['path', 'solid2D', 'segment'] }, |           { key: entity_id, types: ['path', 'solid2d', 'segment'] }, | ||||||
|           engineCommandManager.artifactGraph |           engineCommandManager.artifactGraph | ||||||
|         ) |         ) | ||||||
|         if (err(path)) { |         if (err(path)) { | ||||||
|  | |||||||
| @ -28,7 +28,7 @@ import { base64Decode } from 'lang/wasm' | |||||||
| import { sendTelemetry } from 'lib/textToCad' | import { sendTelemetry } from 'lib/textToCad' | ||||||
| import { Themes } from 'lib/theme' | import { Themes } from 'lib/theme' | ||||||
| import { ActionButton } from './ActionButton' | import { ActionButton } from './ActionButton' | ||||||
| import { commandBarMachine } from 'machines/commandBarMachine' | import { commandBarActor, commandBarMachine } from 'machines/commandBarMachine' | ||||||
| import { EventFrom } from 'xstate' | import { EventFrom } from 'xstate' | ||||||
| import { fileMachine } from 'machines/fileMachine' | import { fileMachine } from 'machines/fileMachine' | ||||||
| import { reportRejection } from 'lib/trap' | import { reportRejection } from 'lib/trap' | ||||||
| @ -43,15 +43,10 @@ export function ToastTextToCadError({ | |||||||
|   toastId, |   toastId, | ||||||
|   message, |   message, | ||||||
|   prompt, |   prompt, | ||||||
|   commandBarSend, |  | ||||||
| }: { | }: { | ||||||
|   toastId: string |   toastId: string | ||||||
|   message: string |   message: string | ||||||
|   prompt: string |   prompt: string | ||||||
|   commandBarSend: ( |  | ||||||
|     event: EventFrom<typeof commandBarMachine>, |  | ||||||
|     data?: unknown |  | ||||||
|   ) => void |  | ||||||
| }) { | }) { | ||||||
|   return ( |   return ( | ||||||
|     <div className="flex flex-col justify-between gap-6"> |     <div className="flex flex-col justify-between gap-6"> | ||||||
| @ -81,7 +76,7 @@ export function ToastTextToCadError({ | |||||||
|           }} |           }} | ||||||
|           name="Edit prompt" |           name="Edit prompt" | ||||||
|           onClick={() => { |           onClick={() => { | ||||||
|             commandBarSend({ |             commandBarActor.send({ | ||||||
|               type: 'Find and select command', |               type: 'Find and select command', | ||||||
|               data: { |               data: { | ||||||
|                 groupId: 'modeling', |                 groupId: 'modeling', | ||||||
|  | |||||||
| @ -1,6 +1,6 @@ | |||||||
| import { toolTips } from 'lang/langHelpers' | import { toolTips } from 'lang/langHelpers' | ||||||
| import { Selection, Selections } from 'lib/selections' | import { Selection, Selections } from 'lib/selections' | ||||||
| import { PathToNode, Program, Expr } from '../../lang/wasm' | import { PathToNode, Program, Expr, topLevelRange } from '../../lang/wasm' | ||||||
| import { getNodeFromPath } from '../../lang/queryAst' | import { getNodeFromPath } from '../../lang/queryAst' | ||||||
| import { | import { | ||||||
|   PathToNodeMap, |   PathToNodeMap, | ||||||
| @ -41,7 +41,7 @@ export function removeConstrainingValuesInfo({ | |||||||
|         graphSelections: nodes.map( |         graphSelections: nodes.map( | ||||||
|           (node): Selection => ({ |           (node): Selection => ({ | ||||||
|             codeRef: codeRefFromRange( |             codeRef: codeRefFromRange( | ||||||
|               [node.start, node.end, true], |               topLevelRange(node.start, node.end), | ||||||
|               kclManager.ast |               kclManager.ast | ||||||
|             ), |             ), | ||||||
|           }) |           }) | ||||||
|  | |||||||
| @ -8,7 +8,6 @@ import { | |||||||
| } from 'react-router-dom' | } from 'react-router-dom' | ||||||
| import { Models } from '@kittycad/lib' | import { Models } from '@kittycad/lib' | ||||||
| import { SettingsAuthProviderJest } from './SettingsAuthProvider' | import { SettingsAuthProviderJest } from './SettingsAuthProvider' | ||||||
| import { CommandBarProvider } from './CommandBar/CommandBarProvider' |  | ||||||
|  |  | ||||||
| type User = Models['User_type'] | type User = Models['User_type'] | ||||||
|  |  | ||||||
| @ -124,9 +123,7 @@ function TestWrap({ children }: { children: React.ReactNode }) { | |||||||
|       <Route |       <Route | ||||||
|         path="/file/:id" |         path="/file/:id" | ||||||
|         element={ |         element={ | ||||||
|           <CommandBarProvider> |           <SettingsAuthProviderJest>{children}</SettingsAuthProviderJest> | ||||||
|             <SettingsAuthProviderJest>{children}</SettingsAuthProviderJest> |  | ||||||
|           </CommandBarProvider> |  | ||||||
|         } |         } | ||||||
|       /> |       /> | ||||||
|     ), |     ), | ||||||
|  | |||||||
| @ -5,7 +5,6 @@ import { engineCommandManager, kclManager } from 'lib/singletons' | |||||||
| import { modelingMachine, ModelingMachineEvent } from 'machines/modelingMachine' | import { modelingMachine, ModelingMachineEvent } from 'machines/modelingMachine' | ||||||
| import { Selections, Selection, processCodeMirrorRanges } from 'lib/selections' | import { Selections, Selection, processCodeMirrorRanges } from 'lib/selections' | ||||||
| import { undo, redo } from '@codemirror/commands' | import { undo, redo } from '@codemirror/commands' | ||||||
| import { CommandBarMachineEvent } from 'machines/commandBarMachine' |  | ||||||
| import { addLineHighlight, addLineHighlightEvent } from './highlightextension' | import { addLineHighlight, addLineHighlightEvent } from './highlightextension' | ||||||
| import { | import { | ||||||
|   Diagnostic, |   Diagnostic, | ||||||
| @ -52,9 +51,6 @@ export default class EditorManager { | |||||||
|   private _modelingSend: (eventInfo: ModelingMachineEvent) => void = () => {} |   private _modelingSend: (eventInfo: ModelingMachineEvent) => void = () => {} | ||||||
|   private _modelingState: StateFrom<typeof modelingMachine> | null = null |   private _modelingState: StateFrom<typeof modelingMachine> | null = null | ||||||
|  |  | ||||||
|   private _commandBarSend: (eventInfo: CommandBarMachineEvent) => void = |  | ||||||
|     () => {} |  | ||||||
|  |  | ||||||
|   private _convertToVariableEnabled: boolean = false |   private _convertToVariableEnabled: boolean = false | ||||||
|   private _convertToVariableCallback: () => void = () => {} |   private _convertToVariableCallback: () => void = () => {} | ||||||
|  |  | ||||||
| @ -161,14 +157,6 @@ export default class EditorManager { | |||||||
|     this._modelingState = state |     this._modelingState = state | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   setCommandBarSend(send: (eventInfo: CommandBarMachineEvent) => void) { |  | ||||||
|     this._commandBarSend = send |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   commandBarSend(eventInfo: CommandBarMachineEvent): void { |  | ||||||
|     return this._commandBarSend(eventInfo) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   get highlightRange(): Array<[number, number]> { |   get highlightRange(): Array<[number, number]> { | ||||||
|     return this._highlightRange |     return this._highlightRange | ||||||
|   } |   } | ||||||
| @ -315,6 +303,21 @@ export default class EditorManager { | |||||||
|     if (selections?.graphSelections?.length === 0) { |     if (selections?.graphSelections?.length === 0) { | ||||||
|       return |       return | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     if (!this._editorView) { | ||||||
|  |       return | ||||||
|  |     } | ||||||
|  |     const codeBaseSelections = this.createEditorSelection(selections) | ||||||
|  |     this._editorView.dispatch({ | ||||||
|  |       selection: codeBaseSelections, | ||||||
|  |       annotations: [ | ||||||
|  |         updateOutsideEditorEvent, | ||||||
|  |         Transaction.addToHistory.of(false), | ||||||
|  |       ], | ||||||
|  |     }) | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   createEditorSelection(selections: Selections) { | ||||||
|     let codeBasedSelections = [] |     let codeBasedSelections = [] | ||||||
|     for (const selection of selections.graphSelections) { |     for (const selection of selections.graphSelections) { | ||||||
|       const safeEnd = Math.min( |       const safeEnd = Math.min( | ||||||
| @ -331,18 +334,7 @@ export default class EditorManager { | |||||||
|         .range[1] |         .range[1] | ||||||
|     const safeEnd = Math.min(end, this._editorView?.state.doc.length || end) |     const safeEnd = Math.min(end, this._editorView?.state.doc.length || end) | ||||||
|     codeBasedSelections.push(EditorSelection.cursor(safeEnd)) |     codeBasedSelections.push(EditorSelection.cursor(safeEnd)) | ||||||
|  |     return EditorSelection.create(codeBasedSelections, 1) | ||||||
|     if (!this._editorView) { |  | ||||||
|       return |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     this._editorView.dispatch({ |  | ||||||
|       selection: EditorSelection.create(codeBasedSelections, 1), |  | ||||||
|       annotations: [ |  | ||||||
|         updateOutsideEditorEvent, |  | ||||||
|         Transaction.addToHistory.of(false), |  | ||||||
|       ], |  | ||||||
|     }) |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // We will ONLY get here if the user called a select event. |   // We will ONLY get here if the user called a select event. | ||||||
|  | |||||||
| @ -1,10 +0,0 @@ | |||||||
| import { CommandsContext } from 'components/CommandBar/CommandBarProvider' |  | ||||||
|  |  | ||||||
| export const useCommandsContext = () => { |  | ||||||
|   const commandBarActor = CommandsContext.useActorRef() |  | ||||||
|   const commandBarState = CommandsContext.useSelector((state) => state) |  | ||||||
|   return { |  | ||||||
|     commandBarSend: commandBarActor.send, |  | ||||||
|     commandBarState, |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @ -1,7 +1,6 @@ | |||||||
| import { useEffect } from 'react' | import { useEffect } from 'react' | ||||||
| import { AnyStateMachine, Actor, StateFrom } from 'xstate' | import { AnyStateMachine, Actor, StateFrom, EventFrom } from 'xstate' | ||||||
| import { createMachineCommand } from '../lib/createMachineCommand' | import { createMachineCommand } from '../lib/createMachineCommand' | ||||||
| import { useCommandsContext } from './useCommandsContext' |  | ||||||
| import { modelingMachine } from 'machines/modelingMachine' | import { modelingMachine } from 'machines/modelingMachine' | ||||||
| import { authMachine } from 'machines/authMachine' | import { authMachine } from 'machines/authMachine' | ||||||
| import { settingsMachine } from 'machines/settingsMachine' | import { settingsMachine } from 'machines/settingsMachine' | ||||||
| @ -15,7 +14,7 @@ import { useKclContext } from 'lang/KclProvider' | |||||||
| import { useNetworkContext } from 'hooks/useNetworkContext' | import { useNetworkContext } from 'hooks/useNetworkContext' | ||||||
| import { NetworkHealthState } from 'hooks/useNetworkStatus' | import { NetworkHealthState } from 'hooks/useNetworkStatus' | ||||||
| import { useAppState } from 'AppState' | import { useAppState } from 'AppState' | ||||||
| import { getActorNextEvents } from 'lib/utils' | import { commandBarActor } from 'machines/commandBarMachine' | ||||||
|  |  | ||||||
| // This might not be necessary, AnyStateMachine from xstate is working | // This might not be necessary, AnyStateMachine from xstate is working | ||||||
| export type AllMachines = | export type AllMachines = | ||||||
| @ -49,7 +48,6 @@ export default function useStateMachineCommands< | |||||||
|   allCommandsRequireNetwork = false, |   allCommandsRequireNetwork = false, | ||||||
|   onCancel, |   onCancel, | ||||||
| }: UseStateMachineCommandsArgs<T, S>) { | }: UseStateMachineCommandsArgs<T, S>) { | ||||||
|   const { commandBarSend } = useCommandsContext() |  | ||||||
|   const { overallState } = useNetworkContext() |   const { overallState } = useNetworkContext() | ||||||
|   const { isExecuting } = useKclContext() |   const { isExecuting } = useKclContext() | ||||||
|   const { isStreamReady } = useAppState() |   const { isStreamReady } = useAppState() | ||||||
| @ -60,30 +58,33 @@ export default function useStateMachineCommands< | |||||||
|         overallState !== NetworkHealthState.Weak) || |         overallState !== NetworkHealthState.Weak) || | ||||||
|       isExecuting || |       isExecuting || | ||||||
|       !isStreamReady |       !isStreamReady | ||||||
|     const newCommands = getActorNextEvents(state) |     const newCommands = Object.keys(commandBarConfig || {}) | ||||||
|       .filter((_) => !allCommandsRequireNetwork || !disableAllButtons) |       .filter((_) => !allCommandsRequireNetwork || !disableAllButtons) | ||||||
|       .filter((e) => !['done.', 'error.'].some((n) => e.includes(n))) |       .flatMap((type) => { | ||||||
|       .flatMap((type) => |         const typeWithProperType = type as EventFrom<T>['type'] | ||||||
|         createMachineCommand<T, S>({ |         return createMachineCommand<T, S>({ | ||||||
|           // The group is the owner machine's ID. |           // The group is the owner machine's ID. | ||||||
|           groupId: machineId, |           groupId: machineId, | ||||||
|           type, |           type: typeWithProperType, | ||||||
|           state, |           state, | ||||||
|           send, |           send, | ||||||
|           actor, |           actor, | ||||||
|           commandBarConfig, |           commandBarConfig, | ||||||
|           onCancel, |           onCancel, | ||||||
|         }) |         }) | ||||||
|       ) |       }) | ||||||
|       .filter((c) => c !== null) as Command[] // TS isn't smart enough to know this filter removes nulls |       .filter((c) => c !== null) as Command[] // TS isn't smart enough to know this filter removes nulls | ||||||
|  |  | ||||||
|     commandBarSend({ type: 'Add commands', data: { commands: newCommands } }) |     commandBarActor.send({ | ||||||
|  |       type: 'Add commands', | ||||||
|  |       data: { commands: newCommands }, | ||||||
|  |     }) | ||||||
|  |  | ||||||
|     return () => { |     return () => { | ||||||
|       commandBarSend({ |       commandBarActor.send({ | ||||||
|         type: 'Remove commands', |         type: 'Remove commands', | ||||||
|         data: { commands: newCommands }, |         data: { commands: newCommands }, | ||||||
|       }) |       }) | ||||||
|     } |     } | ||||||
|   }, [state, overallState, isExecuting, isStreamReady]) |   }, [overallState, isExecuting, isStreamReady]) | ||||||
| } | } | ||||||
|  | |||||||
| @ -22,6 +22,7 @@ import { | |||||||
|   ProgramMemory, |   ProgramMemory, | ||||||
|   recast, |   recast, | ||||||
|   SourceRange, |   SourceRange, | ||||||
|  |   topLevelRange, | ||||||
| } from 'lang/wasm' | } from 'lang/wasm' | ||||||
| import { getNodeFromPath } from './queryAst' | import { getNodeFromPath } from './queryAst' | ||||||
| import { codeManager, editorManager, sceneInfra } from 'lib/singletons' | import { codeManager, editorManager, sceneInfra } from 'lib/singletons' | ||||||
| @ -376,11 +377,7 @@ export class KclManager { | |||||||
|     } |     } | ||||||
|     this.ast = { ...ast } |     this.ast = { ...ast } | ||||||
|     // updateArtifactGraph relies on updated executeState/programMemory |     // updateArtifactGraph relies on updated executeState/programMemory | ||||||
|     await this.engineCommandManager.updateArtifactGraph( |     this.engineCommandManager.updateArtifactGraph(execState.artifactGraph) | ||||||
|       this.ast, |  | ||||||
|       execState.artifactCommands, |  | ||||||
|       execState.artifacts |  | ||||||
|     ) |  | ||||||
|     this._executeCallback() |     this._executeCallback() | ||||||
|     if (!isInterrupted) { |     if (!isInterrupted) { | ||||||
|       sceneInfra.modelingSend({ type: 'code edit during sketch' }) |       sceneInfra.modelingSend({ type: 'code edit during sketch' }) | ||||||
| @ -473,7 +470,7 @@ export class KclManager { | |||||||
|           ...artifact, |           ...artifact, | ||||||
|           codeRef: { |           codeRef: { | ||||||
|             ...artifact.codeRef, |             ...artifact.codeRef, | ||||||
|             range: [node.start, node.end, true], |             range: topLevelRange(node.start, node.end), | ||||||
|           }, |           }, | ||||||
|         }) |         }) | ||||||
|       } |       } | ||||||
| @ -594,7 +591,7 @@ export class KclManager { | |||||||
|         if (start && end) { |         if (start && end) { | ||||||
|           returnVal.graphSelections.push({ |           returnVal.graphSelections.push({ | ||||||
|             codeRef: { |             codeRef: { | ||||||
|               range: [start, end, true], |               range: topLevelRange(start, end), | ||||||
|               pathToNode: path, |               pathToNode: path, | ||||||
|             }, |             }, | ||||||
|           }) |           }) | ||||||
|  | |||||||
| @ -24,7 +24,10 @@ describe('testing AST', () => { | |||||||
|             type: 'Literal', |             type: 'Literal', | ||||||
|             start: 0, |             start: 0, | ||||||
|             end: 1, |             end: 1, | ||||||
|             value: 5, |             value: { | ||||||
|  |               suffix: 'None', | ||||||
|  |               value: 5, | ||||||
|  |             }, | ||||||
|             raw: '5', |             raw: '5', | ||||||
|           }, |           }, | ||||||
|           operator: '+', |           operator: '+', | ||||||
| @ -32,7 +35,10 @@ describe('testing AST', () => { | |||||||
|             type: 'Literal', |             type: 'Literal', | ||||||
|             start: 3, |             start: 3, | ||||||
|             end: 4, |             end: 4, | ||||||
|             value: 6, |             value: { | ||||||
|  |               suffix: 'None', | ||||||
|  |               value: 6, | ||||||
|  |             }, | ||||||
|             raw: '6', |             raw: '6', | ||||||
|           }, |           }, | ||||||
|         }, |         }, | ||||||
|  | |||||||
