Compare commits
	
		
			25 Commits
		
	
	
		
			nightly-v2
			...
			pierremtb/
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 34272b872d | |||
| 668e2afb99 | |||
| 548c664db0 | |||
| d3a3f4410c | |||
| 22eb343171 | |||
| f2cfa4d5cf | |||
| 3f1f40eeba | |||
| ff2d161606 | |||
| 210c78029d | |||
| e27840219b | |||
| c943a3f192 | |||
| 6aa588f09f | |||
| 59a6333aad | |||
| 403f1507ae | |||
| eac7b83504 | |||
| 667500d1b9 | |||
| b15aac9f48 | |||
| 54153aa646 | |||
| 943cf21d34 | |||
| 5a6728c45a | |||
| ff2103d493 | |||
| 2dfa8f2176 | |||
| 29ed330326 | |||
| ca2cc825a6 | |||
| 83fe1b7ce0 | 
							
								
								
									
										6
									
								
								.github/workflows/build-apps.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -365,7 +365,7 @@ jobs: | ||||
|       - name: Set more complete nightly release notes | ||||
|         if: ${{ env.IS_NIGHTLY == 'true' }} | ||||
|         run: | | ||||
|           # Note: prefered going this way instead of a full clone in the checkout step, | ||||
|           # Note: preferred going this way instead of a full clone in the checkout step, | ||||
|           # see https://github.com/actions/checkout/issues/1471 | ||||
|           git fetch --prune --unshallow --tags | ||||
|           export TAG="nightly-${VERSION}" | ||||
| @ -394,6 +394,10 @@ jobs: | ||||
|           parent: false | ||||
|           destination: 'dl.kittycad.io/releases/modeling-app/nightly' | ||||
|  | ||||
|       - name: Invalidate bucket cache on latest*.yml and last_download.json files | ||||
|         if: ${{ env.IS_NIGHTLY == 'true' }} | ||||
|         run: yarn files:invalidate-bucket:nightly | ||||
|  | ||||
|       - name: Tag nightly commit | ||||
|         if: ${{ env.IS_NIGHTLY == 'true' }} | ||||
|         uses: actions/github-script@v7 | ||||
|  | ||||
							
								
								
									
										6
									
								
								.github/workflows/publish-apps-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -126,11 +126,7 @@ jobs: | ||||
|           destination: 'dl.kittycad.io/releases/modeling-app' | ||||
|  | ||||
|       - name: Invalidate bucket cache on latest*.yml and last_download.json files | ||||
|         run: | | ||||
|           gcloud compute url-maps invalidate-cdn-cache dl-url-map --path="/releases/modeling-app/last_download.json" --async | ||||
|           gcloud compute url-maps invalidate-cdn-cache dl-url-map --path="/releases/modeling-app/latest-linux-arm64.yml" --async | ||||
|           gcloud compute url-maps invalidate-cdn-cache dl-url-map --path="/releases/modeling-app/latest-mac.yml" --async | ||||
|           gcloud compute url-maps invalidate-cdn-cache dl-url-map --path="/releases/modeling-app/latest.yml" --async | ||||
|         run: yarn files:invalidate-bucket | ||||
|  | ||||
|       - name: Upload release files to Github | ||||
|         if: ${{ github.event_name == 'release' }} | ||||
|  | ||||
							
								
								
									
										43
									
								
								INSTALL.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,43 @@ | ||||
| # Setting Up Zoo Modeling App | ||||
|  | ||||
| Compared to other CAD software, getting Zoo Modeling App up and running is quick and straightforward across platforms. It's about 100MB to download and is quick to install. | ||||
|  | ||||
| ## Windows | ||||
|  | ||||
| 1. Download the [Zoo Modeling App installer](https://zoo.dev/modeling-app/download) for Windows and for your processor type. | ||||
|  | ||||
| 2. Once downloaded, run the installer `Zoo Modeling App-{version}-{arch}-win.exe` which should take a few seconds. | ||||
|  | ||||
| 3. The installation happens at `C:\Program Files\Zoo Modeling App`. A shortcut in the start menu is also created so you can run the app easily by clicking on it. | ||||
|  | ||||
| ## macOS | ||||
|  | ||||
| 1. Download the [Zoo Modeling App installer](https://zoo.dev/modeling-app/download) for macOS and for your processor type. | ||||
|  | ||||
| 2. Once downloaded, open the disk image `Zoo Modeling App-{version}-{arch}-mac.dmg` and drag the applications to your `Applications` directory. | ||||
|  | ||||
| 3. You can then open your `Applications` directory and double-click on `Zoo Modeling App` to open. | ||||
|  | ||||
|  | ||||
| ## Linux  | ||||
|  | ||||
| 1. Download the [Zoo Modeling App installer](https://zoo.dev/modeling-app/download) for Linux and for your processor type. | ||||
|  | ||||
| 2. Install the dependencies needed to run the [AppImage format](https://appimage.org/). | ||||
|     -  On Ubuntu, install the FUSE library with these commands in a terminal. | ||||
|        ```bash | ||||
|        sudo apt update | ||||
|        sudo apt install libfuse2 | ||||
|        ``` | ||||
|     - Optionally, follow [these steps](https://github.com/probonopd/go-appimage/blob/master/src/appimaged/README.md#initial-setup) to install `appimaged`. It is a daemon that makes interacting with AppImage files more seamless.  | ||||
|     - Once installed, copy the downloaded `Zoo Modeling App-{version}-{arch}-linux.AppImage` to the directory of your choice, for instance `~/Applications`. | ||||
|  | ||||
|    - `appimaged` should automatically find it and make it executable. If not, run: | ||||
|      ```bash | ||||
|      chmod a+x ~/Applications/Zoo\ Modeling\ App-{version}-{arch}-linux.AppImage | ||||
|      ``` | ||||
|  | ||||
| 3. You can double-click on the AppImage to run it, or in a terminal with this command: | ||||
|    ```bash | ||||
|     ~/Applications/Zoo\ Modeling\ App-{version}-{arch}-linux.AppImage | ||||
|    ``` | ||||
| @ -45,7 +45,7 @@ circles = map([1..3], drawCircle) | ||||
| ```js | ||||
| r = 10 // radius | ||||
| // Call `map`, using an anonymous function instead of a named one. | ||||
| circles = map([1..3], (id) { | ||||
| circles = map([1..3], fn(id) { | ||||
|   return startSketchOn("XY") | ||||
|     |> circle({ center = [id * 2 * r, 0], radius = r }, %) | ||||
| }) | ||||
|  | ||||
| @ -61,7 +61,7 @@ assertEqual(sum([1, 2, 3]), 6, 0.00001, "1 + 2 + 3 summed is 6") | ||||
| // an anonymous `add` function as its parameter, instead of declaring a | ||||
| // named function outside. | ||||
| arr = [1, 2, 3] | ||||
| sum = reduce(arr, 0, (i, result_so_far) { | ||||
| sum = reduce(arr, 0, fn(i, result_so_far) { | ||||
|   return i + result_so_far | ||||
| }) | ||||
|  | ||||
| @ -84,7 +84,7 @@ fn decagon(radius) { | ||||
|   // Use a `reduce` to draw the remaining decagon sides. | ||||
|   // For each number in the array 1..10, run the given function, | ||||
|   // which takes a partially-sketched decagon and adds one more edge to it. | ||||
|   fullDecagon = reduce([1..10], startOfDecagonSketch, (i, partialDecagon) { | ||||
|   fullDecagon = reduce([1..10], startOfDecagonSketch, fn(i, partialDecagon) { | ||||
|     // Draw one edge of the decagon. | ||||
|     x = cos(stepAngle * i) * radius | ||||
|     y = sin(stepAngle * i) * radius | ||||
|  | ||||
| @ -100436,7 +100436,7 @@ | ||||
|     "deprecated": false, | ||||
|     "examples": [ | ||||
|       "r = 10 // radius\nfn drawCircle(id) {\n  return startSketchOn(\"XY\")\n    |> circle({ center = [id * 2 * r, 0], radius = r }, %)\n}\n\n// Call `drawCircle`, passing in each element of the array.\n// The outputs from each `drawCircle` form a new array,\n// which is the return value from `map`.\ncircles = map([1..3], drawCircle)", | ||||
|       "r = 10 // radius\n// Call `map`, using an anonymous function instead of a named one.\ncircles = map([1..3], (id) {\n  return startSketchOn(\"XY\")\n    |> circle({ center = [id * 2 * r, 0], radius = r }, %)\n})" | ||||
|       "r = 10 // radius\n// Call `map`, using an anonymous function instead of a named one.\ncircles = map([1..3], fn(id) {\n  return startSketchOn(\"XY\")\n    |> circle({ center = [id * 2 * r, 0], radius = r }, %)\n})" | ||||
|     ] | ||||
|   }, | ||||
|   { | ||||
| @ -146129,8 +146129,8 @@ | ||||
|     "deprecated": false, | ||||
|     "examples": [ | ||||
|       "// This function adds two numbers.\nfn add(a, b) {\n  return a + b\n}\n\n// This function adds an array of numbers.\n// It uses the `reduce` function, to call the `add` function on every\n// element of the `arr` parameter. The starting value is 0.\nfn sum(arr) {\n  return reduce(arr, 0, add)\n}\n\n/* The above is basically like this pseudo-code:\nfn sum(arr):\n    let sumSoFar = 0\n    for i in arr:\n        sumSoFar = add(sumSoFar, i)\n    return sumSoFar */\n\n\n// We use `assertEqual` to check that our `sum` function gives the\n// expected result. It's good to check your work!\nassertEqual(sum([1, 2, 3]), 6, 0.00001, \"1 + 2 + 3 summed is 6\")", | ||||
|       "// This example works just like the previous example above, but it uses\n// an anonymous `add` function as its parameter, instead of declaring a\n// named function outside.\narr = [1, 2, 3]\nsum = reduce(arr, 0, (i, result_so_far) {\n  return i + result_so_far\n})\n\n// We use `assertEqual` to check that our `sum` function gives the\n// expected result. It's good to check your work!\nassertEqual(sum, 6, 0.00001, \"1 + 2 + 3 summed is 6\")", | ||||
|       "// Declare a function that sketches a decagon.\nfn decagon(radius) {\n  // Each side of the decagon is turned this many degrees from the previous angle.\n  stepAngle = 1 / 10 * tau()\n\n  // Start the decagon sketch at this point.\n  startOfDecagonSketch = startSketchAt([cos(0) * radius, sin(0) * radius])\n\n  // Use a `reduce` to draw the remaining decagon sides.\n  // For each number in the array 1..10, run the given function,\n  // which takes a partially-sketched decagon and adds one more edge to it.\n  fullDecagon = reduce([1..10], startOfDecagonSketch, (i, partialDecagon) {\n    // Draw one edge of the decagon.\n    x = cos(stepAngle * i) * radius\n    y = sin(stepAngle * i) * radius\n    return lineTo([x, y], partialDecagon)\n  })\n\n  return fullDecagon\n}\n\n/* The `decagon` above is basically like this pseudo-code:\nfn decagon(radius):\n    let stepAngle = (1/10) * tau()\n    let startOfDecagonSketch = startSketchAt([(cos(0)*radius), (sin(0) * radius)])\n\n    // Here's the reduce part.\n    let partialDecagon = startOfDecagonSketch\n    for i in [1..10]:\n        let x = cos(stepAngle * i) * radius\n        let y = sin(stepAngle * i) * radius\n        partialDecagon = lineTo([x, y], partialDecagon)\n    fullDecagon = partialDecagon // it's now full\n    return fullDecagon */\n\n\n// Use the `decagon` function declared above, to sketch a decagon with radius 5.\ndecagon(5.0)\n  |> close(%)" | ||||
|       "// This example works just like the previous example above, but it uses\n// an anonymous `add` function as its parameter, instead of declaring a\n// named function outside.\narr = [1, 2, 3]\nsum = reduce(arr, 0, fn(i, result_so_far) {\n  return i + result_so_far\n})\n\n// We use `assertEqual` to check that our `sum` function gives the\n// expected result. It's good to check your work!\nassertEqual(sum, 6, 0.00001, \"1 + 2 + 3 summed is 6\")", | ||||
|       "// Declare a function that sketches a decagon.\nfn decagon(radius) {\n  // Each side of the decagon is turned this many degrees from the previous angle.\n  stepAngle = 1 / 10 * tau()\n\n  // Start the decagon sketch at this point.\n  startOfDecagonSketch = startSketchAt([cos(0) * radius, sin(0) * radius])\n\n  // Use a `reduce` to draw the remaining decagon sides.\n  // For each number in the array 1..10, run the given function,\n  // which takes a partially-sketched decagon and adds one more edge to it.\n  fullDecagon = reduce([1..10], startOfDecagonSketch, fn(i, partialDecagon) {\n    // Draw one edge of the decagon.\n    x = cos(stepAngle * i) * radius\n    y = sin(stepAngle * i) * radius\n    return lineTo([x, y], partialDecagon)\n  })\n\n  return fullDecagon\n}\n\n/* The `decagon` above is basically like this pseudo-code:\nfn decagon(radius):\n    let stepAngle = (1/10) * tau()\n    let startOfDecagonSketch = startSketchAt([(cos(0)*radius), (sin(0) * radius)])\n\n    // Here's the reduce part.\n    let partialDecagon = startOfDecagonSketch\n    for i in [1..10]:\n        let x = cos(stepAngle * i) * radius\n        let y = sin(stepAngle * i) * radius\n        partialDecagon = lineTo([x, y], partialDecagon)\n    fullDecagon = partialDecagon // it's now full\n    return fullDecagon */\n\n\n// Use the `decagon` function declared above, to sketch a decagon with radius 5.\ndecagon(5.0)\n  |> close(%)" | ||||
|     ] | ||||
|   }, | ||||
|   { | ||||
|  | ||||
| @ -7,6 +7,7 @@ export class ToolbarFixture { | ||||
|  | ||||
|   extrudeButton!: Locator | ||||
|   loftButton!: Locator | ||||
|   shellButton!: Locator | ||||
|   offsetPlaneButton!: Locator | ||||
|   startSketchBtn!: Locator | ||||
|   lineBtn!: Locator | ||||
| @ -28,6 +29,7 @@ export class ToolbarFixture { | ||||
|     this.page = page | ||||
|     this.extrudeButton = page.getByTestId('extrude') | ||||
|     this.loftButton = page.getByTestId('loft') | ||||
|     this.shellButton = page.getByTestId('shell') | ||||
|     this.offsetPlaneButton = page.getByTestId('plane-offset') | ||||
|     this.startSketchBtn = page.getByTestId('sketch') | ||||
|     this.lineBtn = page.getByTestId('line') | ||||
|  | ||||
| @ -768,3 +768,168 @@ loftPointAndClickCases.forEach(({ shouldPreselect }) => { | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| const shellPointAndClickCapCases = [ | ||||
|   { shouldPreselect: true }, | ||||
|   { shouldPreselect: false }, | ||||
| ] | ||||
| shellPointAndClickCapCases.forEach(({ shouldPreselect }) => { | ||||
|   test(`Shell point-and-click cap (preselected sketches: ${shouldPreselect})`, async ({ | ||||
|     app, | ||||
|     scene, | ||||
|     editor, | ||||
|     toolbar, | ||||
|     cmdBar, | ||||
|   }) => { | ||||
|     const initialCode = `sketch001 = startSketchOn('XZ') | ||||
|     |> circle({ center = [0, 0], radius = 30 }, %) | ||||
|     extrude001 = extrude(30, sketch001) | ||||
|     ` | ||||
|     await app.initialise(initialCode) | ||||
|  | ||||
|     // One dumb hardcoded screen pixel value | ||||
|     const testPoint = { x: 575, y: 200 } | ||||
|     const [clickOnCap] = scene.makeMouseHelpers(testPoint.x, testPoint.y) | ||||
|     const shellDeclaration = | ||||
|       "shell001 = shell({ faces = ['end'], thickness = 5 }, extrude001)" | ||||
|  | ||||
|     await test.step(`Look for the grey of the shape`, async () => { | ||||
|       await scene.expectPixelColor([127, 127, 127], testPoint, 15) | ||||
|     }) | ||||
|  | ||||
|     if (!shouldPreselect) { | ||||
|       await test.step(`Go through the command bar flow without preselected faces`, async () => { | ||||
|         await toolbar.shellButton.click() | ||||
|         await cmdBar.expectState({ | ||||
|           stage: 'arguments', | ||||
|           currentArgKey: 'selection', | ||||
|           currentArgValue: '', | ||||
|           headerArguments: { | ||||
|             Selection: '', | ||||
|             Thickness: '', | ||||
|           }, | ||||
|           highlightedHeaderArg: 'selection', | ||||
|           commandName: 'Shell', | ||||
|         }) | ||||
|         await clickOnCap() | ||||
|         await app.page.waitForTimeout(500) | ||||
|         await cmdBar.progressCmdBar() | ||||
|         await cmdBar.progressCmdBar() | ||||
|         await cmdBar.expectState({ | ||||
|           stage: 'review', | ||||
|           headerArguments: { | ||||
|             Selection: '1 cap', | ||||
|             Thickness: '5', | ||||
|           }, | ||||
|           commandName: 'Shell', | ||||
|         }) | ||||
|         await cmdBar.progressCmdBar() | ||||
|       }) | ||||
|     } else { | ||||
|       await test.step(`Preselect the cap`, async () => { | ||||
|         await clickOnCap() | ||||
|         await app.page.waitForTimeout(500) | ||||
|       }) | ||||
|  | ||||
|       await test.step(`Go through the command bar flow with a preselected face (cap)`, async () => { | ||||
|         await toolbar.shellButton.click() | ||||
|         await cmdBar.progressCmdBar() | ||||
|         await cmdBar.progressCmdBar() | ||||
|         await cmdBar.expectState({ | ||||
|           stage: 'review', | ||||
|           headerArguments: { | ||||
|             Selection: '1 cap', | ||||
|             Thickness: '5', | ||||
|           }, | ||||
|           commandName: 'Shell', | ||||
|         }) | ||||
|         await cmdBar.progressCmdBar() | ||||
|       }) | ||||
|     } | ||||
|  | ||||
|     await test.step(`Confirm code is added to the editor, scene has changed`, async () => { | ||||
|       await editor.expectEditor.toContain(shellDeclaration) | ||||
|       await editor.expectState({ | ||||
|         diagnostics: [], | ||||
|         activeLines: [shellDeclaration], | ||||
|         highlightedCode: '', | ||||
|       }) | ||||
|       await scene.expectPixelColor([146, 146, 146], testPoint, 15) | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| test('Shell point-and-click wall', async ({ | ||||
|   app, | ||||
|   page, | ||||
|   scene, | ||||
|   editor, | ||||
|   toolbar, | ||||
|   cmdBar, | ||||
| }) => { | ||||
|   const initialCode = `sketch001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-20, 20], %) | ||||
|   |> xLine(40, %) | ||||
|   |> yLine(-60, %) | ||||
|   |> xLine(-40, %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
| extrude001 = extrude(40, sketch001) | ||||
|   ` | ||||
|   await app.initialise(initialCode) | ||||
|  | ||||
|   // One dumb hardcoded screen pixel value | ||||
|   const testPoint = { x: 580, y: 180 } | ||||
|   const [clickOnCap] = scene.makeMouseHelpers(testPoint.x, testPoint.y) | ||||
|   const [clickOnWall] = scene.makeMouseHelpers(testPoint.x, testPoint.y + 70) | ||||
|   const mutatedCode = 'xLine(-40, %, $seg01)' | ||||
|   const shellDeclaration = | ||||
|     "shell001 = shell({  faces = ['end', seg01],  thickness = 5}, extrude001)" | ||||
|   const formattedOutLastLine = '}, extrude001)' | ||||
|  | ||||
|   await test.step(`Look for the grey of the shape`, async () => { | ||||
|     await scene.expectPixelColor([99, 99, 99], testPoint, 15) | ||||
|   }) | ||||
|  | ||||
|   await test.step(`Go through the command bar flow, selecting a wall and keeping default thickness`, async () => { | ||||
|     await toolbar.shellButton.click() | ||||
|     await cmdBar.expectState({ | ||||
|       stage: 'arguments', | ||||
|       currentArgKey: 'selection', | ||||
|       currentArgValue: '', | ||||
|       headerArguments: { | ||||
|         Selection: '', | ||||
|         Thickness: '', | ||||
|       }, | ||||
|       highlightedHeaderArg: 'selection', | ||||
|       commandName: 'Shell', | ||||
|     }) | ||||
|     await clickOnCap() | ||||
|     await page.keyboard.down('Shift') | ||||
|     await clickOnWall() | ||||
|     await app.page.waitForTimeout(500) | ||||
|     await page.keyboard.up('Shift') | ||||
|     await cmdBar.progressCmdBar() | ||||
|     await cmdBar.progressCmdBar() | ||||
|     await cmdBar.expectState({ | ||||
|       stage: 'review', | ||||
|       headerArguments: { | ||||
|         Selection: '1 cap, 1 face', | ||||
|         Thickness: '5', | ||||
|       }, | ||||
|       commandName: 'Shell', | ||||
|     }) | ||||
|     await cmdBar.progressCmdBar() | ||||
|   }) | ||||
|  | ||||
|   await test.step(`Confirm code is added to the editor, scene has changed`, async () => { | ||||
|     await editor.expectEditor.toContain(mutatedCode) | ||||
|     await editor.expectEditor.toContain(shellDeclaration) | ||||
|     await editor.expectState({ | ||||
|       diagnostics: [], | ||||
|       activeLines: [formattedOutLastLine], | ||||
|       highlightedCode: '', | ||||
|     }) | ||||
|     await scene.expectPixelColor([49, 49, 49], testPoint, 15) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| Before Width: | Height: | Size: 37 KiB After Width: | Height: | Size: 37 KiB | 
| Before Width: | Height: | Size: 40 KiB After Width: | Height: | Size: 40 KiB | 
| Before Width: | Height: | Size: 38 KiB After Width: | Height: | Size: 38 KiB | 
| Before Width: | Height: | Size: 42 KiB After Width: | Height: | Size: 42 KiB | 
| Before Width: | Height: | Size: 36 KiB After Width: | Height: | Size: 36 KiB | 
| @ -26,7 +26,17 @@ test.describe('Testing constraints', () => { | ||||
|     }) | ||||
|  | ||||
|     const u = await getUtils(page) | ||||
|     const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|     // constants and locators | ||||
|     const lengthValue = { | ||||
|       old: '20', | ||||
|       new: '25', | ||||
|     } | ||||
|     const cmdBarKclInput = page | ||||
|       .getByTestId('cmd-bar-arg-value') | ||||
|       .getByRole('textbox') | ||||
|     const cmdBarSubmitButton = page.getByRole('button', { | ||||
|       name: 'arrow right Continue', | ||||
|     }) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
| @ -36,26 +46,26 @@ test.describe('Testing constraints', () => { | ||||
|     await u.closeDebugPanel() | ||||
|  | ||||
|     // Click the line of code for line. | ||||
|     await page.getByText(`line([0, 20], %)`).click() // TODO remove this and reinstate // await topHorzSegmentClick() | ||||
|     // TODO remove this and reinstate `await topHorzSegmentClick()` | ||||
|     await page.getByText(`line([0, ${lengthValue.old}], %)`).click() | ||||
|     await page.waitForTimeout(100) | ||||
|  | ||||
|     // enter sketch again | ||||
|     await page.getByRole('button', { name: 'Edit Sketch' }).click() | ||||
|     await page.waitForTimeout(500) // wait for animation | ||||
|  | ||||
|     const startXPx = 500 | ||||
|     await page.mouse.move(startXPx + PUR * 15, 250 - PUR * 10) | ||||
|     await page.keyboard.down('Shift') | ||||
|     await page.mouse.click(834, 244) | ||||
|     await page.keyboard.up('Shift') | ||||
|  | ||||
|     await page | ||||
|       .getByRole('button', { name: 'dimension Length', exact: true }) | ||||
|       .click() | ||||
|     await page.getByText('Add constraining value').click() | ||||
|     await expect(cmdBarKclInput).toHaveText('20') | ||||
|     await cmdBarKclInput.fill(lengthValue.new) | ||||
|     await expect( | ||||
|       page.getByText(`Can't calculate`), | ||||
|       `Something went wrong with the KCL expression evaluation` | ||||
|     ).not.toBeVisible() | ||||
|     await cmdBarSubmitButton.click() | ||||
|  | ||||
|     await expect(page.locator('.cm-content')).toHaveText( | ||||
|       `length001 = 20sketch001 = startSketchOn('XY')  |> startProfileAt([-10, -10], %)  |> line([20, 0], %)  |> angledLine([90, length001], %)  |> xLine(-20, %)` | ||||
|       `length001 = ${lengthValue.new}sketch001 = startSketchOn('XY')  |> startProfileAt([-10, -10], %)  |> line([20, 0], %)  |> angledLine([90, length001], %)  |> xLine(-20, %)` | ||||
|     ) | ||||
|  | ||||
|     // Make sure we didn't pop out of sketch mode. | ||||
| @ -66,7 +76,6 @@ test.describe('Testing constraints', () => { | ||||
|     await page.waitForTimeout(500) // wait for animation | ||||
|  | ||||
|     // Exit sketch | ||||
|     await page.mouse.move(startXPx + PUR * 15, 250 - PUR * 10) | ||||
|     await page.keyboard.press('Escape') | ||||
|     await expect( | ||||
|       page.getByRole('button', { name: 'Exit Sketch' }) | ||||
| @ -524,7 +533,7 @@ part002 = startSketchOn('XZ') | ||||
|       }) | ||||
|     } | ||||
|   }) | ||||
|   test.describe('Test Angle/Length constraint single selection', () => { | ||||
|   test.describe('Test Angle constraint single selection', () => { | ||||
|     const cases = [ | ||||
|       { | ||||
|         testName: 'Angle - Add variable', | ||||
| @ -538,18 +547,6 @@ part002 = startSketchOn('XZ') | ||||
|         constraint: 'angle', | ||||
|         value: '83, 78.33', | ||||
|       }, | ||||
|       { | ||||
|         testName: 'Length - Add variable', | ||||
|         addVariable: true, | ||||
|         constraint: 'length', | ||||
|         value: '83, length001', | ||||
|       }, | ||||
|       { | ||||
|         testName: 'Length - No variable', | ||||
|         addVariable: false, | ||||
|         constraint: 'length', | ||||
|         value: '83, 78.33', | ||||
|       }, | ||||
|     ] as const | ||||
|     for (const { testName, addVariable, value, constraint } of cases) { | ||||
|       test(`${testName}`, async ({ page }) => { | ||||
| @ -608,6 +605,90 @@ part002 = startSketchOn('XZ') | ||||
|       }) | ||||
|     } | ||||
|   }) | ||||
|   test.describe('Test Length constraint single selection', () => { | ||||
|     const cases = [ | ||||
|       { | ||||
|         testName: 'Length - Add variable', | ||||
|         addVariable: true, | ||||
|         constraint: 'length', | ||||
|         value: '83, length001', | ||||
|       }, | ||||
|       { | ||||
|         testName: 'Length - No variable', | ||||
|         addVariable: false, | ||||
|         constraint: 'length', | ||||
|         value: '83, 78.33', | ||||
|       }, | ||||
|     ] as const | ||||
|     for (const { testName, addVariable, value, constraint } of cases) { | ||||
|       test(`${testName}`, async ({ page }) => { | ||||
|         // constants and locators | ||||
|         const cmdBarKclInput = page | ||||
|           .getByTestId('cmd-bar-arg-value') | ||||
|           .getByRole('textbox') | ||||
|         const cmdBarKclVariableNameInput = | ||||
|           page.getByPlaceholder('Variable name') | ||||
|         const cmdBarSubmitButton = page.getByRole('button', { | ||||
|           name: 'arrow right Continue', | ||||
|         }) | ||||
|  | ||||
|         await page.addInitScript(async () => { | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `yo = 5 | ||||
| part001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([-7.54, -26.74], %) | ||||
|   |> line([74.36, 130.4], %) | ||||
|   |> line([78.92, -120.11], %) | ||||
|   |> line([9.16, 77.79], %) | ||||
|   |> line([51.19, 48.97], %) | ||||
| part002 = startSketchOn('XZ') | ||||
|   |> startProfileAt([299.05, 231.45], %) | ||||
|   |> xLine(-425.34, %, $seg_what) | ||||
|   |> yLine(-264.06, %) | ||||
|   |> xLine(segLen(seg_what), %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|           ) | ||||
|         }) | ||||
|         const u = await getUtils(page) | ||||
|         await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|  | ||||
|         await u.waitForAuthSkipAppStart() | ||||
|  | ||||
|         await page.getByText('line([74.36, 130.4], %)').click() | ||||
|         await page.getByRole('button', { name: 'Edit Sketch' }).click() | ||||
|  | ||||
|         const line3 = await u.getSegmentBodyCoords( | ||||
|           `[data-overlay-index="${2}"]` | ||||
|         ) | ||||
|  | ||||
|         await page.mouse.click(line3.x, line3.y) | ||||
|         await page | ||||
|           .getByRole('button', { | ||||
|             name: 'Length: open menu', | ||||
|           }) | ||||
|           .click() | ||||
|         await page.getByTestId('dropdown-constraint-' + constraint).click() | ||||
|  | ||||
|         if (!addVariable) { | ||||
|           await test.step(`Clear the variable input`, async () => { | ||||
|             await cmdBarKclVariableNameInput.clear() | ||||
|             await cmdBarKclVariableNameInput.press('Backspace') | ||||
|           }) | ||||
|         } | ||||
|         await expect(cmdBarKclInput).toHaveText('78.33') | ||||
|         await cmdBarSubmitButton.click() | ||||
|  | ||||
|         const changedCode = `|> angledLine([${value}], %)` | ||||
|         await expect(page.locator('.cm-content')).toContainText(changedCode) | ||||
|         // checking active assures the cursor is where it should be | ||||
|         await expect(page.locator('.cm-activeLine')).toHaveText(changedCode) | ||||
|  | ||||
|         // checking the count of the overlays is a good proxy check that the client sketch scene is in a good state | ||||
|         await expect(page.getByTestId('segment-overlay')).toHaveCount(4) | ||||
|       }) | ||||
|     } | ||||
|   }) | ||||
|   test.describe('Many segments - no modal constraints', () => { | ||||
|     const cases = [ | ||||
|       { | ||||
| @ -868,6 +949,15 @@ part002 = startSketchOn('XZ') | ||||
|   |> line([3.13, -2.4], %)` | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     // constants and locators | ||||
|     const cmdBarKclInput = page | ||||
|       .getByTestId('cmd-bar-arg-value') | ||||
|       .getByRole('textbox') | ||||
|     const cmdBarSubmitButton = page.getByRole('button', { | ||||
|       name: 'arrow right Continue', | ||||
|     }) | ||||
|  | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|  | ||||
| @ -928,8 +1018,8 @@ part002 = startSketchOn('XZ') | ||||
|     // await page.getByRole('button', { name: 'length', exact: true }).click() | ||||
|     await page.getByTestId('dropdown-constraint-length').click() | ||||
|  | ||||
|     await page.getByLabel('length Value').fill('10') | ||||
|     await page.getByRole('button', { name: 'Add constraining value' }).click() | ||||
|     await cmdBarKclInput.fill('10') | ||||
|     await cmdBarSubmitButton.click() | ||||
|  | ||||
|     activeLinesContent = await page.locator('.cm-activeLine').all() | ||||
|     await expect(activeLinesContent[0]).toHaveText(`|> xLine(length001, %)`) | ||||
|  | ||||
| @ -91,7 +91,14 @@ test.describe('Testing segment overlays', () => { | ||||
|           await page.getByTestId('constraint-symbol-popover').count() | ||||
|         ).toBeGreaterThan(0) | ||||
|         await unconstrainedLocator.click() | ||||
|         await page.getByText('Add variable').click() | ||||
|         await expect( | ||||
|           page.getByTestId('cmd-bar-arg-value').getByRole('textbox') | ||||
|         ).toBeFocused() | ||||
|         await page | ||||
|           .getByRole('button', { | ||||
|             name: 'arrow right Continue', | ||||
|           }) | ||||
|           .click() | ||||
|         await expect(page.locator('.cm-content')).toContainText(expectFinal) | ||||
|       } | ||||
|  | ||||
| @ -151,7 +158,14 @@ test.describe('Testing segment overlays', () => { | ||||
|           await page.getByTestId('constraint-symbol-popover').count() | ||||
|         ).toBeGreaterThan(0) | ||||
|         await unconstrainedLocator.click() | ||||
|         await page.getByText('Add variable').click() | ||||
|         await expect( | ||||
|           page.getByTestId('cmd-bar-arg-value').getByRole('textbox') | ||||
|         ).toBeFocused() | ||||
|         await page | ||||
|           .getByRole('button', { | ||||
|             name: 'arrow right Continue', | ||||
|           }) | ||||
|           .click() | ||||
|         await expect(page.locator('.cm-content')).toContainText( | ||||
|           expectAfterUnconstrained | ||||
|         ) | ||||
|  | ||||
| @ -81,6 +81,7 @@ | ||||
|     "simpleserver": "yarn pretest && http-server ./public --cors -p 3000", | ||||
|     "simpleserver:ci": "yarn pretest && http-server ./public --cors -p 3000 &", | ||||
|     "simpleserver:bg": "yarn pretest && http-server ./public --cors -p 3000 &", | ||||
|     "simpleserver:stop": "kill-port 3000", | ||||
|     "fmt": "prettier --write ./src *.ts *.json *.js ./e2e ./packages", | ||||
|     "fmt-check": "prettier --check ./src *.ts *.json *.js ./e2e ./packages", | ||||
|     "fetch:wasm": "./get-latest-wasm-bundle.sh", | ||||
| @ -95,6 +96,8 @@ | ||||
|     "files:set-version": "echo \"$(jq --arg v \"$VERSION\" '.version=$v' package.json --indent 2)\" > package.json", | ||||
|     "files:set-notes": "./scripts/set-files-notes.sh", | ||||
|     "files:flip-to-nightly": "./scripts/flip-files-to-nightly.sh", | ||||
|     "files:invalidate-bucket": "./scripts/invalidate-files-bucket.sh", | ||||
|     "files:invalidate-bucket:nightly": "./scripts/invalidate-files-bucket.sh --nightly", | ||||
|     "postinstall": "yarn fetch:samples && yarn xstate:typegen && ./node_modules/.bin/electron-rebuild", | ||||
|     "xstate:typegen": "yarn xstate typegen \"src/**/*.ts?(x)\"", | ||||
|     "make:dev": "make dev", | ||||
|  | ||||
							
								
								
									
										11
									
								
								scripts/invalidate-files-bucket.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @ -0,0 +1,11 @@ | ||||
| #!/bin/bash | ||||
| base_dir="/releases/modeling-app" | ||||
| if [[ $1 = "--nightly" ]]; then | ||||
|     base_dir="/releases/modeling-app/nightly" | ||||
| fi | ||||
|  | ||||
| echo "Invalidating json and yml files at $base_dir in the download bucket" | ||||
| gcloud compute url-maps invalidate-cdn-cache dl-url-map --path="$base_dir/last_download.json" --async | ||||
| gcloud compute url-maps invalidate-cdn-cache dl-url-map --path="$base_dir/latest-linux-arm64.yml" --async | ||||
| gcloud compute url-maps invalidate-cdn-cache dl-url-map --path="$base_dir/latest-mac.yml" --async | ||||
| gcloud compute url-maps invalidate-cdn-cache dl-url-map --path="$base_dir/latest.yml" --async | ||||
| @ -200,7 +200,10 @@ function CoreDump() { | ||||
|     () => new CoreDumpManager(engineCommandManager, codeManager, token), | ||||
|     [] | ||||
|   ) | ||||
|   useHotkeyWrapper(['mod + shift + .'], () => { | ||||
|   // TODO: revisit once progress is made on upstream issue | ||||
|   // https://github.com/JohannesKlauss/react-hotkeys-hook/issues/1064 | ||||
|   // const hotkey = process.platform !== 'linux' ? 'mod + shift + .' : 'mod + shift + >' | ||||
|   useHotkeyWrapper(['mod + shift + .', 'mod + shift + >'], () => { | ||||
|     toast | ||||
|       .promise( | ||||
|         coreDump(coreDumpManager, true), | ||||
|  | ||||
| @ -505,7 +505,8 @@ const ConstraintSymbol = ({ | ||||
|   constrainInfo: ConstrainInfo | ||||
|   verticalPosition: 'top' | 'bottom' | ||||
| }) => { | ||||
|   const { context, send } = useModelingContext() | ||||
|   const { commandBarSend } = useCommandsContext() | ||||
|   const { context } = useModelingContext() | ||||
|   const varNameMap: { | ||||
|     [key in ConstrainInfo['type']]: { | ||||
|       varName: string | ||||
| @ -624,11 +625,18 @@ const ConstraintSymbol = ({ | ||||
|         // disabled={implicitDesc} TODO why does this change styles that are hard to override? | ||||
|         onClick={toSync(async () => { | ||||
|           if (!isConstrained) { | ||||
|             send({ | ||||
|               type: 'Convert to variable', | ||||
|             commandBarSend({ | ||||
|               type: 'Find and select command', | ||||
|               data: { | ||||
|                 name: 'Constrain with named value', | ||||
|                 groupId: 'modeling', | ||||
|                 argDefaultValues: { | ||||
|                   currentValue: { | ||||
|                     pathToNode, | ||||
|                     variableName: varName, | ||||
|                     valueText: value, | ||||
|                   }, | ||||
|                 }, | ||||
|               }, | ||||
|             }) | ||||
|           } else if (isConstrained) { | ||||
|  | ||||
| @ -8,11 +8,16 @@ import { getSystemTheme } from 'lib/theme' | ||||
| import { useCalculateKclExpression } from 'lib/useCalculateKclExpression' | ||||
| import { roundOff } from 'lib/utils' | ||||
| import { varMentions } from 'lib/varCompletionExtension' | ||||
| import { useEffect, useRef, useState } from 'react' | ||||
| import { useEffect, useMemo, useRef, useState } from 'react' | ||||
| import { useHotkeys } from 'react-hotkeys-hook' | ||||
| import styles from './CommandBarKclInput.module.css' | ||||
| import { createIdentifier, createVariableDeclaration } from 'lang/modifyAst' | ||||
| import { useCodeMirror } from 'components/ModelingSidebar/ModelingPanes/CodeEditor' | ||||
| import { useSelector } from '@xstate/react' | ||||
|  | ||||
| const machineContextSelector = (snapshot?: { | ||||
|   context: Record<string, unknown> | ||||
| }) => snapshot?.context | ||||
|  | ||||
| function CommandBarKclInput({ | ||||
|   arg, | ||||
| @ -31,12 +36,44 @@ function CommandBarKclInput({ | ||||
|     arg.name | ||||
|   ] as KclCommandValue | undefined | ||||
|   const { settings } = useSettingsAuthContext() | ||||
|   const defaultValue = (arg.defaultValue as string) || '' | ||||
|   const argMachineContext = useSelector( | ||||
|     arg.machineActor, | ||||
|     machineContextSelector | ||||
|   ) | ||||
|   const defaultValue = useMemo( | ||||
|     () => | ||||
|       arg.defaultValue | ||||
|         ? arg.defaultValue instanceof Function | ||||
|           ? arg.defaultValue(commandBarState.context, argMachineContext) | ||||
|           : arg.defaultValue | ||||
|         : '', | ||||
|     [arg.defaultValue, commandBarState.context, argMachineContext] | ||||
|   ) | ||||
|   const initialVariableName = useMemo(() => { | ||||
|     // Use the configured variable name if it exists | ||||
|     if (arg.variableName !== undefined) { | ||||
|       return arg.variableName instanceof Function | ||||
|         ? arg.variableName(commandBarState.context, argMachineContext) | ||||
|         : arg.variableName | ||||
|     } | ||||
|     // or derive it from the previously set value or the argument name | ||||
|     return previouslySetValue && 'variableName' in previouslySetValue | ||||
|       ? previouslySetValue.variableName | ||||
|       : arg.name | ||||
|   }, [ | ||||
|     arg.variableName, | ||||
|     commandBarState.context, | ||||
|     argMachineContext, | ||||
|     arg.name, | ||||
|     previouslySetValue, | ||||
|   ]) | ||||
|   const [value, setValue] = useState( | ||||
|     previouslySetValue?.valueText || defaultValue || '' | ||||
|   ) | ||||
|   const [createNewVariable, setCreateNewVariable] = useState( | ||||
|     previouslySetValue && 'variableName' in previouslySetValue | ||||
|     (previouslySetValue && 'variableName' in previouslySetValue) || | ||||
|       arg.createVariableByDefault || | ||||
|       false | ||||
|   ) | ||||
|   const [canSubmit, setCanSubmit] = useState(true) | ||||
|   useHotkeys('mod + k, mod + /', () => commandBarSend({ type: 'Close' })) | ||||
| @ -52,10 +89,7 @@ function CommandBarKclInput({ | ||||
|     isNewVariableNameUnique, | ||||
|   } = useCalculateKclExpression({ | ||||
|     value, | ||||
|     initialVariableName: | ||||
|       previouslySetValue && 'variableName' in previouslySetValue | ||||
|         ? previouslySetValue.variableName | ||||
|         : arg.name, | ||||
|     initialVariableName, | ||||
|   }) | ||||
|   const varMentionData: Completion[] = prevVariables.map((v) => ({ | ||||
|     label: v.key, | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| import { APP_VERSION } from 'routes/Settings' | ||||
| import { APP_VERSION, RELEASE_URL } from 'routes/Settings' | ||||
| import { CustomIcon } from 'components/CustomIcon' | ||||
| import Tooltip from 'components/Tooltip' | ||||
| import { PATHS } from 'lib/paths' | ||||
| @ -72,10 +72,8 @@ export function LowerRightControls({ | ||||
|       <menu className="flex items-center justify-end gap-3 pointer-events-auto"> | ||||
|         {!location.pathname.startsWith(PATHS.HOME) && <ModelStateIndicator />} | ||||
|         <a | ||||
|           onClick={openExternalBrowserIfDesktop( | ||||
|             `https://github.com/KittyCAD/modeling-app/releases/tag/v${APP_VERSION}` | ||||
|           )} | ||||
|           href={`https://github.com/KittyCAD/modeling-app/releases/tag/v${APP_VERSION}`} | ||||
|           onClick={openExternalBrowserIfDesktop(RELEASE_URL)} | ||||
|           href={RELEASE_URL} | ||||
|           target="_blank" | ||||
|           rel="noopener noreferrer" | ||||
|           className={'!no-underline font-mono text-xs ' + linkOverrideClassName} | ||||
|  | ||||
| @ -41,7 +41,10 @@ import { | ||||
|   angleBetweenInfo, | ||||
|   applyConstraintAngleBetween, | ||||
| } from './Toolbar/SetAngleBetween' | ||||
| import { applyConstraintAngleLength } from './Toolbar/setAngleLength' | ||||
| import { | ||||
|   applyConstraintAngleLength, | ||||
|   applyConstraintLength, | ||||
| } from './Toolbar/setAngleLength' | ||||
| import { | ||||
|   canSweepSelection, | ||||
|   handleSelectionBatch, | ||||
| @ -51,6 +54,8 @@ import { | ||||
|   Selections, | ||||
|   updateSelections, | ||||
|   canLoftSelection, | ||||
|   canRevolveSelection, | ||||
|   canShellSelection, | ||||
| } from 'lib/selections' | ||||
| import { applyConstraintIntersect } from './Toolbar/Intersect' | ||||
| import { applyConstraintAbsDistance } from './Toolbar/SetAbsDistance' | ||||
| @ -62,13 +67,15 @@ import { | ||||
|   getSketchOrientationDetails, | ||||
| } from 'clientSideScene/sceneEntities' | ||||
| import { | ||||
|   moveValueIntoNewVariablePath, | ||||
|   insertNamedConstant, | ||||
|   replaceValueAtNodePath, | ||||
|   sketchOnExtrudedFace, | ||||
|   sketchOnOffsetPlane, | ||||
|   startSketchOnDefault, | ||||
| } from 'lang/modifyAst' | ||||
| import { Program, parse, recast, resultIsOk } from 'lang/wasm' | ||||
| import { PathToNode, Program, parse, recast, resultIsOk } from 'lang/wasm' | ||||
| import { | ||||
|   doesSceneHaveExtrudedSketch, | ||||
|   doesSceneHaveSweepableSketch, | ||||
|   getNodePathFromSourceRange, | ||||
|   isSingleCursorInPipe, | ||||
| @ -79,7 +86,6 @@ import toast from 'react-hot-toast' | ||||
| import { EditorSelection, Transaction } from '@codemirror/state' | ||||
| import { useLoaderData, useNavigate, useSearchParams } from 'react-router-dom' | ||||
| import { letEngineAnimateAndSyncCamAfter } from 'clientSideScene/CameraControls' | ||||
| import { getVarNameModal } from 'hooks/useToolbarGuards' | ||||
| import { err, reportRejection, trap } from 'lib/trap' | ||||
| import { useCommandsContext } from 'hooks/useCommandsContext' | ||||
| import { modelingMachineEvent } from 'editor/manager' | ||||
| @ -570,6 +576,26 @@ export const ModelingMachineProvider = ({ | ||||
|           if (err(canSweep)) return false | ||||
|           return canSweep | ||||
|         }, | ||||
|         'has valid revolve selection': ({ context: { selectionRanges } }) => { | ||||
|           // A user can begin extruding if they either have 1+ faces selected or nothing selected | ||||
|           // TODO: I believe this guard only allows for extruding a single face at a time | ||||
|           const hasNoSelection = | ||||
|             selectionRanges.graphSelections.length === 0 || | ||||
|             isRangeBetweenCharacters(selectionRanges) || | ||||
|             isSelectionLastLine(selectionRanges, codeManager.code) | ||||
|  | ||||
|           if (hasNoSelection) { | ||||
|             // they have no selection, we should enable the button | ||||
|             // so they can select the face through the cmdbar | ||||
|             // BUT only if there's extrudable geometry | ||||
|             return doesSceneHaveSweepableSketch(kclManager.ast) | ||||
|           } | ||||
|           if (!isSketchPipe(selectionRanges)) return false | ||||
|  | ||||
|           const canSweep = canRevolveSelection(selectionRanges) | ||||
|           if (err(canSweep)) return false | ||||
|           return canSweep | ||||
|         }, | ||||
|         'has valid loft selection': ({ context: { selectionRanges } }) => { | ||||
|           const hasNoSelection = | ||||
|             selectionRanges.graphSelections.length === 0 || | ||||
| @ -585,6 +611,24 @@ export const ModelingMachineProvider = ({ | ||||
|           if (err(canLoft)) return false | ||||
|           return canLoft | ||||
|         }, | ||||
|         'has valid shell selection': ({ | ||||
|           context: { selectionRanges }, | ||||
|           event, | ||||
|         }) => { | ||||
|           const hasNoSelection = | ||||
|             selectionRanges.graphSelections.length === 0 || | ||||
|             isRangeBetweenCharacters(selectionRanges) || | ||||
|             isSelectionLastLine(selectionRanges, codeManager.code) | ||||
|  | ||||
|           if (hasNoSelection) { | ||||
|             return doesSceneHaveExtrudedSketch(kclManager.ast) | ||||
|           } | ||||
|  | ||||
|           const canShell = canShellSelection(selectionRanges) | ||||
|           console.log('canShellSelection', canShellSelection(selectionRanges)) | ||||
|           if (err(canShell)) return false | ||||
|           return canShell | ||||
|         }, | ||||
|         'has valid selection for deletion': ({ | ||||
|           context: { selectionRanges }, | ||||
|         }) => { | ||||
| @ -869,12 +913,18 @@ export const ModelingMachineProvider = ({ | ||||
|             } | ||||
|           } | ||||
|         ), | ||||
|         'Get length info': fromPromise( | ||||
|           async ({ input: { selectionRanges, sketchDetails } }) => { | ||||
|             const { modifiedAst, pathToNodeMap } = | ||||
|               await applyConstraintAngleLength({ | ||||
|         astConstrainLength: fromPromise( | ||||
|           async ({ | ||||
|             input: { selectionRanges, sketchDetails, lengthValue }, | ||||
|           }) => { | ||||
|             if (!lengthValue) | ||||
|               return Promise.reject(new Error('No length value')) | ||||
|             const constraintResult = await applyConstraintLength({ | ||||
|               selectionRanges, | ||||
|               length: lengthValue, | ||||
|             }) | ||||
|             if (err(constraintResult)) return Promise.reject(constraintResult) | ||||
|             const { modifiedAst, pathToNodeMap } = constraintResult | ||||
|             const pResult = parse(recast(modifiedAst)) | ||||
|             if (trap(pResult) || !resultIsOk(pResult)) | ||||
|               return Promise.reject(new Error('Unexpected compilation error')) | ||||
| @ -1043,38 +1093,88 @@ export const ModelingMachineProvider = ({ | ||||
|             } | ||||
|           } | ||||
|         ), | ||||
|         'Get convert to variable info': fromPromise( | ||||
|         'Apply named value constraint': fromPromise( | ||||
|           async ({ input: { selectionRanges, sketchDetails, data } }) => { | ||||
|             if (!sketchDetails) | ||||
|             if (!sketchDetails) { | ||||
|               return Promise.reject(new Error('No sketch details')) | ||||
|             const { variableName } = await getVarNameModal({ | ||||
|               valueName: data?.variableName || 'var', | ||||
|             }) | ||||
|             } | ||||
|             if (!data) { | ||||
|               return Promise.reject(new Error('No data from command flow')) | ||||
|             } | ||||
|             let pResult = parse(recast(kclManager.ast)) | ||||
|             if (trap(pResult) || !resultIsOk(pResult)) | ||||
|               return Promise.reject(new Error('Unexpected compilation error')) | ||||
|             let parsed = pResult.program | ||||
|  | ||||
|             const { modifiedAst: _modifiedAst, pathToReplacedNode } = | ||||
|               moveValueIntoNewVariablePath( | ||||
|                 parsed, | ||||
|                 kclManager.programMemory, | ||||
|                 data?.pathToNode || [], | ||||
|                 variableName | ||||
|             let result: { | ||||
|               modifiedAst: Node<Program> | ||||
|               pathToReplaced: PathToNode | null | ||||
|             } = { | ||||
|               modifiedAst: parsed, | ||||
|               pathToReplaced: null, | ||||
|             } | ||||
|             // If the user provided a constant name, | ||||
|             // we need to insert the named constant | ||||
|             // and then replace the node with the constant's name. | ||||
|             if ('variableName' in data.namedValue) { | ||||
|               const astAfterReplacement = replaceValueAtNodePath({ | ||||
|                 ast: parsed, | ||||
|                 pathToNode: data.currentValue.pathToNode, | ||||
|                 newExpressionString: data.namedValue.variableName, | ||||
|               }) | ||||
|               if (trap(astAfterReplacement)) { | ||||
|                 return Promise.reject(astAfterReplacement) | ||||
|               } | ||||
|               const parseResultAfterInsertion = parse( | ||||
|                 recast( | ||||
|                   insertNamedConstant({ | ||||
|                     node: astAfterReplacement.modifiedAst, | ||||
|                     newExpression: data.namedValue, | ||||
|                   }) | ||||
|                 ) | ||||
|             pResult = parse(recast(_modifiedAst)) | ||||
|               ) | ||||
|               if ( | ||||
|                 trap(parseResultAfterInsertion) || | ||||
|                 !resultIsOk(parseResultAfterInsertion) | ||||
|               ) | ||||
|                 return Promise.reject(parseResultAfterInsertion) | ||||
|               result = { | ||||
|                 modifiedAst: parseResultAfterInsertion.program, | ||||
|                 pathToReplaced: astAfterReplacement.pathToReplaced, | ||||
|               } | ||||
|             } else if ('valueText' in data.namedValue) { | ||||
|               // If they didn't provide a constant name, | ||||
|               // just replace the node with the value. | ||||
|               const astAfterReplacement = replaceValueAtNodePath({ | ||||
|                 ast: parsed, | ||||
|                 pathToNode: data.currentValue.pathToNode, | ||||
|                 newExpressionString: data.namedValue.valueText, | ||||
|               }) | ||||
|               if (trap(astAfterReplacement)) { | ||||
|                 return Promise.reject(astAfterReplacement) | ||||
|               } | ||||
|               // The `replacer` function returns a pathToNode that assumes | ||||
|               // an identifier is also being inserted into the AST, creating an off-by-one error. | ||||
|               // This corrects that error, but TODO we should fix this upstream | ||||
|               // to avoid this kind of error in the future. | ||||
|               astAfterReplacement.pathToReplaced[1][0] = | ||||
|                 (astAfterReplacement.pathToReplaced[1][0] as number) - 1 | ||||
|               result = astAfterReplacement | ||||
|             } | ||||
|  | ||||
|             pResult = parse(recast(result.modifiedAst)) | ||||
|             if (trap(pResult) || !resultIsOk(pResult)) | ||||
|               return Promise.reject(new Error('Unexpected compilation error')) | ||||
|             parsed = pResult.program | ||||
|  | ||||
|             if (trap(parsed)) return Promise.reject(parsed) | ||||
|             parsed = parsed as Node<Program> | ||||
|             if (!pathToReplacedNode) | ||||
|             if (!result.pathToReplaced) | ||||
|               return Promise.reject(new Error('No path to replaced node')) | ||||
|  | ||||
|             const updatedAst = | ||||
|               await sceneEntitiesManager.updateAstAndRejigSketch( | ||||
|                 pathToReplacedNode || [], | ||||
|                 result.pathToReplaced || [], | ||||
|                 parsed, | ||||
|                 sketchDetails.zAxis, | ||||
|                 sketchDetails.yAxis, | ||||
| @ -1087,7 +1187,7 @@ export const ModelingMachineProvider = ({ | ||||
|             ) | ||||
|  | ||||
|             const selection = updateSelections( | ||||
|               { 0: pathToReplacedNode }, | ||||
|               { 0: result.pathToReplaced }, | ||||
|               selectionRanges, | ||||
|               updatedAst.newAst | ||||
|             ) | ||||
| @ -1095,7 +1195,7 @@ export const ModelingMachineProvider = ({ | ||||
|             return { | ||||
|               selectionType: 'completeSelection', | ||||
|               selection, | ||||
|               updatedPathToNode: pathToReplacedNode, | ||||
|               updatedPathToNode: result.pathToReplaced, | ||||
|             } | ||||
|           } | ||||
|         ), | ||||
|  | ||||
| @ -76,7 +76,7 @@ export const ModelingPane = ({ | ||||
|   return ( | ||||
|     <section | ||||
|       {...props} | ||||
|       title={title && typeof title === 'string' ? title : ''} | ||||
|       aria-label={title && typeof title === 'string' ? title : ''} | ||||
|       data-testid={detailsTestId} | ||||
|       id={id} | ||||
|       className={ | ||||
|  | ||||
| @ -10,7 +10,7 @@ interface AllKeybindingsFieldsProps {} | ||||
|  | ||||
| export const AllKeybindingsFields = forwardRef( | ||||
|   ( | ||||
|     props: AllKeybindingsFieldsProps, | ||||
|     _props: AllKeybindingsFieldsProps, | ||||
|     scrollRef: ForwardedRef<HTMLDivElement> | ||||
|   ) => { | ||||
|     // This is how we will get the interaction map from the context | ||||
| @ -25,7 +25,7 @@ export const AllKeybindingsFields = forwardRef( | ||||
|             .map(([category, categoryItems]) => ( | ||||
|               <div className="flex flex-col gap-4 px-2 pr-4"> | ||||
|                 <h2 | ||||
|                   id={`category-${category}`} | ||||
|                   id={`category-${category.replaceAll(/\s/g, '-')}`} | ||||
|                   className="text-xl mt-6 first-of-type:mt-0 capitalize font-bold" | ||||
|                 > | ||||
|                   {category} | ||||
|  | ||||
| @ -13,7 +13,7 @@ import { isDesktop } from 'lib/isDesktop' | ||||
| import { ActionButton } from 'components/ActionButton' | ||||
| import { SettingsFieldInput } from './SettingsFieldInput' | ||||
| import toast from 'react-hot-toast' | ||||
| import { APP_VERSION, PACKAGE_NAME } from 'routes/Settings' | ||||
| import { APP_VERSION, IS_NIGHTLY, RELEASE_URL } from 'routes/Settings' | ||||
| import { PATHS } from 'lib/paths' | ||||
| import { | ||||
|   createAndOpenNewTutorialProject, | ||||
| @ -246,10 +246,8 @@ export const AllSettingsFields = forwardRef( | ||||
|                   to inject the version from package.json */} | ||||
|               App version {APP_VERSION}.{' '} | ||||
|               <a | ||||
|                 onClick={openExternalBrowserIfDesktop( | ||||
|                   `https://github.com/KittyCAD/modeling-app/releases/tag/v${APP_VERSION}` | ||||
|                 )} | ||||
|                 href={`https://github.com/KittyCAD/modeling-app/releases/tag/v${APP_VERSION}`} | ||||
|                 onClick={openExternalBrowserIfDesktop(RELEASE_URL)} | ||||
|                 href={RELEASE_URL} | ||||
|                 target="_blank" | ||||
|                 rel="noopener noreferrer" | ||||
|               > | ||||
| @ -271,7 +269,7 @@ export const AllSettingsFields = forwardRef( | ||||
|               , and start a discussion if you don't see it! Your feedback will | ||||
|               help us prioritize what to build next. | ||||
|             </p> | ||||
|             {PACKAGE_NAME.indexOf('-nightly') === -1 && ( | ||||
|             {!IS_NIGHTLY && ( | ||||
|               <p className="max-w-2xl mt-6"> | ||||
|                 Want to experience the latest and (hopefully) greatest from our | ||||
|                 main development branch?{' '} | ||||
|  | ||||
| @ -19,7 +19,7 @@ export function KeybindingsSectionsList({ | ||||
|             key={category} | ||||
|             onClick={() => | ||||
|               scrollRef.current | ||||
|                 ?.querySelector(`#category-${category}`) | ||||
|                 ?.querySelector(`#category-${category.replaceAll(/\s/g, '-')}`) | ||||
|                 ?.scrollIntoView({ | ||||
|                   block: 'center', | ||||
|                   behavior: 'smooth', | ||||
|  | ||||
| @ -22,6 +22,7 @@ import { removeDoubleNegatives } from '../AvailableVarsHelpers' | ||||
| import { normaliseAngle } from '../../lib/utils' | ||||
| import { kclManager } from 'lib/singletons' | ||||
| import { err } from 'lib/trap' | ||||
| import { KclCommandValue } from 'lib/commandTypes' | ||||
|  | ||||
| const getModalInfo = createSetAngleLengthModal(SetAngleLengthModal) | ||||
|  | ||||
| @ -63,6 +64,57 @@ export function angleLengthInfo({ | ||||
|   return { enabled, transforms } | ||||
| } | ||||
|  | ||||
| export async function applyConstraintLength({ | ||||
|   length, | ||||
|   selectionRanges, | ||||
| }: { | ||||
|   length: KclCommandValue | ||||
|   selectionRanges: Selections | ||||
| }) { | ||||
|   const ast = kclManager.ast | ||||
|   const angleLength = angleLengthInfo({ selectionRanges }) | ||||
|   if (err(angleLength)) return angleLength | ||||
|   const { transforms } = angleLength | ||||
|  | ||||
|   let distanceExpression: Expr = length.valueAst | ||||
|  | ||||
|   /** | ||||
|    * To be "constrained", the value must be a binary expression, a named value, or a function call. | ||||
|    * If it has a variable name, we need to insert a variable declaration at the correct index. | ||||
|    */ | ||||
|   if ( | ||||
|     'variableName' in length && | ||||
|     length.variableName && | ||||
|     length.insertIndex !== undefined | ||||
|   ) { | ||||
|     const newBody = [...ast.body] | ||||
|     newBody.splice(length.insertIndex, 0, length.variableDeclarationAst) | ||||
|     ast.body = newBody | ||||
|     distanceExpression = createIdentifier(length.variableName) | ||||
|   } | ||||
|  | ||||
|   if (!isExprBinaryPart(distanceExpression)) { | ||||
|     return new Error('Invalid valueNode, is not a BinaryPart') | ||||
|   } | ||||
|  | ||||
|   const retval = transformAstSketchLines({ | ||||
|     ast, | ||||
|     selectionRanges, | ||||
|     transformInfos: transforms, | ||||
|     programMemory: kclManager.programMemory, | ||||
|     referenceSegName: '', | ||||
|     forceValueUsedInTransform: distanceExpression, | ||||
|   }) | ||||
|   if (err(retval)) return Promise.reject(retval) | ||||
|  | ||||
|   const { modifiedAst: _modifiedAst, pathToNodeMap } = retval | ||||
|  | ||||
|   return { | ||||
|     modifiedAst: _modifiedAst, | ||||
|     pathToNodeMap, | ||||
|   } | ||||
| } | ||||
|  | ||||
| export async function applyConstraintAngleLength({ | ||||
|   selectionRanges, | ||||
|   angleOrLength = 'setLength', | ||||
|  | ||||
| @ -24,6 +24,8 @@ export function useConvertToVariable(range?: SourceRange) { | ||||
|   }, [enable]) | ||||
|  | ||||
|   useEffect(() => { | ||||
|     // Return early if there are no selection ranges for whatever reason | ||||
|     if (!context.selectionRanges) return | ||||
|     const parsed = ast | ||||
|  | ||||
|     const meta = isNodeSafeToReplace( | ||||
|  | ||||
| @ -45,6 +45,7 @@ import { TagDeclarator } from 'wasm-lib/kcl/bindings/TagDeclarator' | ||||
| import { Models } from '@kittycad/lib' | ||||
| import { ExtrudeFacePlane } from 'machines/modelingMachine' | ||||
| import { Node } from 'wasm-lib/kcl/bindings/Node' | ||||
| import { KclExpressionWithVariable } from 'lib/commandTypes' | ||||
|  | ||||
| export function startSketchOnDefault( | ||||
|   node: Node<Program>, | ||||
| @ -590,6 +591,25 @@ export function addOffsetPlane({ | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Return a modified clone of an AST with a named constant inserted into the body | ||||
|  */ | ||||
| export function insertNamedConstant({ | ||||
|   node, | ||||
|   newExpression, | ||||
| }: { | ||||
|   node: Node<Program> | ||||
|   newExpression: KclExpressionWithVariable | ||||
| }): Node<Program> { | ||||
|   const ast = structuredClone(node) | ||||
|   ast.body.splice( | ||||
|     newExpression.insertIndex, | ||||
|     0, | ||||
|     newExpression.variableDeclarationAst | ||||
|   ) | ||||
|   return ast | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Modify the AST to create a new sketch using the variable declaration | ||||
|  * of an offset plane. The new sketch just has to come after the offset | ||||
| @ -933,6 +953,31 @@ export function giveSketchFnCallTag( | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Replace a | ||||
|  */ | ||||
| export function replaceValueAtNodePath({ | ||||
|   ast, | ||||
|   pathToNode, | ||||
|   newExpressionString, | ||||
| }: { | ||||
|   ast: Node<Program> | ||||
|   pathToNode: PathToNode | ||||
|   newExpressionString: string | ||||
| }) { | ||||
|   const replaceCheckResult = isNodeSafeToReplacePath(ast, pathToNode) | ||||
|   if (err(replaceCheckResult)) { | ||||
|     return replaceCheckResult | ||||
|   } | ||||
|   const { isSafe, value, replacer } = replaceCheckResult | ||||
|  | ||||
|   if (!isSafe || value.type === 'Identifier') { | ||||
|     return new Error('Not safe to replace') | ||||
|   } | ||||
|  | ||||
|   return replacer(ast, newExpressionString) | ||||
| } | ||||
|  | ||||
| export function moveValueIntoNewVariablePath( | ||||
|   ast: Node<Program>, | ||||
|   programMemory: ProgramMemory, | ||||
|  | ||||
| @ -22,7 +22,7 @@ import { | ||||
| import { getNodeFromPath, getNodePathFromSourceRange } from '../queryAst' | ||||
| import { createLiteral } from 'lang/modifyAst' | ||||
| import { err } from 'lib/trap' | ||||
| import { Selections } from 'lib/selections' | ||||
| import { Selection, Selections } from 'lib/selections' | ||||
| import { engineCommandManager, kclManager } from 'lib/singletons' | ||||
| import { VITE_KC_DEV_TOKEN } from 'env' | ||||
| import { isOverlap } from 'lib/utils' | ||||
| @ -118,13 +118,8 @@ const runGetPathToExtrudeForSegmentSelectionTest = async ( | ||||
|     code.indexOf(selectedSegmentSnippet) + selectedSegmentSnippet.length, | ||||
|     true, | ||||
|   ] | ||||
|   const selection: Selections = { | ||||
|     graphSelections: [ | ||||
|       { | ||||
|   const selection: Selection = { | ||||
|     codeRef: codeRefFromRange(segmentRange, ast), | ||||
|       }, | ||||
|     ], | ||||
|     otherSelections: [], | ||||
|   } | ||||
|  | ||||
|   // executeAst and artifactGraph | ||||
|  | ||||
| @ -29,7 +29,7 @@ import { | ||||
|   sketchLineHelperMap, | ||||
| } from '../std/sketch' | ||||
| import { err, trap } from 'lib/trap' | ||||
| import { Selections } from 'lib/selections' | ||||
| import { Selection, Selections } from 'lib/selections' | ||||
| import { KclCommandValue } from 'lib/commandTypes' | ||||
| import { | ||||
|   Artifact, | ||||
| @ -99,14 +99,9 @@ export function modifyAstWithEdgeTreatmentAndTag( | ||||
|   const lookupMap: Map<string, PathToNode> = new Map() // work around for Map key comparison | ||||
|  | ||||
|   for (const selection of selections.graphSelections) { | ||||
|     const singleSelection = { | ||||
|       graphSelections: [selection], | ||||
|       otherSelections: [], | ||||
|     } | ||||
|  | ||||
|     const result = getPathToExtrudeForSegmentSelection( | ||||
|       clonedAstForGetExtrude, | ||||
|       singleSelection, | ||||
|       selection, | ||||
|       artifactGraph | ||||
|     ) | ||||
|     if (err(result)) return result | ||||
| @ -259,12 +254,12 @@ function insertParametersIntoAst( | ||||
|  | ||||
| export function getPathToExtrudeForSegmentSelection( | ||||
|   ast: Program, | ||||
|   selection: Selections, | ||||
|   selection: Selection, | ||||
|   artifactGraph: ArtifactGraph | ||||
| ): { pathToSegmentNode: PathToNode; pathToExtrudeNode: PathToNode } | Error { | ||||
|   const pathToSegmentNode = getNodePathFromSourceRange( | ||||
|     ast, | ||||
|     selection.graphSelections[0]?.codeRef?.range | ||||
|     selection.codeRef?.range | ||||
|   ) | ||||
|  | ||||
|   const varDecNode = getNodeFromPath<VariableDeclaration>( | ||||
| @ -308,7 +303,7 @@ async function updateAstAndFocus( | ||||
|   } | ||||
| } | ||||
|  | ||||
| function mutateAstWithTagForSketchSegment( | ||||
| export function mutateAstWithTagForSketchSegment( | ||||
|   astClone: Node<Program>, | ||||
|   pathToSegmentNode: PathToNode | ||||
| ): { modifiedAst: Program; tag: string } | Error { | ||||
| @ -340,7 +335,7 @@ function mutateAstWithTagForSketchSegment( | ||||
|   return { modifiedAst: astClone, tag } | ||||
| } | ||||
|  | ||||
| function getEdgeTagCall( | ||||
| export function getEdgeTagCall( | ||||
|   tag: string, | ||||
|   artifact: Artifact | ||||
| ): Node<Identifier | CallExpression> { | ||||
|  | ||||
							
								
								
									
										154
									
								
								src/lang/modifyAst/addRevolve.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,154 @@ | ||||
| import { err } from 'lib/trap' | ||||
| import { KCL_DEFAULT_CONSTANT_PREFIXES } from 'lib/constants' | ||||
| import { | ||||
|   Program, | ||||
|   PathToNode, | ||||
|   Expr, | ||||
|   CallExpression, | ||||
|   PipeExpression, | ||||
|   VariableDeclarator, | ||||
| } from 'lang/wasm' | ||||
| import { Selections } from 'lib/selections' | ||||
| import { Node } from 'wasm-lib/kcl/bindings/Node' | ||||
| import { | ||||
|   createLiteral, | ||||
|   createCallExpressionStdLib, | ||||
|   createObjectExpression, | ||||
|   createIdentifier, | ||||
|   createPipeExpression, | ||||
|   findUniqueName, | ||||
|   createVariableDeclaration, | ||||
| } from 'lang/modifyAst' | ||||
| import { getNodeFromPath, getNodePathFromSourceRange } from 'lang/queryAst' | ||||
| import { | ||||
|   mutateAstWithTagForSketchSegment, | ||||
|   getEdgeTagCall, | ||||
| } from 'lang/modifyAst/addEdgeTreatment' | ||||
| export function revolveSketch( | ||||
|   ast: Node<Program>, | ||||
|   pathToSketchNode: PathToNode, | ||||
|   shouldPipe = false, | ||||
|   angle: Expr = createLiteral(4), | ||||
|   axis: Selections | ||||
| ): | ||||
|   | { | ||||
|       modifiedAst: Node<Program> | ||||
|       pathToSketchNode: PathToNode | ||||
|       pathToRevolveArg: PathToNode | ||||
|     } | ||||
|   | Error { | ||||
|   const clonedAst = structuredClone(ast) | ||||
|   const sketchNode = getNodeFromPath(clonedAst, pathToSketchNode) | ||||
|   if (err(sketchNode)) return sketchNode | ||||
|  | ||||
|   // testing code | ||||
|   const pathToAxisSelection = getNodePathFromSourceRange( | ||||
|     clonedAst, | ||||
|     axis.graphSelections[0]?.codeRef.range | ||||
|   ) | ||||
|  | ||||
|   const lineNode = getNodeFromPath<CallExpression>( | ||||
|     clonedAst, | ||||
|     pathToAxisSelection, | ||||
|     'CallExpression' | ||||
|   ) | ||||
|   if (err(lineNode)) return lineNode | ||||
|  | ||||
|   // TODO Kevin: What if |> close(%)? | ||||
|   // TODO Kevin: What if opposite edge | ||||
|   // TODO Kevin: What if the edge isn't planar to the sketch? | ||||
|   // TODO Kevin: add a tag. | ||||
|   const tagResult = mutateAstWithTagForSketchSegment( | ||||
|     clonedAst, | ||||
|     pathToAxisSelection | ||||
|   ) | ||||
|  | ||||
|   // Have the tag whether it is already created or a new one is generated | ||||
|   if (err(tagResult)) return tagResult | ||||
|   const { tag } = tagResult | ||||
|  | ||||
|   /* Original Code */ | ||||
|   const { node: sketchExpression } = sketchNode | ||||
|  | ||||
|   // determine if sketchExpression is in a pipeExpression or not | ||||
|   const sketchPipeExpressionNode = getNodeFromPath<PipeExpression>( | ||||
|     clonedAst, | ||||
|     pathToSketchNode, | ||||
|     'PipeExpression' | ||||
|   ) | ||||
|   if (err(sketchPipeExpressionNode)) return sketchPipeExpressionNode | ||||
|   const { node: sketchPipeExpression } = sketchPipeExpressionNode | ||||
|   const isInPipeExpression = sketchPipeExpression.type === 'PipeExpression' | ||||
|  | ||||
|   const sketchVariableDeclaratorNode = getNodeFromPath<VariableDeclarator>( | ||||
|     clonedAst, | ||||
|     pathToSketchNode, | ||||
|     'VariableDeclarator' | ||||
|   ) | ||||
|   if (err(sketchVariableDeclaratorNode)) return sketchVariableDeclaratorNode | ||||
|   const { | ||||
|     node: sketchVariableDeclarator, | ||||
|     shallowPath: sketchPathToDecleration, | ||||
|   } = sketchVariableDeclaratorNode | ||||
|  | ||||
|   const axisSelection = axis?.graphSelections[0]?.artifact | ||||
|  | ||||
|   if (!axisSelection) return new Error('Axis selection is missing.') | ||||
|  | ||||
|   const revolveCall = createCallExpressionStdLib('revolve', [ | ||||
|     createObjectExpression({ | ||||
|       angle: angle, | ||||
|       axis: getEdgeTagCall(tag, axisSelection), | ||||
|     }), | ||||
|     createIdentifier(sketchVariableDeclarator.id.name), | ||||
|   ]) | ||||
|  | ||||
|   if (shouldPipe) { | ||||
|     const pipeChain = createPipeExpression( | ||||
|       isInPipeExpression | ||||
|         ? [...sketchPipeExpression.body, revolveCall] | ||||
|         : [sketchExpression as any, revolveCall] | ||||
|     ) | ||||
|  | ||||
|     sketchVariableDeclarator.init = pipeChain | ||||
|     const pathToRevolveArg: PathToNode = [ | ||||
|       ...sketchPathToDecleration, | ||||
|       ['init', 'VariableDeclarator'], | ||||
|       ['body', ''], | ||||
|       [pipeChain.body.length - 1, 'index'], | ||||
|       ['arguments', 'CallExpression'], | ||||
|       [0, 'index'], | ||||
|     ] | ||||
|  | ||||
|     return { | ||||
|       modifiedAst: clonedAst, | ||||
|       pathToSketchNode, | ||||
|       pathToRevolveArg, | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // We're not creating a pipe expression, | ||||
|   // but rather a separate constant for the extrusion | ||||
|   const name = findUniqueName(clonedAst, KCL_DEFAULT_CONSTANT_PREFIXES.REVOLVE) | ||||
|   const VariableDeclaration = createVariableDeclaration(name, revolveCall) | ||||
|   const sketchIndexInPathToNode = | ||||
|     sketchPathToDecleration.findIndex((a) => a[0] === 'body') + 1 | ||||
|   const sketchIndexInBody = sketchPathToDecleration[sketchIndexInPathToNode][0] | ||||
|   if (typeof sketchIndexInBody !== 'number') | ||||
|     return new Error('expected sketchIndexInBody to be a number') | ||||
|   clonedAst.body.splice(sketchIndexInBody + 1, 0, VariableDeclaration) | ||||
|  | ||||
|   const pathToRevolveArg: PathToNode = [ | ||||
|     ['body', ''], | ||||
|     [sketchIndexInBody + 1, 'index'], | ||||
|     ['declaration', 'VariableDeclaration'], | ||||
|     ['init', 'VariableDeclarator'], | ||||
|     ['arguments', 'CallExpression'], | ||||
|     [0, 'index'], | ||||
|   ] | ||||
|   return { | ||||
|     modifiedAst: clonedAst, | ||||
|     pathToSketchNode: [...pathToSketchNode.slice(0, -1), [-1, 'index']], | ||||
|     pathToRevolveArg, | ||||
|   } | ||||
| } | ||||
							
								
								
									
										123
									
								
								src/lang/modifyAst/addShell.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,123 @@ | ||||
| import { ArtifactGraph } from 'lang/std/artifactGraph' | ||||
| import { Selections } from 'lib/selections' | ||||
| import { Expr } from 'wasm-lib/kcl/bindings/Expr' | ||||
| import { Program } from 'wasm-lib/kcl/bindings/Program' | ||||
| import { Node } from 'wasm-lib/kcl/bindings/Node' | ||||
| import { PathToNode, VariableDeclarator } from 'lang/wasm' | ||||
| import { | ||||
|   getPathToExtrudeForSegmentSelection, | ||||
|   mutateAstWithTagForSketchSegment, | ||||
| } from './addEdgeTreatment' | ||||
| import { getNodeFromPath } from 'lang/queryAst' | ||||
| import { err } from 'lib/trap' | ||||
| import { | ||||
|   createLiteral, | ||||
|   createIdentifier, | ||||
|   findUniqueName, | ||||
|   createCallExpressionStdLib, | ||||
|   createObjectExpression, | ||||
|   createArrayExpression, | ||||
|   createVariableDeclaration, | ||||
| } from 'lang/modifyAst' | ||||
| import { KCL_DEFAULT_CONSTANT_PREFIXES } from 'lib/constants' | ||||
|  | ||||
| export function addShell({ | ||||
|   node, | ||||
|   selection, | ||||
|   artifactGraph, | ||||
|   thickness, | ||||
| }: { | ||||
|   node: Node<Program> | ||||
|   selection: Selections | ||||
|   artifactGraph: ArtifactGraph | ||||
|   thickness: Expr | ||||
| }): Error | { modifiedAst: Node<Program>; pathToNode: PathToNode } { | ||||
|   const modifiedAst = structuredClone(node) | ||||
|  | ||||
|   // Look up the corresponding extrude | ||||
|   const clonedAstForGetExtrude = structuredClone(modifiedAst) | ||||
|  | ||||
|   const expressions: Expr[] = [] | ||||
|   let pathToExtrudeNode: PathToNode | undefined = undefined | ||||
|   for (const graphSelection of selection.graphSelections) { | ||||
|     const extrudeLookupResult = getPathToExtrudeForSegmentSelection( | ||||
|       clonedAstForGetExtrude, | ||||
|       graphSelection, | ||||
|       artifactGraph | ||||
|     ) | ||||
|     if (err(extrudeLookupResult)) { | ||||
|       return new Error("Couldn't find extrude") | ||||
|     } | ||||
|  | ||||
|     pathToExtrudeNode = extrudeLookupResult.pathToExtrudeNode | ||||
|     // Get the sketch ref from the selection | ||||
|     // TODO: this assumes the segment is piped directly from the sketch, with no intermediate `VariableDeclarator` between. | ||||
|     // We must find a technique for these situations that is robust to intermediate declarations | ||||
|     const sketchNode = getNodeFromPath<VariableDeclarator>( | ||||
|       modifiedAst, | ||||
|       graphSelection.codeRef.pathToNode, | ||||
|       'VariableDeclarator' | ||||
|     ) | ||||
|     if (err(sketchNode)) { | ||||
|       return sketchNode | ||||
|     } | ||||
|  | ||||
|     const selectedArtifact = graphSelection.artifact | ||||
|     if (!selectedArtifact) { | ||||
|       return new Error('Bad artifact') | ||||
|     } | ||||
|  | ||||
|     // Check on the selection, and handle the wall vs cap casees | ||||
|     let expr: Expr | ||||
|     if (selectedArtifact.type === 'cap') { | ||||
|       expr = createLiteral(selectedArtifact.subType) | ||||
|     } else if (selectedArtifact.type === 'wall') { | ||||
|       const tagResult = mutateAstWithTagForSketchSegment( | ||||
|         modifiedAst, | ||||
|         extrudeLookupResult.pathToSegmentNode | ||||
|       ) | ||||
|       if (err(tagResult)) return tagResult | ||||
|       const { tag } = tagResult | ||||
|       expr = createIdentifier(tag) | ||||
|     } else { | ||||
|       continue | ||||
|     } | ||||
|     expressions.push(expr) | ||||
|   } | ||||
|  | ||||
|   if (!pathToExtrudeNode) return new Error('No extrude found') | ||||
|  | ||||
|   const extrudeNode = getNodeFromPath<VariableDeclarator>( | ||||
|     modifiedAst, | ||||
|     pathToExtrudeNode, | ||||
|     'VariableDeclarator' | ||||
|   ) | ||||
|   if (err(extrudeNode)) { | ||||
|     return extrudeNode | ||||
|   } | ||||
|  | ||||
|   const name = findUniqueName(node, KCL_DEFAULT_CONSTANT_PREFIXES.SHELL) | ||||
|   const shell = createCallExpressionStdLib('shell', [ | ||||
|     createObjectExpression({ | ||||
|       faces: createArrayExpression(expressions), | ||||
|       thickness, | ||||
|     }), | ||||
|     createIdentifier(extrudeNode.node.id.name), | ||||
|   ]) | ||||
|   const declaration = createVariableDeclaration(name, shell) | ||||
|  | ||||
|   // TODO: check if we should append at the end like here or right after the extrude | ||||
|   modifiedAst.body.push(declaration) | ||||
|   const pathToNode: PathToNode = [ | ||||
|     ['body', ''], | ||||
|     [modifiedAst.body.length - 1, 'index'], | ||||
|     ['declaration', 'VariableDeclaration'], | ||||
|     ['init', 'VariableDeclarator'], | ||||
|     ['arguments', 'CallExpression'], | ||||
|     [0, 'index'], | ||||
|   ] | ||||
|   return { | ||||
|     modifiedAst, | ||||
|     pathToNode, | ||||
|   } | ||||
| } | ||||
| @ -17,6 +17,7 @@ import { | ||||
|   doesSceneHaveSweepableSketch, | ||||
|   traverse, | ||||
|   getNodeFromPath, | ||||
|   doesSceneHaveExtrudedSketch, | ||||
| } from './queryAst' | ||||
| import { enginelessExecutor } from '../lib/testHelpers' | ||||
| import { | ||||
| @ -654,6 +655,38 @@ extrude001 = extrude(10, sketch001) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| describe('Testing doesSceneHaveExtrudedSketch', () => { | ||||
|   it('finds extruded sketch as variable', async () => { | ||||
|     const exampleCode = `sketch001 = startSketchOn('XZ') | ||||
|   |> circle({ center = [0, 0], radius = 1 }, %) | ||||
| extrude001 = extrude(1, sketch001) | ||||
| ` | ||||
|     const ast = assertParse(exampleCode) | ||||
|     if (err(ast)) throw ast | ||||
|     const extrudable = doesSceneHaveExtrudedSketch(ast) | ||||
|     expect(extrudable).toBeTruthy() | ||||
|   }) | ||||
|   it('finds extruded sketch in pipe', async () => { | ||||
|     const exampleCode = `extrude001 = startSketchOn('XZ') | ||||
|   |> circle({ center = [0, 0], radius = 1 }, %) | ||||
|   |> extrude(1, %) | ||||
| ` | ||||
|     const ast = assertParse(exampleCode) | ||||
|     if (err(ast)) throw ast | ||||
|     const extrudable = doesSceneHaveExtrudedSketch(ast) | ||||
|     expect(extrudable).toBeTruthy() | ||||
|   }) | ||||
|   it('finds no extrusion with sketch only', async () => { | ||||
|     const exampleCode = `extrude001 = startSketchOn('XZ') | ||||
|   |> circle({ center = [0, 0], radius = 1 }, %) | ||||
| ` | ||||
|     const ast = assertParse(exampleCode) | ||||
|     if (err(ast)) throw ast | ||||
|     const extrudable = doesSceneHaveExtrudedSketch(ast) | ||||
|     expect(extrudable).toBeFalsy() | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| describe('Testing traverse and pathToNode', () => { | ||||
|   it.each([ | ||||
|     ['basic', '2.73'], | ||||
|  | ||||
| @ -1064,6 +1064,35 @@ export function doesSceneHaveSweepableSketch(ast: Node<Program>, count = 1) { | ||||
|   return Object.keys(theMap).length >= count | ||||
| } | ||||
|  | ||||
| export function doesSceneHaveExtrudedSketch(ast: Node<Program>) { | ||||
|   const theMap: any = {} | ||||
|   traverse(ast as any, { | ||||
|     enter(node) { | ||||
|       if ( | ||||
|         node.type === 'VariableDeclarator' && | ||||
|         node.init?.type === 'PipeExpression' | ||||
|       ) { | ||||
|         for (const pipe of node.init.body) { | ||||
|           if ( | ||||
|             pipe.type === 'CallExpression' && | ||||
|             pipe.callee.name === 'extrude' | ||||
|           ) { | ||||
|             theMap[node.id.name] = true | ||||
|             break | ||||
|           } | ||||
|         } | ||||
|       } else if ( | ||||
|         node.type === 'CallExpression' && | ||||
|         node.callee.name === 'extrude' && | ||||
|         node.arguments[1]?.type === 'Identifier' | ||||
|       ) { | ||||
|         theMap[node.moduleId] = true | ||||
|       } | ||||
|     }, | ||||
|   }) | ||||
|   return Object.keys(theMap).length > 0 | ||||
| } | ||||
|  | ||||
| export function getObjExprProperty( | ||||
|   node: ObjectExpression, | ||||
|   propName: string | ||||
|  | ||||
| @ -871,3 +871,15 @@ export function codeRefFromRange(range: SourceRange, ast: Program): CodeRef { | ||||
|     pathToNode: getNodePathFromSourceRange(ast, range), | ||||
|   } | ||||
| } | ||||
|  | ||||
| export function isSolid2D(artifact: Artifact): artifact is solid2D { | ||||
|   return (artifact as solid2D).pathId !== undefined | ||||
| } | ||||
|  | ||||
| export function isSegment(artifact: Artifact): artifact is SegmentArtifact { | ||||
|   return (artifact as SegmentArtifact).pathId !== undefined | ||||
| } | ||||
|  | ||||
| export function isSweep(artifact: Artifact): artifact is SweepArtifact { | ||||
|   return (artifact as SweepArtifact).pathId !== undefined | ||||
| } | ||||
|  | ||||
| @ -1,9 +1,13 @@ | ||||
| import { err } from 'lib/trap' | ||||
| import { parse, ParseResult } from './wasm' | ||||
| import { initPromise, parse, ParseResult } from './wasm' | ||||
| import { enginelessExecutor } from 'lib/testHelpers' | ||||
| import { Node } from 'wasm-lib/kcl/bindings/Node' | ||||
| import { Program } from '../wasm-lib/kcl/bindings/Program' | ||||
|  | ||||
| beforeEach(async () => { | ||||
|   await initPromise | ||||
| }) | ||||
|  | ||||
| it('can execute parsed AST', async () => { | ||||
|   const code = `x = 1 | ||||
| // A comment.` | ||||
|  | ||||
| @ -32,8 +32,14 @@ export function mouseControlsToCameraSystem( | ||||
|   mouseControl: MouseControlType | undefined | ||||
| ): CameraSystem | undefined { | ||||
|   switch (mouseControl) { | ||||
|     // TODO: understand why the values come back without underscores and fix the root cause | ||||
|     // @ts-ignore: TS2678 | ||||
|     case 'kittycad': | ||||
|     case 'kitty_cad': | ||||
|       return 'KittyCAD' | ||||
|     // TODO: understand why the values come back without underscores and fix the root cause | ||||
|     // @ts-ignore: TS2678 | ||||
|     case 'onshape': | ||||
|     case 'on_shape': | ||||
|       return 'OnShape' | ||||
|     case 'trackpad_friendly': | ||||
| @ -44,6 +50,9 @@ export function mouseControlsToCameraSystem( | ||||
|       return 'NX' | ||||
|     case 'creo': | ||||
|       return 'Creo' | ||||
|     // TODO: understand why the values come back without underscores and fix the root cause | ||||
|     // @ts-ignore: TS2678 | ||||
|     case 'autocad': | ||||
|     case 'auto_cad': | ||||
|       return 'AutoCAD' | ||||
|     default: | ||||
|  | ||||
| @ -1,9 +1,15 @@ | ||||
| import { Models } from '@kittycad/lib' | ||||
| import { angleLengthInfo } from 'components/Toolbar/setAngleLength' | ||||
| import { transformAstSketchLines } from 'lang/std/sketchcombos' | ||||
| import { PathToNode } from 'lang/wasm' | ||||
| import { StateMachineCommandSetConfig, KclCommandValue } from 'lib/commandTypes' | ||||
| import { KCL_DEFAULT_LENGTH, KCL_DEFAULT_DEGREE } from 'lib/constants' | ||||
| import { components } from 'lib/machine-api' | ||||
| import { Selections } from 'lib/selections' | ||||
| import { kclManager } from 'lib/singletons' | ||||
| import { err } from 'lib/trap' | ||||
| import { modelingMachine, SketchTool } from 'machines/modelingMachine' | ||||
| import { revolveAxisValidator } from './validators' | ||||
|  | ||||
| type OutputFormat = Models['OutputFormat_type'] | ||||
| type OutputTypeKey = OutputFormat['type'] | ||||
| @ -34,9 +40,14 @@ export type ModelingCommandSchema = { | ||||
|   Loft: { | ||||
|     selection: Selections | ||||
|   } | ||||
|   Shell: { | ||||
|     selection: Selections | ||||
|     thickness: KclCommandValue | ||||
|   } | ||||
|   Revolve: { | ||||
|     selection: Selections | ||||
|     angle: KclCommandValue | ||||
|     axis: Selections | ||||
|   } | ||||
|   Fillet: { | ||||
|     // todo | ||||
| @ -50,6 +61,18 @@ export type ModelingCommandSchema = { | ||||
|   'change tool': { | ||||
|     tool: SketchTool | ||||
|   } | ||||
|   'Constrain length': { | ||||
|     selection: Selections | ||||
|     length: KclCommandValue | ||||
|   } | ||||
|   'Constrain with named value': { | ||||
|     currentValue: { | ||||
|       valueText: string | ||||
|       pathToNode: PathToNode | ||||
|       variableName: string | ||||
|     } | ||||
|     namedValue: KclCommandValue | ||||
|   } | ||||
|   'Text-to-CAD': { | ||||
|     prompt: string | ||||
|   } | ||||
| @ -277,6 +300,25 @@ export const modelingMachineCommandConfig: StateMachineCommandSetConfig< | ||||
|       }, | ||||
|     }, | ||||
|   }, | ||||
|   Shell: { | ||||
|     description: 'Hollow out a 3D solid.', | ||||
|     icon: 'shell', | ||||
|     needsReview: true, | ||||
|     args: { | ||||
|       selection: { | ||||
|         inputType: 'selection', | ||||
|         selectionTypes: ['cap', 'wall'], | ||||
|         multiple: true, | ||||
|         required: true, | ||||
|         skip: false, | ||||
|       }, | ||||
|       thickness: { | ||||
|         inputType: 'kcl', | ||||
|         defaultValue: KCL_DEFAULT_LENGTH, | ||||
|         required: true, | ||||
|       }, | ||||
|     }, | ||||
|   }, | ||||
|   // TODO: Update this configuration, copied from extrude for MVP of revolve, specifically the args.selection | ||||
|   Revolve: { | ||||
|     description: 'Create a 3D body by rotating a sketch region about an axis.', | ||||
| @ -290,6 +332,13 @@ export const modelingMachineCommandConfig: StateMachineCommandSetConfig< | ||||
|         required: true, | ||||
|         skip: true, | ||||
|       }, | ||||
|       axis: { | ||||
|         required: true, | ||||
|         inputType: 'selection', | ||||
|         selectionTypes: ['segment', 'sweepEdge', 'edgeCutEdge'], | ||||
|         multiple: false, | ||||
|         validation: revolveAxisValidator, | ||||
|       }, | ||||
|       angle: { | ||||
|         inputType: 'kcl', | ||||
|         defaultValue: KCL_DEFAULT_DEGREE, | ||||
| @ -337,6 +386,88 @@ export const modelingMachineCommandConfig: StateMachineCommandSetConfig< | ||||
|       }, | ||||
|     }, | ||||
|   }, | ||||
|   'Constrain length': { | ||||
|     description: 'Constrain the length of one or more segments.', | ||||
|     icon: 'dimension', | ||||
|     args: { | ||||
|       selection: { | ||||
|         inputType: 'selection', | ||||
|         selectionTypes: ['segment'], | ||||
|         multiple: false, | ||||
|         required: true, | ||||
|         skip: true, | ||||
|       }, | ||||
|       length: { | ||||
|         inputType: 'kcl', | ||||
|         required: true, | ||||
|         createVariableByDefault: true, | ||||
|         defaultValue(_, machineContext) { | ||||
|           const selectionRanges = machineContext?.selectionRanges | ||||
|           if (!selectionRanges) return KCL_DEFAULT_LENGTH | ||||
|           const angleLength = angleLengthInfo({ | ||||
|             selectionRanges, | ||||
|             angleOrLength: 'setLength', | ||||
|           }) | ||||
|           if (err(angleLength)) return KCL_DEFAULT_LENGTH | ||||
|           const { transforms } = angleLength | ||||
|  | ||||
|           // QUESTION: is it okay to reference kclManager here? will its state be up to date? | ||||
|           const sketched = transformAstSketchLines({ | ||||
|             ast: structuredClone(kclManager.ast), | ||||
|             selectionRanges, | ||||
|             transformInfos: transforms, | ||||
|             programMemory: kclManager.programMemory, | ||||
|             referenceSegName: '', | ||||
|           }) | ||||
|           if (err(sketched)) return KCL_DEFAULT_LENGTH | ||||
|           const { valueUsedInTransform } = sketched | ||||
|           return valueUsedInTransform?.toString() || KCL_DEFAULT_LENGTH | ||||
|         }, | ||||
|       }, | ||||
|     }, | ||||
|   }, | ||||
|   'Constrain with named value': { | ||||
|     description: 'Constrain a value by making it a named constant.', | ||||
|     icon: 'make-variable', | ||||
|     args: { | ||||
|       currentValue: { | ||||
|         description: | ||||
|           'Path to the node in the AST to constrain. This is never shown to the user.', | ||||
|         inputType: 'text', | ||||
|         required: false, | ||||
|         skip: true, | ||||
|       }, | ||||
|       namedValue: { | ||||
|         inputType: 'kcl', | ||||
|         required: true, | ||||
|         createVariableByDefault: true, | ||||
|         variableName(commandBarContext, machineContext) { | ||||
|           const { currentValue } = commandBarContext.argumentsToSubmit | ||||
|           if ( | ||||
|             !currentValue || | ||||
|             !(currentValue instanceof Object) || | ||||
|             !('variableName' in currentValue) || | ||||
|             typeof currentValue.variableName !== 'string' | ||||
|           ) { | ||||
|             return 'value' | ||||
|           } | ||||
|           return currentValue.variableName | ||||
|         }, | ||||
|         defaultValue: (commandBarContext) => { | ||||
|           const { currentValue } = commandBarContext.argumentsToSubmit | ||||
|           if ( | ||||
|             !currentValue || | ||||
|             !(currentValue instanceof Object) || | ||||
|             !('valueText' in currentValue) || | ||||
|             typeof currentValue.valueText !== 'string' | ||||
|           ) { | ||||
|             return KCL_DEFAULT_LENGTH | ||||
|           } | ||||
|           return currentValue.valueText | ||||
|         }, | ||||
|       }, | ||||
|     }, | ||||
|   }, | ||||
|   'Text-to-CAD': { | ||||
|     description: 'Use the Zoo Text-to-CAD API to generate part starters.', | ||||
|     icon: 'chat', | ||||
|  | ||||
							
								
								
									
										107
									
								
								src/lib/commandBarConfigs/validators.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,107 @@ | ||||
| import { Models } from '@kittycad/lib' | ||||
| import { engineCommandManager } from 'lib/singletons' | ||||
| import { uuidv4 } from 'lib/utils' | ||||
| import { CommandBarContext } from 'machines/commandBarMachine' | ||||
| import { Selections } from 'lib/selections' | ||||
| import { isSolid2D, isSegment, isSweep } from 'lang/std/artifactGraph' | ||||
|  | ||||
| export const disableDryRunWithRetry = async (numberOfRetries = 3) => { | ||||
|   for (let tries = 0; tries < numberOfRetries; tries++) { | ||||
|     try { | ||||
|       await engineCommandManager.sendSceneCommand({ | ||||
|         type: 'modeling_cmd_req', | ||||
|         cmd_id: uuidv4(), | ||||
|         cmd: { type: 'disable_dry_run' }, | ||||
|       }) | ||||
|       // Exit out since the command was successful | ||||
|       return | ||||
|     } catch (e) { | ||||
|       console.error(e) | ||||
|       console.error('disable_dry_run failed. This is bad!') | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| // Takes a callback function and wraps it around enable_dry_run and disable_dry_run | ||||
| export const dryRunWrapper = async (callback: () => Promise<any>) => { | ||||
|   // Gotcha: What about race conditions? | ||||
|   try { | ||||
|     await engineCommandManager.sendSceneCommand({ | ||||
|       type: 'modeling_cmd_req', | ||||
|       cmd_id: uuidv4(), | ||||
|       cmd: { type: 'enable_dry_run' }, | ||||
|     }) | ||||
|     const result = await callback() | ||||
|     return result | ||||
|   } catch (e) { | ||||
|     console.error(e) | ||||
|   } finally { | ||||
|     await disableDryRunWithRetry(5) | ||||
|   } | ||||
| } | ||||
|  | ||||
| function isSelections(selections: unknown): selections is Selections { | ||||
|   return ( | ||||
|     (selections as Selections).graphSelections !== undefined && | ||||
|     (selections as Selections).otherSelections !== undefined | ||||
|   ) | ||||
| } | ||||
|  | ||||
| export const revolveAxisValidator = async ({ | ||||
|   data, | ||||
|   context, | ||||
| }: { | ||||
|   data: { [key: string]: Selections } | ||||
|   context: CommandBarContext | ||||
| }): Promise<boolean | string> => { | ||||
|   if (!isSelections(context.argumentsToSubmit.selection)) { | ||||
|     return 'Unable to revolve, selections are missing' | ||||
|   } | ||||
|   const artifact = | ||||
|     context.argumentsToSubmit.selection.graphSelections[0].artifact | ||||
|  | ||||
|   if (!artifact) { | ||||
|     return 'Unable to revolve, sketch not found' | ||||
|   } | ||||
|  | ||||
|   if (!(isSolid2D(artifact) || isSegment(artifact) || isSweep(artifact))) { | ||||
|     return 'Unable to revolve, sketch has no path' | ||||
|   } | ||||
|  | ||||
|   const sketchSelection = artifact.pathId | ||||
|   let edgeSelection = data.axis.graphSelections[0].artifact?.id | ||||
|  | ||||
|   if (!sketchSelection) { | ||||
|     return 'Unable to revolve, sketch is missing' | ||||
|   } | ||||
|  | ||||
|   if (!edgeSelection) { | ||||
|     return 'Unable to revolve, edge is missing' | ||||
|   } | ||||
|  | ||||
|   const angleInDegrees: Models['Angle_type'] = { | ||||
|     unit: 'degrees', | ||||
|     value: 360, | ||||
|   } | ||||
|  | ||||
|   const revolveAboutEdgeCommand = async () => { | ||||
|     return await engineCommandManager.sendSceneCommand({ | ||||
|       type: 'modeling_cmd_req', | ||||
|       cmd_id: uuidv4(), | ||||
|       cmd: { | ||||
|         type: 'revolve_about_edge', | ||||
|         angle: angleInDegrees, | ||||
|         edge_id: edgeSelection, | ||||
|         target: sketchSelection, | ||||
|         tolerance: 0.0001, | ||||
|       }, | ||||
|     }) | ||||
|   } | ||||
|   const attemptRevolve = await dryRunWrapper(revolveAboutEdgeCommand) | ||||
|   if (attemptRevolve?.success) { | ||||
|     return true | ||||
|   } else { | ||||
|     // return error message for the toast | ||||
|     return 'Unable to revolve with selected axis' | ||||
|   } | ||||
| } | ||||
| @ -7,7 +7,7 @@ import { ReactNode } from 'react' | ||||
| import { MachineManager } from 'components/MachineManagerProvider' | ||||
| import { Node } from 'wasm-lib/kcl/bindings/Node' | ||||
| import { Artifact } from 'lang/std/artifactGraph' | ||||
|  | ||||
| import { CommandBarContext } from 'machines/commandBarMachine' | ||||
| type Icon = CustomIconName | ||||
| const PLATFORMS = ['both', 'web', 'desktop'] as const | ||||
| const INPUT_TYPES = [ | ||||
| @ -147,8 +147,30 @@ export type CommandArgumentConfig< | ||||
|       inputType: 'selection' | ||||
|       selectionTypes: Artifact['type'][] | ||||
|       multiple: boolean | ||||
|       validation?: ({ | ||||
|         data, | ||||
|         context, | ||||
|       }: { | ||||
|         data: any | ||||
|         context: CommandBarContext | ||||
|       }) => Promise<boolean | string> | ||||
|     } | ||||
|   | { | ||||
|       inputType: 'kcl' | ||||
|       createVariableByDefault?: boolean | ||||
|       variableName?: | ||||
|         | string | ||||
|         | (( | ||||
|             commandBarContext: ContextFrom<typeof commandBarMachine>, | ||||
|             machineContext?: C | ||||
|           ) => string) | ||||
|       defaultValue?: | ||||
|         | string | ||||
|         | (( | ||||
|             commandBarContext: ContextFrom<typeof commandBarMachine>, | ||||
|             machineContext?: C | ||||
|           ) => string) | ||||
|     } | ||||
|   | { inputType: 'kcl'; defaultValue?: string } // KCL expression inputs have simple strings as default values | ||||
|   | { | ||||
|       inputType: 'string' | ||||
|       defaultValue?: | ||||
| @ -221,8 +243,30 @@ export type CommandArgument< | ||||
|       inputType: 'selection' | ||||
|       selectionTypes: Artifact['type'][] | ||||
|       multiple: boolean | ||||
|       validation?: ({ | ||||
|         data, | ||||
|         context, | ||||
|       }: { | ||||
|         data: any | ||||
|         context: CommandBarContext | ||||
|       }) => Promise<boolean | string> | ||||
|     } | ||||
|   | { | ||||
|       inputType: 'kcl' | ||||
|       createVariableByDefault?: boolean | ||||
|       variableName?: | ||||
|         | string | ||||
|         | (( | ||||
|             commandBarContext: ContextFrom<typeof commandBarMachine>, | ||||
|             machineContext?: ContextFrom<T> | ||||
|           ) => string) | ||||
|       defaultValue?: | ||||
|         | string | ||||
|         | (( | ||||
|             commandBarContext: ContextFrom<typeof commandBarMachine>, | ||||
|             machineContext?: ContextFrom<T> | ||||
|           ) => string) | ||||
|     } | ||||
|   | { inputType: 'kcl'; defaultValue?: string } // KCL expression inputs have simple strings as default value | ||||
|   | { | ||||
|       inputType: 'string' | ||||
|       defaultValue?: | ||||
|  | ||||
| @ -53,6 +53,7 @@ export const KCL_DEFAULT_CONSTANT_PREFIXES = { | ||||
|   SKETCH: 'sketch', | ||||
|   EXTRUDE: 'extrude', | ||||
|   LOFT: 'loft', | ||||
|   SHELL: 'shell', | ||||
|   SEGMENT: 'seg', | ||||
|   REVOLVE: 'revolve', | ||||
|   PLANE: 'plane', | ||||
| @ -110,3 +111,10 @@ export const KCL_SAMPLES_MANIFEST_URLS = { | ||||
|  | ||||
| /** Toast id for the app auto-updater toast */ | ||||
| export const AUTO_UPDATER_TOAST_ID = 'auto-updater-toast' | ||||
|  | ||||
| /** Local sketch axis values in KCL for operations, it could either be 'X' or 'Y' */ | ||||
| export const KCL_AXIS_X = 'X' | ||||
| export const KCL_AXIS_Y = 'Y' | ||||
| export const KCL_AXIS_NEG_X = '-X' | ||||
| export const KCL_AXIS_NEG_Y = '-Y' | ||||
| export const KCL_DEFAULT_AXIS = 'X' | ||||
|  | ||||
| @ -155,6 +155,8 @@ export function buildCommandArgument< | ||||
|   context: ContextFrom<T>, | ||||
|   machineActor: Actor<T> | ||||
| ): CommandArgument<O, T> & { inputType: typeof arg.inputType } { | ||||
|   // GOTCHA: modelingCommandConfig is not a 1:1 mapping to this baseCommandArgument | ||||
|   // You need to manually add key/value pairs here. | ||||
|   const baseCommandArgument = { | ||||
|     description: arg.description, | ||||
|     required: arg.required, | ||||
| @ -181,10 +183,13 @@ export function buildCommandArgument< | ||||
|       ...baseCommandArgument, | ||||
|       multiple: arg.multiple, | ||||
|       selectionTypes: arg.selectionTypes, | ||||
|       validation: arg.validation, | ||||
|     } satisfies CommandArgument<O, T> & { inputType: 'selection' } | ||||
|   } else if (arg.inputType === 'kcl') { | ||||
|     return { | ||||
|       inputType: arg.inputType, | ||||
|       createVariableByDefault: arg.createVariableByDefault, | ||||
|       variableName: arg.variableName, | ||||
|       defaultValue: arg.defaultValue, | ||||
|       ...baseCommandArgument, | ||||
|     } satisfies CommandArgument<O, T> & { inputType: 'kcl' } | ||||
|  | ||||
| @ -569,6 +569,17 @@ export function canSweepSelection(selection: Selections) { | ||||
|   ) | ||||
| } | ||||
|  | ||||
| export function canRevolveSelection(selection: Selections) { | ||||
|   const commonNodes = selection.graphSelections.map((_, i) => | ||||
|     buildCommonNodeFromSelection(selection, i) | ||||
|   ) | ||||
|   return ( | ||||
|     !!isSketchPipe(selection) && | ||||
|     (commonNodes.every((n) => nodeHasClose(n)) || | ||||
|       commonNodes.every((n) => nodeHasCircle(n))) | ||||
|   ) | ||||
| } | ||||
|  | ||||
| export function canLoftSelection(selection: Selections) { | ||||
|   const commonNodes = selection.graphSelections.map((_, i) => | ||||
|     buildCommonNodeFromSelection(selection, i) | ||||
| @ -585,6 +596,17 @@ export function canLoftSelection(selection: Selections) { | ||||
|   ) | ||||
| } | ||||
|  | ||||
| export function canShellSelection(selection: Selections) { | ||||
|   const commonNodes = selection.graphSelections.map((_, i) => | ||||
|     buildCommonNodeFromSelection(selection, i) | ||||
|   ) | ||||
|   return commonNodes.every( | ||||
|     (n) => | ||||
|       n.selection.artifact?.type === 'cap' || | ||||
|       n.selection.artifact?.type === 'wall' | ||||
|   ) | ||||
| } | ||||
|  | ||||
| // This accounts for non-geometry selections under "other" | ||||
| export type ResolvedSelectionType = Artifact['type'] | 'other' | ||||
| export type SelectionCountsByType = Map<ResolvedSelectionType, number> | ||||
| @ -619,12 +641,29 @@ export function getSelectionCountByType( | ||||
|     } | ||||
|   }) | ||||
|  | ||||
|   selection.graphSelections.forEach((selection) => { | ||||
|     if (!selection.artifact) { | ||||
|   selection.graphSelections.forEach((graphSelection) => { | ||||
|     if (!graphSelection.artifact) { | ||||
|       /** | ||||
|        * TODO: remove this heuristic-based selection type detection. | ||||
|        * Currently, if you've created a sketch and have not left sketch mode, | ||||
|        * the selection will be a segment selection with no artifact. | ||||
|        * This is because the mock execution does not update the artifact graph. | ||||
|        * Once we move the artifactGraph creation to WASM, we can remove this, | ||||
|        * as the artifactGraph will always be up-to-date. | ||||
|        */ | ||||
|       if (isSingleCursorInPipe(selection, kclManager.ast)) { | ||||
|         incrementOrInitializeSelectionType('segment') | ||||
|         return | ||||
|       } else { | ||||
|         console.warn( | ||||
|           'Selection is outside of a sketch but has no artifact. Sketch segment selections are the only kind that can have a valid selection with no artifact.', | ||||
|           JSON.stringify(graphSelection) | ||||
|         ) | ||||
|         incrementOrInitializeSelectionType('other') | ||||
|         return | ||||
|       } | ||||
|     incrementOrInitializeSelectionType(selection.artifact.type) | ||||
|     } | ||||
|     incrementOrInitializeSelectionType(graphSelection.artifact.type) | ||||
|   }) | ||||
|  | ||||
|   return selectionsByType | ||||
|  | ||||
| @ -12,7 +12,7 @@ export type InteractionMapItem = { | ||||
|  * Controls both the available names for interaction map categories | ||||
|  * and the order in which they are displayed. | ||||
|  */ | ||||
| export const interactionMapCategories = [ | ||||
| const interactionMapCategories = [ | ||||
|   'Sketching', | ||||
|   'Modeling', | ||||
|   'Command Palette', | ||||
|  | ||||
| @ -190,9 +190,15 @@ export const toolbarConfig: Record<ToolbarModeName, ToolbarMode> = { | ||||
|       }, | ||||
|       { | ||||
|         id: 'shell', | ||||
|         onClick: () => console.error('Shell not yet implemented'), | ||||
|         onClick: ({ commandBarSend }) => { | ||||
|           commandBarSend({ | ||||
|             type: 'Find and select command', | ||||
|             data: { name: 'Shell', groupId: 'modeling' }, | ||||
|           }) | ||||
|         }, | ||||
|         disabled: (state) => !state.can({ type: 'Shell' }), | ||||
|         icon: 'shell', | ||||
|         status: 'kcl-only', | ||||
|         status: 'available', | ||||
|         title: 'Shell', | ||||
|         description: 'Hollow out a 3D solid.', | ||||
|         links: [{ label: 'KCL docs', url: 'https://zoo.dev/docs/kcl/shell' }], | ||||
| @ -534,13 +540,15 @@ export const toolbarConfig: Record<ToolbarModeName, ToolbarMode> = { | ||||
|       [ | ||||
|         { | ||||
|           id: 'constraint-length', | ||||
|           disabled: (state) => | ||||
|             !( | ||||
|               state.matches({ Sketch: 'SketchIdle' }) && | ||||
|               state.can({ type: 'Constrain length' }) | ||||
|             ), | ||||
|           onClick: ({ modelingSend }) => | ||||
|             modelingSend({ type: 'Constrain length' }), | ||||
|           disabled: (state) => !state.matches({ Sketch: 'SketchIdle' }), | ||||
|           onClick: ({ commandBarSend }) => | ||||
|             commandBarSend({ | ||||
|               type: 'Find and select command', | ||||
|               data: { | ||||
|                 name: 'Constrain length', | ||||
|                 groupId: 'modeling', | ||||
|               }, | ||||
|             }), | ||||
|           icon: 'dimension', | ||||
|           status: 'available', | ||||
|           title: 'Length', | ||||
|  | ||||
| @ -8,6 +8,7 @@ import { | ||||
| import { Selections__old } from 'lib/selections' | ||||
| import { getCommandArgumentKclValuesOnly } from 'lib/commandUtils' | ||||
| import { MachineManager } from 'components/MachineManagerProvider' | ||||
| import toast from 'react-hot-toast' | ||||
|  | ||||
| export type CommandBarContext = { | ||||
|   commands: Command[] | ||||
| @ -247,14 +248,69 @@ export const commandBarMachine = setup({ | ||||
|     'All arguments are skippable': () => false, | ||||
|   }, | ||||
|   actors: { | ||||
|     'Validate argument': fromPromise(({ input }) => { | ||||
|     'Validate argument': fromPromise( | ||||
|       ({ | ||||
|         input, | ||||
|       }: { | ||||
|         input: { | ||||
|           context: CommandBarContext | undefined | ||||
|           event: CommandBarMachineEvent | undefined | ||||
|         } | ||||
|       }) => { | ||||
|         return new Promise((resolve, reject) => { | ||||
|         // TODO: figure out if we should validate argument data here or in the form itself, | ||||
|         // and if we should support people configuring a argument's validation function | ||||
|           if (!input || input?.event?.type !== 'Submit argument') { | ||||
|             toast.error(`Unable to validate, wrong event type.`) | ||||
|             return reject(`Unable to validate, wrong event type`) | ||||
|           } | ||||
|  | ||||
|         resolve(input) | ||||
|           const context = input?.context | ||||
|  | ||||
|           if (!context) { | ||||
|             toast.error(`Unable to validate, wrong argument.`) | ||||
|             return reject(`Unable to validate, wrong argument`) | ||||
|           } | ||||
|  | ||||
|           const data = input.event.data | ||||
|           const argName = context.currentArgument?.name | ||||
|           const args = context?.selectedCommand?.args | ||||
|           const argConfig = args && argName ? args[argName] : undefined | ||||
|           // Only do a validation check if the argument, selectedCommand, and the validation function are defined | ||||
|           if ( | ||||
|             context.currentArgument && | ||||
|             context.selectedCommand && | ||||
|             argConfig?.inputType === 'selection' && | ||||
|             argConfig?.validation | ||||
|           ) { | ||||
|             argConfig | ||||
|               .validation({ context, data }) | ||||
|               .then((result) => { | ||||
|                 if (typeof result === 'boolean' && result === true) { | ||||
|                   return resolve(data) | ||||
|                 } else { | ||||
|                   // validation failed | ||||
|                   if (typeof result === 'string') { | ||||
|                     // The result of the validation is the error message | ||||
|                     toast.error(result) | ||||
|                     return reject( | ||||
|                       `unable to validate ${argName}, Message: ${result}` | ||||
|                     ) | ||||
|                   } else { | ||||
|                     // Default message if there is not a custom one sent | ||||
|                     toast.error(`Unable to validate ${argName}`) | ||||
|                     return reject(`unable to validate ${argName}}`) | ||||
|                   } | ||||
|                 } | ||||
|               }) | ||||
|     }), | ||||
|               .catch(() => { | ||||
|                 return reject(`unable to validate ${argName}}`) | ||||
|               }) | ||||
|           } else { | ||||
|             // Missing several requirements for validate argument, just bypass | ||||
|             return resolve(data) | ||||
|           } | ||||
|         }) | ||||
|       } | ||||
|     ), | ||||
|     'Validate all arguments': fromPromise( | ||||
|       ({ input }: { input: CommandBarContext }) => { | ||||
|         return new Promise((resolve, reject) => { | ||||
| @ -449,9 +505,10 @@ export const commandBarMachine = setup({ | ||||
|           invoke: { | ||||
|             src: 'Validate argument', | ||||
|             id: 'validateSingleArgument', | ||||
|             input: ({ event }) => { | ||||
|               if (event.type !== 'Submit argument') return {} | ||||
|               return event.data | ||||
|             input: ({ event, context }) => { | ||||
|               if (event.type !== 'Submit argument') | ||||
|                 return { event: undefined, context: undefined } | ||||
|               return { event, context } | ||||
|             }, | ||||
|             onDone: { | ||||
|               target: '#Command Bar.Checking Arguments', | ||||
|  | ||||
							
								
								
									
										31
									
								
								src/main.ts
									
									
									
									
									
								
							
							
						
						| @ -23,6 +23,15 @@ import argvFromYargs from './commandLineArgs' | ||||
|  | ||||
| let mainWindow: BrowserWindow | null = null | ||||
|  | ||||
| // Supporting multiple instances instead of multiple applications | ||||
| let cmdQPressed = false | ||||
| const instances: BrowserWindow[] = [] | ||||
| const gotTheLock = app.requestSingleInstanceLock() | ||||
| if (!gotTheLock) { | ||||
|   app.quit() | ||||
|   process.exit(0) | ||||
| } | ||||
|  | ||||
| // Check the command line arguments for a project path | ||||
| const args = parseCLIArgs() | ||||
|  | ||||
| @ -117,16 +126,34 @@ const createWindow = (filePath?: string): BrowserWindow => { | ||||
|  | ||||
|   newWindow.show() | ||||
|  | ||||
|   instances.push(newWindow) | ||||
|   return newWindow | ||||
| } | ||||
|  | ||||
| // before-quit with multiple instances | ||||
| if (process.platform === 'darwin') { | ||||
|   // Quit from the dock context menu should quit the application directly | ||||
|   app.on('before-quit', () => { | ||||
|     cmdQPressed = true | ||||
|   }) | ||||
| } | ||||
|  | ||||
| // Quit when all windows are closed, even on macOS. There, it's common | ||||
| // for applications and their menu bar to stay active until the user quits | ||||
| // explicitly with Cmd + Q, but it is a really weird behavior with our app. | ||||
| // app.on('window-all-closed', () => { | ||||
| //   app.quit() | ||||
| // }) | ||||
| app.on('window-all-closed', () => { | ||||
|   if (cmdQPressed || process.platform !== 'darwin') { | ||||
|     app.quit() | ||||
|   } | ||||
| }) | ||||
|  | ||||
| // Various actions can trigger this event, such as launching the application for the first time, | ||||
| // attempting to re-launch the application when it's already running, or clicking on the application's dock or taskbar icon. | ||||
| app.on('activate', () => createWindow()) | ||||
|  | ||||
| // This method will be called when Electron has finished | ||||
| // initialization and is ready to create browser windows. | ||||
| // Some APIs can only be used after this event occurs. | ||||
| @ -135,6 +162,10 @@ app.on('ready', (event, data) => { | ||||
|   mainWindow = createWindow() | ||||
| }) | ||||
|  | ||||
| // This event will be emitted inside the primary instance of your application when a second instance | ||||
| // has been executed and calls app.requestSingleInstanceLock(). | ||||
| app.on('second-instance', (event, argv, workingDirectory) => createWindow()) | ||||
|  | ||||
| // For now there is no good reason to separate these out to another file(s) | ||||
| // There is just not enough code to warrant it and further abstracts everything | ||||
| // which is already quite abstracted | ||||
|  | ||||
| @ -30,6 +30,12 @@ export const PACKAGE_NAME = isDesktop() | ||||
|   ? window.electron.packageJson.name | ||||
|   : 'zoo-modeling-app' | ||||
|  | ||||
| export const IS_NIGHTLY = PACKAGE_NAME.indexOf('-nightly') > -1 | ||||
|  | ||||
| export const RELEASE_URL = `https://github.com/KittyCAD/modeling-app/releases/tag/${ | ||||
|   IS_NIGHTLY ? 'nightly-' : '' | ||||
| }v${APP_VERSION}` | ||||
|  | ||||
| export const Settings = () => { | ||||
|   const navigate = useNavigate() | ||||
|   const [searchParams, setSearchParams] = useSearchParams() | ||||
|  | ||||
| @ -79,7 +79,10 @@ kittycad = { version = "0.3.28", default-features = false, features = ["js", "re | ||||
| kittycad-modeling-cmds = { version = "0.2.77", features = ["websocket"] } | ||||
|  | ||||
| [workspace.lints.clippy] | ||||
| assertions_on_result_states = "warn" | ||||
| dbg_macro = "warn" | ||||
| iter_over_hash_type = "warn" | ||||
| lossy_float_literal = "warn" | ||||
|  | ||||
| [[test]] | ||||
| name = "executor" | ||||
|  | ||||
| @ -12,8 +12,8 @@ redo-kcl-stdlib-docs-no-imgs: | ||||
| # Generate the stdlib image artifacts | ||||
| # Then run the stdlib docs generation | ||||
| redo-kcl-stdlib-docs: | ||||
|     TWENTY_TWENTY=overwrite {{cnr}} -p kcl-lib kcl_test_example | ||||
|     EXPECTORATE=overwrite {{cnr}} -p kcl-lib docs::gen_std_tests::test_generate_stdlib | ||||
|     TWENTY_TWENTY=overwrite {{cnr}} -p kcl-lib --no-fail-fast -- kcl_test_example | ||||
|     EXPECTORATE=overwrite {{cnr}} -p kcl-lib --no-fail-fast -- docs::gen_std_tests::test_generate_stdlib | ||||
|  | ||||
| # Copy a test KCL file from executor tests into a new simulation test. | ||||
| copy-exec-test-into-sim-test test_name: | ||||
|  | ||||
| @ -15,5 +15,5 @@ async fn kcl_to_core_test() { | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     assert!(result.is_ok()); | ||||
|     result.unwrap(); | ||||
| } | ||||
|  | ||||
| @ -366,9 +366,11 @@ impl Node<CallExpressionKw> { | ||||
|     #[async_recursion] | ||||
|     pub async fn execute(&self, exec_state: &mut ExecState, ctx: &ExecutorContext) -> Result<KclValue, KclError> { | ||||
|         let fn_name = &self.callee.name; | ||||
|         let callsite: SourceRange = self.into(); | ||||
|  | ||||
|         // Build a hashmap from argument labels to the final evaluated values. | ||||
|         let mut fn_args = HashMap::with_capacity(self.arguments.len()); | ||||
|         let mut tag_declarator_args = Vec::new(); | ||||
|         for arg_expr in &self.arguments { | ||||
|             let source_range = SourceRange::from(arg_expr.arg.clone()); | ||||
|             let metadata = Metadata { source_range }; | ||||
| @ -376,8 +378,12 @@ impl Node<CallExpressionKw> { | ||||
|                 .execute_expr(&arg_expr.arg, exec_state, &metadata, StatementKind::Expression) | ||||
|                 .await?; | ||||
|             fn_args.insert(arg_expr.label.name.clone(), Arg::new(value, source_range)); | ||||
|             if let Expr::TagDeclarator(td) = &arg_expr.arg { | ||||
|                 tag_declarator_args.push((td.inner.clone(), source_range)); | ||||
|             } | ||||
|         } | ||||
|         let fn_args = fn_args; // remove mutability | ||||
|         let tag_declarator_args = tag_declarator_args; // remove mutability | ||||
|  | ||||
|         // Evaluate the unlabeled first param, if any exists. | ||||
|         let unlabeled = if let Some(ref arg_expr) = self.unlabeled { | ||||
| @ -403,11 +409,43 @@ impl Node<CallExpressionKw> { | ||||
|             FunctionKind::Core(func) => { | ||||
|                 // Attempt to call the function. | ||||
|                 let mut result = func.std_lib_fn()(exec_state, args).await?; | ||||
|                 update_memory_for_tags_of_geometry(&mut result, exec_state)?; | ||||
|                 update_memory_for_tags_of_geometry(&mut result, &tag_declarator_args, exec_state)?; | ||||
|                 Ok(result) | ||||
|             } | ||||
|             FunctionKind::UserDefined => { | ||||
|                 todo!("Part of modeling-app#4600: Support keyword arguments for user-defined functions") | ||||
|                 let source_range = SourceRange::from(self); | ||||
|                 // Clone the function so that we can use a mutable reference to | ||||
|                 // exec_state. | ||||
|                 let func = exec_state.memory.get(fn_name, source_range)?.clone(); | ||||
|                 let fn_dynamic_state = exec_state.dynamic_state.merge(&exec_state.memory); | ||||
|  | ||||
|                 let return_value = { | ||||
|                     let previous_dynamic_state = std::mem::replace(&mut exec_state.dynamic_state, fn_dynamic_state); | ||||
|                     let result = func | ||||
|                         .call_fn_kw(args, exec_state, ctx.clone(), callsite) | ||||
|                         .await | ||||
|                         .map_err(|e| { | ||||
|                             // Add the call expression to the source ranges. | ||||
|                             // TODO currently ignored by the frontend | ||||
|                             e.add_source_ranges(vec![source_range]) | ||||
|                         }); | ||||
|                     exec_state.dynamic_state = previous_dynamic_state; | ||||
|                     result? | ||||
|                 }; | ||||
|  | ||||
|                 let result = return_value.ok_or_else(move || { | ||||
|                     let mut source_ranges: Vec<SourceRange> = vec![source_range]; | ||||
|                     // We want to send the source range of the original function. | ||||
|                     if let KclValue::Function { meta, .. } = func { | ||||
|                         source_ranges = meta.iter().map(|m| m.source_range).collect(); | ||||
|                     }; | ||||
|                     KclError::UndefinedValue(KclErrorDetails { | ||||
|                         message: format!("Result of user-defined function {} is undefined", fn_name), | ||||
|                         source_ranges, | ||||
|                     }) | ||||
|                 })?; | ||||
|  | ||||
|                 Ok(result) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| @ -419,6 +457,7 @@ impl Node<CallExpression> { | ||||
|         let fn_name = &self.callee.name; | ||||
|  | ||||
|         let mut fn_args: Vec<Arg> = Vec::with_capacity(self.arguments.len()); | ||||
|         let mut tag_declarator_args = Vec::new(); | ||||
|  | ||||
|         for arg_expr in &self.arguments { | ||||
|             let metadata = Metadata { | ||||
| @ -428,15 +467,19 @@ impl Node<CallExpression> { | ||||
|                 .execute_expr(arg_expr, exec_state, &metadata, StatementKind::Expression) | ||||
|                 .await?; | ||||
|             let arg = Arg::new(value, SourceRange::from(arg_expr)); | ||||
|             if let Expr::TagDeclarator(td) = arg_expr { | ||||
|                 tag_declarator_args.push((td.inner.clone(), arg.source_range)); | ||||
|             } | ||||
|             fn_args.push(arg); | ||||
|         } | ||||
|         let tag_declarator_args = tag_declarator_args; // remove mutability | ||||
|  | ||||
|         match ctx.stdlib.get_either(fn_name) { | ||||
|             FunctionKind::Core(func) => { | ||||
|                 // Attempt to call the function. | ||||
|                 let args = crate::std::Args::new(fn_args, self.into(), ctx.clone()); | ||||
|                 let mut result = func.std_lib_fn()(exec_state, args).await?; | ||||
|                 update_memory_for_tags_of_geometry(&mut result, exec_state)?; | ||||
|                 update_memory_for_tags_of_geometry(&mut result, &tag_declarator_args, exec_state)?; | ||||
|                 Ok(result) | ||||
|             } | ||||
|             FunctionKind::UserDefined => { | ||||
| @ -475,7 +518,24 @@ impl Node<CallExpression> { | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn update_memory_for_tags_of_geometry(result: &mut KclValue, exec_state: &mut ExecState) -> Result<(), KclError> { | ||||
| /// `tag_declarator_args` should only contain tag declarator literals, which | ||||
| /// will be defined as local variables.  Non-literals that evaluate to tag | ||||
| /// declarators should not be defined. | ||||
| fn update_memory_for_tags_of_geometry( | ||||
|     result: &mut KclValue, | ||||
|     tag_declarator_args: &[(TagDeclarator, SourceRange)], | ||||
|     exec_state: &mut ExecState, | ||||
| ) -> Result<(), KclError> { | ||||
|     // Define all the tags in the memory. | ||||
|     for (tag_declarator, arg_sr) in tag_declarator_args { | ||||
|         let tag = TagIdentifier { | ||||
|             value: tag_declarator.name.clone(), | ||||
|             info: None, | ||||
|             meta: vec![Metadata { source_range: *arg_sr }], | ||||
|         }; | ||||
|  | ||||
|         exec_state.memory.add_tag(&tag.value, tag.clone(), *arg_sr)?; | ||||
|     } | ||||
|     // If the return result is a sketch or solid, we want to update the | ||||
|     // memory for the tags of the group. | ||||
|     // TODO: This could probably be done in a better way, but as of now this was my only idea | ||||
| @ -483,7 +543,7 @@ fn update_memory_for_tags_of_geometry(result: &mut KclValue, exec_state: &mut Ex | ||||
|     match result { | ||||
|         KclValue::Sketch { value: ref mut sketch } => { | ||||
|             for (_, tag) in sketch.tags.iter() { | ||||
|                 exec_state.memory.update_tag(&tag.value, tag.clone())?; | ||||
|                 exec_state.memory.update_tag_if_defined(&tag.value, tag.clone()); | ||||
|             } | ||||
|         } | ||||
|         KclValue::Solid(ref mut solid) => { | ||||
| @ -521,7 +581,7 @@ fn update_memory_for_tags_of_geometry(result: &mut KclValue, exec_state: &mut Ex | ||||
|                     info.sketch = solid.id; | ||||
|                     t.info = Some(info); | ||||
|  | ||||
|                     exec_state.memory.update_tag(&tag.name, t.clone())?; | ||||
|                     exec_state.memory.update_tag_if_defined(&tag.name, t.clone()); | ||||
|  | ||||
|                     // update the sketch tags. | ||||
|                     solid.sketch.tags.insert(tag.name.clone(), t); | ||||
| @ -542,22 +602,6 @@ fn update_memory_for_tags_of_geometry(result: &mut KclValue, exec_state: &mut Ex | ||||
|     Ok(()) | ||||
| } | ||||
|  | ||||
| impl Node<TagDeclarator> { | ||||
|     pub async fn execute(&self, exec_state: &mut ExecState) -> Result<KclValue, KclError> { | ||||
|         let memory_item = KclValue::TagIdentifier(Box::new(TagIdentifier { | ||||
|             value: self.name.clone(), | ||||
|             info: None, | ||||
|             meta: vec![Metadata { | ||||
|                 source_range: self.into(), | ||||
|             }], | ||||
|         })); | ||||
|  | ||||
|         exec_state.memory.add(&self.name, memory_item.clone(), self.into())?; | ||||
|  | ||||
|         Ok(self.into()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl Node<ArrayExpression> { | ||||
|     #[async_recursion] | ||||
|     pub async fn execute(&self, exec_state: &mut ExecState, ctx: &ExecutorContext) -> Result<KclValue, KclError> { | ||||
|  | ||||
| @ -72,6 +72,10 @@ pub enum KclValue { | ||||
|     ImportedGeometry(ImportedGeometry), | ||||
|     #[ts(skip)] | ||||
|     Function { | ||||
|         /// Adam Chalmers speculation: | ||||
|         /// Reference to a KCL stdlib function (written in Rust). | ||||
|         /// Some if the KCL value is an alias of a stdlib function, | ||||
|         /// None if it's a KCL function written/declared in KCL. | ||||
|         #[serde(skip)] | ||||
|         func: Option<MemoryFunction>, | ||||
|         #[schemars(skip)] | ||||
| @ -503,4 +507,39 @@ impl KclValue { | ||||
|             .await | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// If this is a function, call it by applying keyword arguments. | ||||
|     /// If it's not a function, returns an error. | ||||
|     pub async fn call_fn_kw( | ||||
|         &self, | ||||
|         args: crate::std::Args, | ||||
|         exec_state: &mut ExecState, | ||||
|         ctx: ExecutorContext, | ||||
|         callsite: SourceRange, | ||||
|     ) -> Result<Option<KclValue>, KclError> { | ||||
|         let KclValue::Function { | ||||
|             func, | ||||
|             expression, | ||||
|             memory: closure_memory, | ||||
|             meta: _, | ||||
|         } = &self | ||||
|         else { | ||||
|             return Err(KclError::Semantic(KclErrorDetails { | ||||
|                 message: "cannot call this because it isn't a function".to_string(), | ||||
|                 source_ranges: vec![callsite], | ||||
|             })); | ||||
|         }; | ||||
|         if let Some(_func) = func { | ||||
|             todo!("Implement calling KCL stdlib fns that are aliased. Part of https://github.com/KittyCAD/modeling-app/issues/4600"); | ||||
|         } else { | ||||
|             crate::execution::call_user_defined_function_kw( | ||||
|                 args.kw_args, | ||||
|                 closure_memory.as_ref(), | ||||
|                 expression.as_ref(), | ||||
|                 exec_state, | ||||
|                 &ctx, | ||||
|             ) | ||||
|             .await | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| @ -125,10 +125,16 @@ impl ProgramMemory { | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     pub fn update_tag(&mut self, tag: &str, value: TagIdentifier) -> Result<(), KclError> { | ||||
|         self.environments[self.current_env.index()].insert(tag.to_string(), KclValue::TagIdentifier(Box::new(value))); | ||||
|     pub fn add_tag(&mut self, tag: &str, value: TagIdentifier, source_range: SourceRange) -> Result<(), KclError> { | ||||
|         self.add(tag, KclValue::TagIdentifier(Box::new(value)), source_range) | ||||
|     } | ||||
|  | ||||
|         Ok(()) | ||||
|     pub fn update_tag_if_defined(&mut self, tag: &str, value: TagIdentifier) { | ||||
|         if !self.environments[self.current_env.index()].contains_key(tag) { | ||||
|             // Do nothing if the tag isn't defined. | ||||
|             return; | ||||
|         } | ||||
|         self.environments[self.current_env.index()].insert(tag.to_string(), KclValue::TagIdentifier(Box::new(value))); | ||||
|     } | ||||
|  | ||||
|     /// Get a value from the program memory. | ||||
| @ -845,7 +851,7 @@ impl GetTangentialInfoFromPathsResult { | ||||
|  | ||||
| impl Sketch { | ||||
|     pub(crate) fn add_tag(&mut self, tag: NodeRef<'_, TagDeclarator>, current_path: &Path) { | ||||
|         let mut tag_identifier: TagIdentifier = tag.into(); | ||||
|         let mut tag_identifier = TagIdentifier::from(tag); | ||||
|         let base = current_path.get_base(); | ||||
|         tag_identifier.info = Some(TagEngineInfo { | ||||
|             id: base.geo_meta.id, | ||||
| @ -2085,7 +2091,7 @@ impl ExecutorContext { | ||||
|         let item = match init { | ||||
|             Expr::None(none) => KclValue::from(none), | ||||
|             Expr::Literal(literal) => KclValue::from(literal), | ||||
|             Expr::TagDeclarator(tag) => tag.execute(exec_state).await?, | ||||
|             Expr::TagDeclarator(tag) => KclValue::from(tag), | ||||
|             Expr::Identifier(identifier) => { | ||||
|                 let value = exec_state.memory.get(&identifier.name, identifier.into())?; | ||||
|                 value.clone() | ||||
| @ -2247,6 +2253,59 @@ fn assign_args_to_params( | ||||
|     Ok(fn_memory) | ||||
| } | ||||
|  | ||||
| fn assign_args_to_params_kw( | ||||
|     function_expression: NodeRef<'_, FunctionExpression>, | ||||
|     mut args: crate::std::args::KwArgs, | ||||
|     mut fn_memory: ProgramMemory, | ||||
| ) -> Result<ProgramMemory, KclError> { | ||||
|     // Add the arguments to the memory.  A new call frame should have already | ||||
|     // been created. | ||||
|     let source_ranges = vec![function_expression.into()]; | ||||
|     for param in function_expression.params.iter() { | ||||
|         if param.labeled { | ||||
|             let arg = args.labeled.get(¶m.identifier.name); | ||||
|             let arg_val = match arg { | ||||
|                 Some(arg) => arg.value.clone(), | ||||
|                 None => match param.default_value { | ||||
|                     Some(ref default_val) => KclValue::from(default_val.clone()), | ||||
|                     None => { | ||||
|                         return Err(KclError::Semantic(KclErrorDetails { | ||||
|                             source_ranges, | ||||
|                             message: format!( | ||||
|                                 "This function requires a parameter {}, but you haven't passed it one.", | ||||
|                                 param.identifier.name | ||||
|                             ), | ||||
|                         })); | ||||
|                     } | ||||
|                 }, | ||||
|             }; | ||||
|             fn_memory.add(¶m.identifier.name, arg_val, (¶m.identifier).into())?; | ||||
|         } else { | ||||
|             let Some(unlabeled) = args.unlabeled.take() else { | ||||
|                 let param_name = ¶m.identifier.name; | ||||
|                 return Err(if args.labeled.contains_key(param_name) { | ||||
|                     KclError::Semantic(KclErrorDetails { | ||||
|                         source_ranges, | ||||
|                         message: format!("The function does declare a parameter named '{param_name}', but this parameter doesn't use a label. Try removing the `{param_name}:`"), | ||||
|                     }) | ||||
|                 } else { | ||||
|                     KclError::Semantic(KclErrorDetails { | ||||
|                         source_ranges, | ||||
|                         message: "This function expects an unlabeled first parameter, but you haven't passed it one." | ||||
|                             .to_owned(), | ||||
|                     }) | ||||
|                 }); | ||||
|             }; | ||||
|             fn_memory.add( | ||||
|                 ¶m.identifier.name, | ||||
|                 unlabeled.value.clone(), | ||||
|                 (¶m.identifier).into(), | ||||
|             )?; | ||||
|         } | ||||
|     } | ||||
|     Ok(fn_memory) | ||||
| } | ||||
|  | ||||
| pub(crate) async fn call_user_defined_function( | ||||
|     args: Vec<Arg>, | ||||
|     memory: &ProgramMemory, | ||||
| @ -2277,6 +2336,36 @@ pub(crate) async fn call_user_defined_function( | ||||
|     result.map(|_| fn_memory.return_) | ||||
| } | ||||
|  | ||||
| pub(crate) async fn call_user_defined_function_kw( | ||||
|     args: crate::std::args::KwArgs, | ||||
|     memory: &ProgramMemory, | ||||
|     function_expression: NodeRef<'_, FunctionExpression>, | ||||
|     exec_state: &mut ExecState, | ||||
|     ctx: &ExecutorContext, | ||||
| ) -> Result<Option<KclValue>, KclError> { | ||||
|     // Create a new environment to execute the function body in so that local | ||||
|     // variables shadow variables in the parent scope.  The new environment's | ||||
|     // parent should be the environment of the closure. | ||||
|     let mut body_memory = memory.clone(); | ||||
|     let body_env = body_memory.new_env_for_call(memory.current_env); | ||||
|     body_memory.current_env = body_env; | ||||
|     let fn_memory = assign_args_to_params_kw(function_expression, args, body_memory)?; | ||||
|  | ||||
|     // Execute the function body using the memory we just created. | ||||
|     let (result, fn_memory) = { | ||||
|         let previous_memory = std::mem::replace(&mut exec_state.memory, fn_memory); | ||||
|         let result = ctx | ||||
|             .inner_execute(&function_expression.body, exec_state, BodyType::Block) | ||||
|             .await; | ||||
|         // Restore the previous memory. | ||||
|         let fn_memory = std::mem::replace(&mut exec_state.memory, previous_memory); | ||||
|  | ||||
|         (result, fn_memory) | ||||
|     }; | ||||
|  | ||||
|     result.map(|_| fn_memory.return_) | ||||
| } | ||||
|  | ||||
| pub enum StatementKind<'a> { | ||||
|     Declaration { name: &'a str }, | ||||
|     Expression, | ||||
| @ -2888,8 +2977,10 @@ let notTagDeclarator = !myTagDeclarator"; | ||||
|         ); | ||||
|  | ||||
|         let code9 = " | ||||
| let myTagDeclarator = $myTag | ||||
| let notTagIdentifier = !myTag"; | ||||
| sk = startSketchOn('XY') | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line([5, 0], %, $myTag) | ||||
| notTagIdentifier = !myTag"; | ||||
|         let tag_identifier_err = parse_execute(code9).await.unwrap_err().downcast::<KclError>().unwrap(); | ||||
|         // These are currently printed out as JSON objects, so we don't want to | ||||
|         // check the full error. | ||||
|  | ||||
| @ -138,7 +138,7 @@ pub use lsp::test_util::kcl_lsp_server; | ||||
| impl Program { | ||||
|     pub fn parse(input: &str) -> Result<(Option<Program>, Vec<CompilationError>), KclError> { | ||||
|         let module_id = ModuleId::default(); | ||||
|         let tokens = parsing::token::lexer(input, module_id)?; | ||||
|         let tokens = parsing::token::lex(input, module_id)?; | ||||
|         let (ast, errs) = parsing::parse_tokens(tokens).0?; | ||||
|  | ||||
|         Ok((ast.map(|ast| Program { ast }), errs)) | ||||
| @ -146,7 +146,7 @@ impl Program { | ||||
|  | ||||
|     pub fn parse_no_errs(input: &str) -> Result<Program, KclError> { | ||||
|         let module_id = ModuleId::default(); | ||||
|         let tokens = parsing::token::lexer(input, module_id)?; | ||||
|         let tokens = parsing::token::lex(input, module_id)?; | ||||
|         let ast = parsing::parse_tokens(tokens).parse_errs_as_err()?; | ||||
|  | ||||
|         Ok(Program { ast }) | ||||
|  | ||||
| @ -49,34 +49,31 @@ use crate::{ | ||||
|             cache::{CacheInformation, OldAstState}, | ||||
|             types::{Expr, Node, VariableKind}, | ||||
|         }, | ||||
|         token::TokenType, | ||||
|         token::TokenStream, | ||||
|         PIPE_OPERATOR, | ||||
|     }, | ||||
|     ModuleId, Program, SourceRange, | ||||
| }; | ||||
|  | ||||
| lazy_static::lazy_static! { | ||||
|     pub static ref SEMANTIC_TOKEN_TYPES: Vec<SemanticTokenType> = { | ||||
|         // This is safe to unwrap because we know all the token types are valid. | ||||
|         // And the test would fail if they were not. | ||||
|         let mut gen = TokenType::all_semantic_token_types().unwrap(); | ||||
|         gen.extend(vec![ | ||||
| const SEMANTIC_TOKEN_TYPES: [SemanticTokenType; 10] = [ | ||||
|     SemanticTokenType::NUMBER, | ||||
|     SemanticTokenType::VARIABLE, | ||||
|     SemanticTokenType::KEYWORD, | ||||
|     SemanticTokenType::TYPE, | ||||
|     SemanticTokenType::STRING, | ||||
|     SemanticTokenType::OPERATOR, | ||||
|     SemanticTokenType::COMMENT, | ||||
|     SemanticTokenType::FUNCTION, | ||||
|     SemanticTokenType::PARAMETER, | ||||
|     SemanticTokenType::PROPERTY, | ||||
|         ]); | ||||
|         gen | ||||
|     }; | ||||
| ]; | ||||
|  | ||||
|     pub static ref SEMANTIC_TOKEN_MODIFIERS: Vec<SemanticTokenModifier> = { | ||||
|         vec![ | ||||
| const SEMANTIC_TOKEN_MODIFIERS: [SemanticTokenModifier; 5] = [ | ||||
|     SemanticTokenModifier::DECLARATION, | ||||
|     SemanticTokenModifier::DEFINITION, | ||||
|     SemanticTokenModifier::DEFAULT_LIBRARY, | ||||
|     SemanticTokenModifier::READONLY, | ||||
|     SemanticTokenModifier::STATIC, | ||||
|         ] | ||||
|     }; | ||||
| } | ||||
| ]; | ||||
|  | ||||
| /// A subcommand for running the server. | ||||
| #[derive(Clone, Debug)] | ||||
| @ -105,7 +102,7 @@ pub struct Backend { | ||||
|     /// The stdlib signatures for the language. | ||||
|     pub stdlib_signatures: HashMap<String, SignatureHelp>, | ||||
|     /// Token maps. | ||||
|     pub token_map: DashMap<String, Vec<crate::parsing::token::Token>>, | ||||
|     pub(super) token_map: DashMap<String, TokenStream>, | ||||
|     /// AST maps. | ||||
|     pub ast_map: DashMap<String, Node<crate::parsing::ast::types::Program>>, | ||||
|     /// Last successful execution. | ||||
| @ -284,7 +281,7 @@ impl crate::lsp::backend::Backend for Backend { | ||||
|  | ||||
|         // Lets update the tokens. | ||||
|         let module_id = ModuleId::default(); | ||||
|         let tokens = match crate::parsing::token::lexer(¶ms.text, module_id) { | ||||
|         let tokens = match crate::parsing::token::lex(¶ms.text, module_id) { | ||||
|             Ok(tokens) => tokens, | ||||
|             Err(err) => { | ||||
|                 self.add_to_diagnostics(¶ms, &[err], true).await; | ||||
| @ -410,11 +407,11 @@ impl Backend { | ||||
|         self.executor_ctx.read().await | ||||
|     } | ||||
|  | ||||
|     async fn update_semantic_tokens(&self, tokens: &[crate::parsing::token::Token], params: &TextDocumentItem) { | ||||
|     async fn update_semantic_tokens(&self, tokens: &TokenStream, params: &TextDocumentItem) { | ||||
|         // Update the semantic tokens map. | ||||
|         let mut semantic_tokens = vec![]; | ||||
|         let mut last_position = Position::new(0, 0); | ||||
|         for token in tokens { | ||||
|         for token in tokens.as_slice() { | ||||
|             let Ok(token_type) = SemanticTokenType::try_from(token.token_type) else { | ||||
|                 // We continue here because not all tokens can be converted this way, we will get | ||||
|                 // the rest from the ast. | ||||
| @ -444,8 +441,11 @@ impl Backend { | ||||
|             let token_modifiers_bitset = if let Some(ast) = self.ast_map.get(params.uri.as_str()) { | ||||
|                 let token_index = Arc::new(Mutex::new(token_type_index)); | ||||
|                 let modifier_index: Arc<Mutex<u32>> = Arc::new(Mutex::new(0)); | ||||
|                 crate::walk::walk(&ast, &|node: crate::walk::Node| { | ||||
|                     let node_range: SourceRange = (&node).into(); | ||||
|                 crate::walk::walk(&ast, |node: crate::walk::Node| { | ||||
|                     let Ok(node_range): Result<SourceRange, _> = (&node).try_into() else { | ||||
|                         return Ok(true); | ||||
|                     }; | ||||
|  | ||||
|                     if !node_range.contains(source_range.start()) { | ||||
|                         return Ok(true); | ||||
|                     } | ||||
| @ -563,7 +563,7 @@ impl Backend { | ||||
|                     let semantic_token = SemanticToken { | ||||
|                         delta_line: position.line - last_position.line + 1, | ||||
|                         delta_start: 0, | ||||
|                         length: token.value.len() as u32, | ||||
|                         length: (token.end - token.start) as u32, | ||||
|                         token_type: token_type_index, | ||||
|                         token_modifiers_bitset, | ||||
|                     }; | ||||
| @ -582,7 +582,7 @@ impl Backend { | ||||
|                 } else { | ||||
|                     position.character - last_position.character | ||||
|                 }, | ||||
|                 length: token.value.len() as u32, | ||||
|                 length: (token.end - token.start) as u32, | ||||
|                 token_type: token_type_index, | ||||
|                 token_modifiers_bitset, | ||||
|             }; | ||||
| @ -963,8 +963,8 @@ impl LanguageServer for Backend { | ||||
|                         semantic_tokens_options: SemanticTokensOptions { | ||||
|                             work_done_progress_options: WorkDoneProgressOptions::default(), | ||||
|                             legend: SemanticTokensLegend { | ||||
|                                 token_types: SEMANTIC_TOKEN_TYPES.clone(), | ||||
|                                 token_modifiers: SEMANTIC_TOKEN_MODIFIERS.clone(), | ||||
|                                 token_types: SEMANTIC_TOKEN_TYPES.to_vec(), | ||||
|                                 token_modifiers: SEMANTIC_TOKEN_MODIFIERS.to_vec(), | ||||
|                             }, | ||||
|                             range: Some(false), | ||||
|                             full: Some(SemanticTokensFullOptions::Bool(true)), | ||||
|  | ||||
| @ -645,7 +645,7 @@ async fn test_kcl_lsp_completions() { | ||||
|                 uri: "file:///test.kcl".try_into().unwrap(), | ||||
|                 language_id: "kcl".to_string(), | ||||
|                 version: 1, | ||||
|                 text: r#"const thing= 1 | ||||
|                 text: r#"thing= 1 | ||||
| st"# | ||||
|                 .to_string(), | ||||
|             }, | ||||
| @ -688,7 +688,7 @@ async fn test_kcl_lsp_completions_empty_in_comment() { | ||||
|                 uri: "file:///test.kcl".try_into().unwrap(), | ||||
|                 language_id: "kcl".to_string(), | ||||
|                 version: 1, | ||||
|                 text: r#"const thing= 1 // st"#.to_string(), | ||||
|                 text: r#"thing= 1 // st"#.to_string(), | ||||
|             }, | ||||
|         }) | ||||
|         .await; | ||||
| @ -700,7 +700,7 @@ async fn test_kcl_lsp_completions_empty_in_comment() { | ||||
|                 text_document: tower_lsp::lsp_types::TextDocumentIdentifier { | ||||
|                     uri: "file:///test.kcl".try_into().unwrap(), | ||||
|                 }, | ||||
|                 position: tower_lsp::lsp_types::Position { line: 0, character: 19 }, | ||||
|                 position: tower_lsp::lsp_types::Position { line: 0, character: 13 }, | ||||
|             }, | ||||
|             context: None, | ||||
|             partial_result_params: Default::default(), | ||||
| @ -723,7 +723,7 @@ async fn test_kcl_lsp_completions_tags() { | ||||
|                 uri: "file:///test.kcl".try_into().unwrap(), | ||||
|                 language_id: "kcl".to_string(), | ||||
|                 version: 1, | ||||
|                 text: r#"const part001 = startSketchOn('XY') | ||||
|                 text: r#"part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([11.19, 28.35], %) | ||||
|   |> line([28.67, -13.25], %, $here) | ||||
|   |> line([-4.12, -22.81], %) | ||||
| @ -1058,7 +1058,7 @@ async fn test_kcl_lsp_semantic_tokens_with_modifiers() { | ||||
|                 uri: "file:///test.kcl".try_into().unwrap(), | ||||
|                 language_id: "kcl".to_string(), | ||||
|                 version: 1, | ||||
|                 text: r#"const part001 = startSketchOn('XY') | ||||
|                 text: r#"part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %, $seg01) | ||||
| @ -1066,8 +1066,8 @@ async fn test_kcl_lsp_semantic_tokens_with_modifiers() { | ||||
|   |> close(%) | ||||
|   |> extrude(3.14, %) | ||||
|  | ||||
| const thing = {blah: "foo"} | ||||
| const bar = thing.blah | ||||
| thing = {blah: "foo"} | ||||
| bar = thing.blah | ||||
|  | ||||
| fn myFn = (param1) => { | ||||
|     return param1 | ||||
| @ -1082,7 +1082,7 @@ fn myFn = (param1) => { | ||||
|  | ||||
|     // Get the token map. | ||||
|     let token_map = server.token_map.get("file:///test.kcl").unwrap().clone(); | ||||
|     assert!(token_map != vec![]); | ||||
|     assert!(!token_map.is_empty()); | ||||
|  | ||||
|     // Get the ast. | ||||
|     let ast = server.ast_map.get("file:///test.kcl").unwrap().clone(); | ||||
| @ -1230,7 +1230,7 @@ async fn test_kcl_lsp_semantic_tokens_multiple_comments() { | ||||
| // 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.  | ||||
|  | ||||
| // Define constants like ball diameter, inside diameter, overhange length, and thickness | ||||
| const sphereDia = 0.5"# | ||||
| sphereDia = 0.5"# | ||||
|                     .to_string(), | ||||
|             }, | ||||
|         }) | ||||
| @ -1251,7 +1251,7 @@ const sphereDia = 0.5"# | ||||
|  | ||||
|     // Check the semantic tokens. | ||||
|     if let tower_lsp::lsp_types::SemanticTokensResult::Tokens(semantic_tokens) = semantic_tokens { | ||||
|         assert_eq!(semantic_tokens.data.len(), 7); | ||||
|         assert_eq!(semantic_tokens.data.len(), 6); | ||||
|         assert_eq!(semantic_tokens.data[0].length, 15); | ||||
|         assert_eq!(semantic_tokens.data[0].delta_start, 0); | ||||
|         assert_eq!(semantic_tokens.data[0].delta_line, 0); | ||||
| @ -1279,36 +1279,27 @@ const sphereDia = 0.5"# | ||||
|                 .get_semantic_token_type_index(&SemanticTokenType::COMMENT) | ||||
|                 .unwrap() | ||||
|         ); | ||||
|         assert_eq!(semantic_tokens.data[3].length, 5); | ||||
|         assert_eq!(semantic_tokens.data[3].length, 9); | ||||
|         assert_eq!(semantic_tokens.data[3].delta_start, 0); | ||||
|         assert_eq!(semantic_tokens.data[3].delta_line, 1); | ||||
|         assert_eq!( | ||||
|             semantic_tokens.data[3].token_type, | ||||
|             server | ||||
|                 .get_semantic_token_type_index(&SemanticTokenType::KEYWORD) | ||||
|                 .unwrap() | ||||
|         ); | ||||
|         assert_eq!(semantic_tokens.data[4].length, 9); | ||||
|         assert_eq!(semantic_tokens.data[4].delta_start, 6); | ||||
|         assert_eq!(semantic_tokens.data[4].delta_line, 0); | ||||
|         assert_eq!( | ||||
|             semantic_tokens.data[4].token_type, | ||||
|             server | ||||
|                 .get_semantic_token_type_index(&SemanticTokenType::VARIABLE) | ||||
|                 .unwrap() | ||||
|         ); | ||||
|         assert_eq!(semantic_tokens.data[5].length, 1); | ||||
|         assert_eq!(semantic_tokens.data[5].delta_start, 10); | ||||
|         assert_eq!(semantic_tokens.data[4].length, 1); | ||||
|         assert_eq!(semantic_tokens.data[4].delta_start, 10); | ||||
|         assert_eq!( | ||||
|             semantic_tokens.data[5].token_type, | ||||
|             semantic_tokens.data[4].token_type, | ||||
|             server | ||||
|                 .get_semantic_token_type_index(&SemanticTokenType::OPERATOR) | ||||
|                 .unwrap() | ||||
|         ); | ||||
|         assert_eq!(semantic_tokens.data[6].length, 3); | ||||
|         assert_eq!(semantic_tokens.data[6].delta_start, 2); | ||||
|         assert_eq!(semantic_tokens.data[5].length, 3); | ||||
|         assert_eq!(semantic_tokens.data[5].delta_start, 2); | ||||
|         assert_eq!( | ||||
|             semantic_tokens.data[6].token_type, | ||||
|             semantic_tokens.data[5].token_type, | ||||
|             server | ||||
|                 .get_semantic_token_type_index(&SemanticTokenType::NUMBER) | ||||
|                 .unwrap() | ||||
| @ -1329,7 +1320,7 @@ async fn test_kcl_lsp_document_symbol() { | ||||
|                 uri: "file:///test.kcl".try_into().unwrap(), | ||||
|                 language_id: "kcl".to_string(), | ||||
|                 version: 1, | ||||
|                 text: r#"const myVar = 1 | ||||
|                 text: r#"myVar = 1 | ||||
| startSketchOn('XY')"# | ||||
|                     .to_string(), | ||||
|             }, | ||||
| @ -1369,7 +1360,7 @@ async fn test_kcl_lsp_document_symbol_tag() { | ||||
|                 uri: "file:///test.kcl".try_into().unwrap(), | ||||
|                 language_id: "kcl".to_string(), | ||||
|                 version: 1, | ||||
|                 text: r#"const part001 = startSketchOn('XY') | ||||
|                 text: r#"part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([11.19, 28.35], %) | ||||
|   |> line([28.67, -13.25], %, $here) | ||||
|   |> line([-4.12, -22.81], %) | ||||
| @ -1466,13 +1457,13 @@ async fn test_kcl_lsp_formatting_extra_parens() { | ||||
| // 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.  | ||||
|  | ||||
| // Define constants like ball diameter, inside diameter, overhange length, and thickness | ||||
| const sphereDia = 0.5 | ||||
| const insideDia = 1 | ||||
| const thickness = 0.25 | ||||
| const overHangLength = .4 | ||||
| sphereDia = 0.5 | ||||
| insideDia = 1 | ||||
| thickness = 0.25 | ||||
| overHangLength = .4 | ||||
|  | ||||
| // Sketch and revolve the inside bearing piece | ||||
| const insideRevolve = startSketchOn('XZ') | ||||
| insideRevolve = startSketchOn('XZ') | ||||
|   |> startProfileAt([insideDia / 2, 0], %) | ||||
|   |> line([0, thickness + sphereDia / 2], %) | ||||
|   |> line([overHangLength, 0], %) | ||||
| @ -1486,7 +1477,7 @@ const insideRevolve = startSketchOn('XZ') | ||||
|   |> revolve({ axis: 'y' }, %) | ||||
|  | ||||
| // Sketch and revolve one of the balls and duplicate it using a circular pattern. (This is currently a workaround, we have a bug with rotating on a sketch that touches the rotation axis) | ||||
| const sphere = startSketchOn('XZ') | ||||
| sphere = startSketchOn('XZ') | ||||
|   |> startProfileAt([ | ||||
|        0.05 + insideDia / 2 + thickness, | ||||
|        0 - 0.05 | ||||
| @ -1508,7 +1499,7 @@ const sphere = startSketchOn('XZ') | ||||
|      }, %) | ||||
|  | ||||
| // Sketch and revolve the outside bearing | ||||
| const outsideRevolve = startSketchOn('XZ') | ||||
| outsideRevolve = startSketchOn('XZ') | ||||
|   |> startProfileAt([ | ||||
|        insideDia / 2 + thickness + sphereDia, | ||||
|        0 | ||||
| @ -1638,7 +1629,7 @@ async fn test_kcl_lsp_rename() { | ||||
|                 uri: "file:///test.kcl".try_into().unwrap(), | ||||
|                 language_id: "kcl".to_string(), | ||||
|                 version: 1, | ||||
|                 text: r#"const thing= 1"#.to_string(), | ||||
|                 text: r#"thing= 1"#.to_string(), | ||||
|             }, | ||||
|         }) | ||||
|         .await; | ||||
| @ -1650,7 +1641,7 @@ async fn test_kcl_lsp_rename() { | ||||
|                 text_document: tower_lsp::lsp_types::TextDocumentIdentifier { | ||||
|                     uri: "file:///test.kcl".try_into().unwrap(), | ||||
|                 }, | ||||
|                 position: tower_lsp::lsp_types::Position { line: 0, character: 8 }, | ||||
|                 position: tower_lsp::lsp_types::Position { line: 0, character: 2 }, | ||||
|             }, | ||||
|             new_name: "newName".to_string(), | ||||
|             work_done_progress_params: Default::default(), | ||||
| @ -1667,7 +1658,7 @@ async fn test_kcl_lsp_rename() { | ||||
|         vec![tower_lsp::lsp_types::TextEdit { | ||||
|             range: tower_lsp::lsp_types::Range { | ||||
|                 start: tower_lsp::lsp_types::Position { line: 0, character: 0 }, | ||||
|                 end: tower_lsp::lsp_types::Position { line: 0, character: 13 } | ||||
|                 end: tower_lsp::lsp_types::Position { line: 0, character: 7 } | ||||
|             }, | ||||
|             new_text: "newName = 1\n".to_string() | ||||
|         }] | ||||
| @ -1773,7 +1764,7 @@ async fn test_kcl_lsp_diagnostic_has_lints() { | ||||
|                 uri: "file:///testlint.kcl".try_into().unwrap(), | ||||
|                 language_id: "kcl".to_string(), | ||||
|                 version: 1, | ||||
|                 text: r#"let THING = 10"#.to_string(), | ||||
|                 text: r#"THING = 10"#.to_string(), | ||||
|             }, | ||||
|         }) | ||||
|         .await; | ||||
| @ -1859,7 +1850,7 @@ async fn test_copilot_lsp_completions_raw() { | ||||
|     let completions = server | ||||
|         .get_completions( | ||||
|             "kcl".to_string(), | ||||
|             r#"const bracket = startSketchOn('XY') | ||||
|             r#"bracket = startSketchOn('XY') | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   "# | ||||
|             .to_string(), | ||||
| @ -1878,7 +1869,7 @@ async fn test_copilot_lsp_completions_raw() { | ||||
|     let completions_hit_cache = server | ||||
|         .get_completions( | ||||
|             "kcl".to_string(), | ||||
|             r#"const bracket = startSketchOn('XY') | ||||
|             r#"bracket = startSketchOn('XY') | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   "# | ||||
|             .to_string(), | ||||
| @ -1918,7 +1909,7 @@ async fn test_copilot_lsp_completions() { | ||||
|             path: "file:///test.copilot".to_string(), | ||||
|             position: crate::lsp::copilot::types::CopilotPosition { line: 3, character: 3 }, | ||||
|             relative_path: "test.copilot".to_string(), | ||||
|             source: r#"const bracket = startSketchOn('XY') | ||||
|             source: r#"bracket = startSketchOn('XY') | ||||
|   |> startProfileAt([0, 0], %) | ||||
|    | ||||
|   |> close(%) | ||||
| @ -2066,7 +2057,7 @@ async fn test_lsp_initialized() { | ||||
| async fn test_kcl_lsp_on_change_update_ast() { | ||||
|     let server = kcl_lsp_server(false).await.unwrap(); | ||||
|  | ||||
|     let same_text = r#"const thing = 1"#.to_string(); | ||||
|     let same_text = r#"thing = 1"#.to_string(); | ||||
|  | ||||
|     // Send open file. | ||||
|     server | ||||
| @ -2102,7 +2093,7 @@ async fn test_kcl_lsp_on_change_update_ast() { | ||||
|     assert_eq!(ast, server.ast_map.get("file:///test.kcl").unwrap().clone()); | ||||
|  | ||||
|     // Update the text. | ||||
|     let new_text = r#"const thing = 2"#.to_string(); | ||||
|     let new_text = r#"thing = 2"#.to_string(); | ||||
|     // Send change file. | ||||
|     server | ||||
|         .did_change(tower_lsp::lsp_types::DidChangeTextDocumentParams { | ||||
| @ -2128,7 +2119,7 @@ async fn test_kcl_lsp_on_change_update_ast() { | ||||
| async fn kcl_test_kcl_lsp_on_change_update_memory() { | ||||
|     let server = kcl_lsp_server(true).await.unwrap(); | ||||
|  | ||||
|     let same_text = r#"const thing = 1"#.to_string(); | ||||
|     let same_text = r#"thing = 1"#.to_string(); | ||||
|  | ||||
|     // Send open file. | ||||
|     server | ||||
| @ -2164,7 +2155,7 @@ async fn kcl_test_kcl_lsp_on_change_update_memory() { | ||||
|     assert_eq!(memory, server.memory_map.get("file:///test.kcl").unwrap().clone()); | ||||
|  | ||||
|     // Update the text. | ||||
|     let new_text = r#"const thing = 2"#.to_string(); | ||||
|     let new_text = r#"thing = 2"#.to_string(); | ||||
|     // Send change file. | ||||
|     server | ||||
|         .did_change(tower_lsp::lsp_types::DidChangeTextDocumentParams { | ||||
| @ -2188,7 +2179,7 @@ async fn kcl_test_kcl_lsp_update_units() { | ||||
|     let server = kcl_lsp_server(true).await.unwrap(); | ||||
|  | ||||
|     let same_text = r#"fn cube = (pos, scale) => { | ||||
|   const sg = startSketchOn('XY') | ||||
|   sg = startSketchOn('XY') | ||||
|     |> startProfileAt(pos, %) | ||||
|     |> line([0, scale], %) | ||||
|     |> line([scale, 0], %) | ||||
| @ -2196,7 +2187,7 @@ async fn kcl_test_kcl_lsp_update_units() { | ||||
|  | ||||
|   return sg | ||||
| } | ||||
| const part001 = cube([0,0], 20) | ||||
| part001 = cube([0,0], 20) | ||||
|     |> close(%) | ||||
|     |> extrude(20, %)"# | ||||
|         .to_string(); | ||||
| @ -2215,7 +2206,7 @@ const part001 = cube([0,0], 20) | ||||
|  | ||||
|     // Get the tokens. | ||||
|     let tokens = server.token_map.get("file:///test.kcl").unwrap().clone(); | ||||
|     assert_eq!(tokens.len(), 124); | ||||
|     assert_eq!(tokens.as_slice().len(), 120); | ||||
|  | ||||
|     // Get the ast. | ||||
|     let ast = server.ast_map.get("file:///test.kcl").unwrap().clone(); | ||||
| @ -2305,7 +2296,7 @@ async fn test_kcl_lsp_diagnostics_on_parse_error() { | ||||
|     assert_diagnostic_count(server.diagnostics_map.get("file:///test.kcl").as_deref(), 1); | ||||
|  | ||||
|     // Update the text. | ||||
|     let new_text = r#"const thing = 2"#.to_string(); | ||||
|     let new_text = r#"thing = 2"#.to_string(); | ||||
|     // Send change file. | ||||
|     server | ||||
|         .did_change(tower_lsp::lsp_types::DidChangeTextDocumentParams { | ||||
| @ -2336,7 +2327,7 @@ async fn kcl_test_kcl_lsp_diagnostics_on_execution_error() { | ||||
|                 uri: "file:///test.kcl".try_into().unwrap(), | ||||
|                 language_id: "kcl".to_string(), | ||||
|                 version: 1, | ||||
|                 text: r#"const part001 = startSketchOn('XY') | ||||
|                 text: r#"part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
| @ -2356,7 +2347,7 @@ async fn kcl_test_kcl_lsp_diagnostics_on_execution_error() { | ||||
|     assert_diagnostic_count(server.diagnostics_map.get("file:///test.kcl").as_deref(), 1); | ||||
|  | ||||
|     // Update the text. | ||||
|     let new_text = r#"const part001 = startSketchOn('XY') | ||||
|     let new_text = r#"part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
| @ -2394,7 +2385,7 @@ async fn kcl_test_kcl_lsp_full_to_empty_file_updates_ast_and_memory() { | ||||
|                 uri: "file:///test.kcl".try_into().unwrap(), | ||||
|                 language_id: "kcl".to_string(), | ||||
|                 version: 1, | ||||
|                 text: r#"const part001 = startSketchOn('XY') | ||||
|                 text: r#"part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
| @ -2443,7 +2434,7 @@ async fn kcl_test_kcl_lsp_full_to_empty_file_updates_ast_and_memory() { | ||||
| async fn kcl_test_kcl_lsp_code_unchanged_but_has_diagnostics_reexecute() { | ||||
|     let server = kcl_lsp_server(true).await.unwrap(); | ||||
|  | ||||
|     let code = r#"const part001 = startSketchOn('XY') | ||||
|     let code = r#"part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
| @ -2536,7 +2527,7 @@ async fn kcl_test_kcl_lsp_code_unchanged_but_has_diagnostics_reexecute() { | ||||
| async fn kcl_test_kcl_lsp_code_and_ast_unchanged_but_has_diagnostics_reexecute() { | ||||
|     let server = kcl_lsp_server(true).await.unwrap(); | ||||
|  | ||||
|     let code = r#"const part001 = startSketchOn('XY') | ||||
|     let code = r#"part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
| @ -2624,7 +2615,7 @@ async fn kcl_test_kcl_lsp_code_and_ast_unchanged_but_has_diagnostics_reexecute() | ||||
| async fn kcl_test_kcl_lsp_code_and_ast_units_unchanged_but_has_diagnostics_reexecute_on_unit_change() { | ||||
|     let server = kcl_lsp_server(true).await.unwrap(); | ||||
|  | ||||
|     let code = r#"const part001 = startSketchOn('XY') | ||||
|     let code = r#"part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
| @ -2715,7 +2706,7 @@ async fn kcl_test_kcl_lsp_code_and_ast_units_unchanged_but_has_diagnostics_reexe | ||||
| async fn kcl_test_kcl_lsp_code_and_ast_units_unchanged_but_has_memory_reexecute_on_unit_change() { | ||||
|     let server = kcl_lsp_server(true).await.unwrap(); | ||||
|  | ||||
|     let code = r#"const part001 = startSketchOn('XY') | ||||
|     let code = r#"part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
| @ -2785,7 +2776,7 @@ async fn kcl_test_kcl_lsp_code_and_ast_units_unchanged_but_has_memory_reexecute_ | ||||
| async fn kcl_test_kcl_lsp_cant_execute_set() { | ||||
|     let server = kcl_lsp_server(true).await.unwrap(); | ||||
|  | ||||
|     let code = r#"const part001 = startSketchOn('XY') | ||||
|     let code = r#"part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
| @ -2982,7 +2973,7 @@ async fn test_kcl_lsp_folding() { | ||||
| async fn kcl_test_kcl_lsp_code_with_parse_error_and_ast_unchanged_but_has_diagnostics_reparse() { | ||||
|     let server = kcl_lsp_server(false).await.unwrap(); | ||||
|  | ||||
|     let code = r#"const part001 = startSketchOn('XY') | ||||
|     let code = r#"part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
| @ -3036,8 +3027,8 @@ async fn kcl_test_kcl_lsp_code_with_parse_error_and_ast_unchanged_but_has_diagno | ||||
| async fn kcl_test_kcl_lsp_code_with_lint_and_ast_unchanged_but_has_diagnostics_reparse() { | ||||
|     let server = kcl_lsp_server(false).await.unwrap(); | ||||
|  | ||||
|     let code = r#"const LINT = 1 | ||||
| const part001 = startSketchOn('XY') | ||||
|     let code = r#"LINT = 1 | ||||
| part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
| @ -3090,8 +3081,8 @@ const part001 = startSketchOn('XY') | ||||
| async fn kcl_test_kcl_lsp_code_with_lint_and_parse_error_and_ast_unchanged_but_has_diagnostics_reparse() { | ||||
|     let server = kcl_lsp_server(false).await.unwrap(); | ||||
|  | ||||
|     let code = r#"const LINT = 1 | ||||
| const part001 = startSketchOn('XY') | ||||
|     let code = r#"LINT = 1 | ||||
| part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
| @ -3145,8 +3136,8 @@ const part001 = startSketchOn('XY') | ||||
| async fn kcl_test_kcl_lsp_code_lint_and_ast_unchanged_but_has_diagnostics_reexecute() { | ||||
|     let server = kcl_lsp_server(true).await.unwrap(); | ||||
|  | ||||
|     let code = r#"const LINT = 1 | ||||
| const part001 = startSketchOn('XY') | ||||
|     let code = r#"LINT = 1 | ||||
| part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %, $seg01) | ||||
| @ -3210,8 +3201,8 @@ const part001 = startSketchOn('XY') | ||||
| async fn kcl_test_kcl_lsp_code_lint_reexecute_new_lint() { | ||||
|     let server = kcl_lsp_server(true).await.unwrap(); | ||||
|  | ||||
|     let code = r#"const LINT = 1 | ||||
| const part001 = startSketchOn('XY') | ||||
|     let code = r#"LINT = 1 | ||||
| part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %, $seg01) | ||||
| @ -3253,14 +3244,14 @@ const part001 = startSketchOn('XY') | ||||
|             content_changes: vec![tower_lsp::lsp_types::TextDocumentContentChangeEvent { | ||||
|                 range: None, | ||||
|                 range_length: None, | ||||
|                 text: r#"const part001 = startSketchOn('XY') | ||||
|                 text: r#"part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %, $seg01) | ||||
|   |> line([-20, 0], %, $seg01) | ||||
|   |> close(%) | ||||
|   |> extrude(3.14, %) | ||||
| const NEW_LINT = 1"# | ||||
| NEW_LINT = 1"# | ||||
|                     .to_string(), | ||||
|             }], | ||||
|         }) | ||||
| @ -3283,8 +3274,8 @@ const NEW_LINT = 1"# | ||||
| async fn kcl_test_kcl_lsp_code_lint_reexecute_new_ast_error() { | ||||
|     let server = kcl_lsp_server(true).await.unwrap(); | ||||
|  | ||||
|     let code = r#"const LINT = 1 | ||||
| const part001 = startSketchOn('XY') | ||||
|     let code = r#"LINT = 1 | ||||
| part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %, $seg01) | ||||
| @ -3326,14 +3317,14 @@ const part001 = startSketchOn('XY') | ||||
|             content_changes: vec![tower_lsp::lsp_types::TextDocumentContentChangeEvent { | ||||
|                 range: None, | ||||
|                 range_length: None, | ||||
|                 text: r#"const part001 = startSketchOn('XY') | ||||
|                 text: r#"part001 = startSketchOn('XY') | ||||
|   |> ^^^^startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %, $seg01) | ||||
|   |> line([-20, 0], %, $seg01) | ||||
|   |> close(%) | ||||
|   |> extrude(3.14, %) | ||||
| const NEW_LINT = 1"# | ||||
| NEW_LINT = 1"# | ||||
|                     .to_string(), | ||||
|             }], | ||||
|         }) | ||||
| @ -3356,8 +3347,8 @@ const NEW_LINT = 1"# | ||||
| async fn kcl_test_kcl_lsp_code_lint_reexecute_had_lint_new_parse_error() { | ||||
|     let server = kcl_lsp_server(true).await.unwrap(); | ||||
|  | ||||
|     let code = r#"const LINT = 1 | ||||
| const part001 = startSketchOn('XY') | ||||
|     let code = r#"LINT = 1 | ||||
| part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
| @ -3388,11 +3379,11 @@ const part001 = startSketchOn('XY') | ||||
|  | ||||
|     // Get the symbols map. | ||||
|     let symbols_map = server.symbols_map.get("file:///test.kcl").unwrap().clone(); | ||||
|     assert!(symbols_map != vec![]); | ||||
|     assert!(!symbols_map.is_empty()); | ||||
|  | ||||
|     // Get the semantic tokens map. | ||||
|     let semantic_tokens_map = server.semantic_tokens_map.get("file:///test.kcl").unwrap().clone(); | ||||
|     assert!(semantic_tokens_map != vec![]); | ||||
|     assert!(!semantic_tokens_map.is_empty()); | ||||
|  | ||||
|     // Get the memory. | ||||
|     let memory = server.memory_map.get("file:///test.kcl").unwrap().clone(); | ||||
| @ -3408,14 +3399,14 @@ const part001 = startSketchOn('XY') | ||||
|             content_changes: vec![tower_lsp::lsp_types::TextDocumentContentChangeEvent { | ||||
|                 range: None, | ||||
|                 range_length: None, | ||||
|                 text: r#"const part001 = startSketchOn('XY') | ||||
|                 text: r#"part001 = startSketchOn('XY') | ||||
|   |> ^^^^startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
|   |> line([-20, 0], %) | ||||
|   |> close(%) | ||||
|   |> extrude(3.14, %) | ||||
| const NEW_LINT = 1"# | ||||
| NEW_LINT = 1"# | ||||
|                     .to_string(), | ||||
|             }], | ||||
|         }) | ||||
| @ -3431,7 +3422,7 @@ const NEW_LINT = 1"# | ||||
|  | ||||
|     // Get the semantic tokens map. | ||||
|     let semantic_tokens_map = server.semantic_tokens_map.get("file:///test.kcl").unwrap().clone(); | ||||
|     assert!(semantic_tokens_map != vec![]); | ||||
|     assert!(!semantic_tokens_map.is_empty()); | ||||
|  | ||||
|     // Get the memory. | ||||
|     let memory = server.memory_map.get("file:///test.kcl"); | ||||
| @ -3447,8 +3438,8 @@ const NEW_LINT = 1"# | ||||
| async fn kcl_test_kcl_lsp_code_lint_reexecute_had_lint_new_execution_error() { | ||||
|     let server = kcl_lsp_server(true).await.unwrap(); | ||||
|  | ||||
|     let code = r#"const LINT = 1 | ||||
| const part001 = startSketchOn('XY') | ||||
|     let code = r#"LINT = 1 | ||||
| part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
| @ -3475,7 +3466,7 @@ const part001 = startSketchOn('XY') | ||||
|  | ||||
|     // Get the token map. | ||||
|     let token_map = server.token_map.get("file:///test.kcl").unwrap().clone(); | ||||
|     assert!(token_map != vec![]); | ||||
|     assert!(!token_map.is_empty()); | ||||
|  | ||||
|     // Get the ast. | ||||
|     let ast = server.ast_map.get("file:///test.kcl").unwrap().clone(); | ||||
| @ -3483,11 +3474,11 @@ const part001 = startSketchOn('XY') | ||||
|  | ||||
|     // Get the symbols map. | ||||
|     let symbols_map = server.symbols_map.get("file:///test.kcl").unwrap().clone(); | ||||
|     assert!(symbols_map != vec![]); | ||||
|     assert!(!symbols_map.is_empty()); | ||||
|  | ||||
|     // Get the semantic tokens map. | ||||
|     let semantic_tokens_map = server.semantic_tokens_map.get("file:///test.kcl").unwrap().clone(); | ||||
|     assert!(semantic_tokens_map != vec![]); | ||||
|     assert!(!semantic_tokens_map.is_empty()); | ||||
|  | ||||
|     // Get the memory. | ||||
|     let memory = server.memory_map.get("file:///test.kcl").unwrap().clone(); | ||||
| @ -3503,8 +3494,8 @@ const part001 = startSketchOn('XY') | ||||
|             content_changes: vec![tower_lsp::lsp_types::TextDocumentContentChangeEvent { | ||||
|                 range: None, | ||||
|                 range_length: None, | ||||
|                 text: r#"const LINT = 1 | ||||
| const part001 = startSketchOn('XY') | ||||
|                 text: r#"LINT = 1 | ||||
| part001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %, $seg01) | ||||
|   |> line([0, 20], %, $seg01) | ||||
| @ -3518,7 +3509,7 @@ const part001 = startSketchOn('XY') | ||||
|  | ||||
|     // Get the token map. | ||||
|     let token_map = server.token_map.get("file:///test.kcl").unwrap().clone(); | ||||
|     assert!(token_map != vec![]); | ||||
|     assert!(!token_map.is_empty()); | ||||
|  | ||||
|     // Get the ast. | ||||
|     let ast = server.ast_map.get("file:///test.kcl").unwrap().clone(); | ||||
| @ -3526,11 +3517,11 @@ const part001 = startSketchOn('XY') | ||||
|  | ||||
|     // Get the symbols map. | ||||
|     let symbols_map = server.symbols_map.get("file:///test.kcl").unwrap().clone(); | ||||
|     assert!(symbols_map != vec![]); | ||||
|     assert!(!symbols_map.is_empty()); | ||||
|  | ||||
|     // Get the semantic tokens map. | ||||
|     let semantic_tokens_map = server.semantic_tokens_map.get("file:///test.kcl").unwrap().clone(); | ||||
|     assert!(semantic_tokens_map != vec![]); | ||||
|     assert!(!semantic_tokens_map.is_empty()); | ||||
|  | ||||
|     // Get the memory. | ||||
|     let memory = server.memory_map.get("file:///test.kcl"); | ||||
| @ -3552,7 +3543,7 @@ async fn kcl_test_kcl_lsp_completions_number_literal() { | ||||
|                 uri: "file:///test.kcl".try_into().unwrap(), | ||||
|                 language_id: "kcl".to_string(), | ||||
|                 version: 1, | ||||
|                 text: "const thing = 10".to_string(), | ||||
|                 text: "thing = 10".to_string(), | ||||
|             }, | ||||
|         }) | ||||
|         .await; | ||||
| @ -3563,7 +3554,7 @@ async fn kcl_test_kcl_lsp_completions_number_literal() { | ||||
|                 text_document: tower_lsp::lsp_types::TextDocumentIdentifier { | ||||
|                     uri: "file:///test.kcl".try_into().unwrap(), | ||||
|                 }, | ||||
|                 position: tower_lsp::lsp_types::Position { line: 0, character: 15 }, | ||||
|                 position: tower_lsp::lsp_types::Position { line: 0, character: 10 }, | ||||
|             }, | ||||
|             context: None, | ||||
|             partial_result_params: Default::default(), | ||||
|  | ||||
| @ -184,7 +184,7 @@ impl Node<Program> { | ||||
|     /// Walk the ast and get all the variables and tags as completion items. | ||||
|     pub fn completion_items<'a>(&'a self) -> Result<Vec<CompletionItem>> { | ||||
|         let completions = Arc::new(Mutex::new(vec![])); | ||||
|         crate::walk::walk(self, &|node: crate::walk::Node<'a>| { | ||||
|         crate::walk::walk(self, |node: crate::walk::Node<'a>| { | ||||
|             let mut findings = completions.lock().map_err(|_| anyhow::anyhow!("mutex"))?; | ||||
|             match node { | ||||
|                 crate::walk::Node::TagDeclarator(tag) => { | ||||
| @ -195,7 +195,7 @@ impl Node<Program> { | ||||
|                 } | ||||
|                 _ => {} | ||||
|             } | ||||
|             Ok(true) | ||||
|             Ok::<bool, anyhow::Error>(true) | ||||
|         })?; | ||||
|         let x = completions.lock().unwrap(); | ||||
|         Ok(x.clone()) | ||||
| @ -204,7 +204,7 @@ impl Node<Program> { | ||||
|     /// Returns all the lsp symbols in the program. | ||||
|     pub fn get_lsp_symbols<'a>(&'a self, code: &str) -> Result<Vec<DocumentSymbol>> { | ||||
|         let symbols = Arc::new(Mutex::new(vec![])); | ||||
|         crate::walk::walk(self, &|node: crate::walk::Node<'a>| { | ||||
|         crate::walk::walk(self, |node: crate::walk::Node<'a>| { | ||||
|             let mut findings = symbols.lock().map_err(|_| anyhow::anyhow!("mutex"))?; | ||||
|             match node { | ||||
|                 crate::walk::Node::TagDeclarator(tag) => { | ||||
| @ -215,7 +215,7 @@ impl Node<Program> { | ||||
|                 } | ||||
|                 _ => {} | ||||
|             } | ||||
|             Ok(true) | ||||
|             Ok::<bool, anyhow::Error>(true) | ||||
|         })?; | ||||
|         let x = symbols.lock().unwrap(); | ||||
|         Ok(x.clone()) | ||||
| @ -227,10 +227,10 @@ impl Node<Program> { | ||||
|         RuleT: crate::lint::Rule<'a>, | ||||
|     { | ||||
|         let v = Arc::new(Mutex::new(vec![])); | ||||
|         crate::walk::walk(self, &|node: crate::walk::Node<'a>| { | ||||
|         crate::walk::walk(self, |node: crate::walk::Node<'a>| { | ||||
|             let mut findings = v.lock().map_err(|_| anyhow::anyhow!("mutex"))?; | ||||
|             findings.append(&mut rule.check(node)?); | ||||
|             Ok(true) | ||||
|             Ok::<bool, anyhow::Error>(true) | ||||
|         })?; | ||||
|         let x = v.lock().unwrap(); | ||||
|         Ok(x.clone()) | ||||
| @ -1228,6 +1228,7 @@ impl ImportSelector { | ||||
| pub struct ImportStatement { | ||||
|     pub selector: ImportSelector, | ||||
|     pub path: String, | ||||
|     #[serde(default, skip_serializing_if = "ItemVisibility::is_default")] | ||||
|     pub visibility: ItemVisibility, | ||||
|  | ||||
|     #[serde(default, skip_serializing_if = "Option::is_none")] | ||||
| @ -2809,7 +2810,8 @@ pub struct Parameter { | ||||
|     pub identifier: Node<Identifier>, | ||||
|     /// The type of the parameter. | ||||
|     /// This is optional if the user defines a type. | ||||
|     #[serde(skip)] | ||||
|     #[serde(default, skip_serializing_if = "Option::is_none")] | ||||
|     #[ts(skip)] | ||||
|     pub type_: Option<FnArgType>, | ||||
|     /// Is the parameter optional? | ||||
|     /// If so, what is its default value? | ||||
|  | ||||
| @ -2,7 +2,7 @@ use crate::{ | ||||
|     errors::{CompilationError, KclError, KclErrorDetails}, | ||||
|     parsing::{ | ||||
|         ast::types::{Node, Program}, | ||||
|         token::{Token, TokenType}, | ||||
|         token::TokenStream, | ||||
|     }, | ||||
|     source_range::{ModuleId, SourceRange}, | ||||
| }; | ||||
| @ -34,15 +34,13 @@ pub fn top_level_parse(code: &str) -> ParseResult { | ||||
|  | ||||
| /// Parse the given KCL code into an AST. | ||||
| pub fn parse_str(code: &str, module_id: ModuleId) -> ParseResult { | ||||
|     let tokens = pr_try!(crate::parsing::token::lexer(code, module_id)); | ||||
|     let tokens = pr_try!(crate::parsing::token::lex(code, module_id)); | ||||
|     parse_tokens(tokens) | ||||
| } | ||||
|  | ||||
| /// Parse the supplied tokens into an AST. | ||||
| pub fn parse_tokens(tokens: Vec<Token>) -> ParseResult { | ||||
|     let (tokens, unknown_tokens): (Vec<Token>, Vec<Token>) = tokens | ||||
|         .into_iter() | ||||
|         .partition(|token| token.token_type != TokenType::Unknown); | ||||
| pub fn parse_tokens(mut tokens: TokenStream) -> ParseResult { | ||||
|     let unknown_tokens = tokens.remove_unknown(); | ||||
|  | ||||
|     if !unknown_tokens.is_empty() { | ||||
|         let source_ranges = unknown_tokens.iter().map(SourceRange::from).collect(); | ||||
| @ -69,7 +67,7 @@ pub fn parse_tokens(tokens: Vec<Token>) -> ParseResult { | ||||
|         return Node::<Program>::default().into(); | ||||
|     } | ||||
|  | ||||
|     parser::run_parser(&mut tokens.as_slice()) | ||||
|     parser::run_parser(tokens.as_slice()) | ||||
| } | ||||
|  | ||||
| /// Result of parsing. | ||||
|  | ||||
| @ -6,11 +6,11 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 143, | ||||
|         "end": 137, | ||||
|         "id": { | ||||
|           "end": 15, | ||||
|           "end": 9, | ||||
|           "name": "boxSketch", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
| @ -20,36 +20,36 @@ expression: actual | ||||
|                 { | ||||
|                   "elements": [ | ||||
|                     { | ||||
|                       "end": 34, | ||||
|                       "end": 28, | ||||
|                       "raw": "0", | ||||
|                       "start": 33, | ||||
|                       "start": 27, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 0.0 | ||||
|                     }, | ||||
|                     { | ||||
|                       "end": 37, | ||||
|                       "end": 31, | ||||
|                       "raw": "0", | ||||
|                       "start": 36, | ||||
|                       "start": 30, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 0.0 | ||||
|                     } | ||||
|                   ], | ||||
|                   "end": 38, | ||||
|                   "start": 32, | ||||
|                   "end": 32, | ||||
|                   "start": 26, | ||||
|                   "type": "ArrayExpression", | ||||
|                   "type": "ArrayExpression" | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 31, | ||||
|                 "end": 25, | ||||
|                 "name": "startSketchAt", | ||||
|                 "start": 18, | ||||
|                 "start": 12, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 39, | ||||
|               "start": 18, | ||||
|               "end": 33, | ||||
|               "start": 12, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             }, | ||||
| @ -58,42 +58,42 @@ expression: actual | ||||
|                 { | ||||
|                   "elements": [ | ||||
|                     { | ||||
|                       "end": 54, | ||||
|                       "end": 48, | ||||
|                       "raw": "0", | ||||
|                       "start": 53, | ||||
|                       "start": 47, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 0.0 | ||||
|                     }, | ||||
|                     { | ||||
|                       "end": 58, | ||||
|                       "end": 52, | ||||
|                       "raw": "10", | ||||
|                       "start": 56, | ||||
|                       "start": 50, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 10.0 | ||||
|                     } | ||||
|                   ], | ||||
|                   "end": 59, | ||||
|                   "start": 52, | ||||
|                   "end": 53, | ||||
|                   "start": 46, | ||||
|                   "type": "ArrayExpression", | ||||
|                   "type": "ArrayExpression" | ||||
|                 }, | ||||
|                 { | ||||
|                   "end": 62, | ||||
|                   "start": 61, | ||||
|                   "end": 56, | ||||
|                   "start": 55, | ||||
|                   "type": "PipeSubstitution", | ||||
|                   "type": "PipeSubstitution" | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 51, | ||||
|                 "end": 45, | ||||
|                 "name": "line", | ||||
|                 "start": 47, | ||||
|                 "start": 41, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 63, | ||||
|               "start": 47, | ||||
|               "end": 57, | ||||
|               "start": 41, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             }, | ||||
| @ -103,48 +103,48 @@ expression: actual | ||||
|                   "elements": [ | ||||
|                     { | ||||
|                       "argument": { | ||||
|                         "end": 88, | ||||
|                         "end": 82, | ||||
|                         "raw": "5", | ||||
|                         "start": 87, | ||||
|                         "start": 81, | ||||
|                         "type": "Literal", | ||||
|                         "type": "Literal", | ||||
|                         "value": 5.0 | ||||
|                       }, | ||||
|                       "end": 88, | ||||
|                       "end": 82, | ||||
|                       "operator": "-", | ||||
|                       "start": 86, | ||||
|                       "start": 80, | ||||
|                       "type": "UnaryExpression", | ||||
|                       "type": "UnaryExpression" | ||||
|                     }, | ||||
|                     { | ||||
|                       "end": 91, | ||||
|                       "end": 85, | ||||
|                       "raw": "5", | ||||
|                       "start": 90, | ||||
|                       "start": 84, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 5.0 | ||||
|                     } | ||||
|                   ], | ||||
|                   "end": 92, | ||||
|                   "start": 85, | ||||
|                   "end": 86, | ||||
|                   "start": 79, | ||||
|                   "type": "ArrayExpression", | ||||
|                   "type": "ArrayExpression" | ||||
|                 }, | ||||
|                 { | ||||
|                   "end": 95, | ||||
|                   "start": 94, | ||||
|                   "end": 89, | ||||
|                   "start": 88, | ||||
|                   "type": "PipeSubstitution", | ||||
|                   "type": "PipeSubstitution" | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 84, | ||||
|                 "end": 78, | ||||
|                 "name": "tangentialArc", | ||||
|                 "start": 71, | ||||
|                 "start": 65, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 96, | ||||
|               "start": 71, | ||||
|               "end": 90, | ||||
|               "start": 65, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             }, | ||||
| @ -153,96 +153,96 @@ expression: actual | ||||
|                 { | ||||
|                   "elements": [ | ||||
|                     { | ||||
|                       "end": 111, | ||||
|                       "end": 105, | ||||
|                       "raw": "5", | ||||
|                       "start": 110, | ||||
|                       "start": 104, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 5.0 | ||||
|                     }, | ||||
|                     { | ||||
|                       "argument": { | ||||
|                         "end": 116, | ||||
|                         "end": 110, | ||||
|                         "raw": "15", | ||||
|                         "start": 114, | ||||
|                         "start": 108, | ||||
|                         "type": "Literal", | ||||
|                         "type": "Literal", | ||||
|                         "value": 15.0 | ||||
|                       }, | ||||
|                       "end": 116, | ||||
|                       "end": 110, | ||||
|                       "operator": "-", | ||||
|                       "start": 113, | ||||
|                       "start": 107, | ||||
|                       "type": "UnaryExpression", | ||||
|                       "type": "UnaryExpression" | ||||
|                     } | ||||
|                   ], | ||||
|                   "end": 117, | ||||
|                   "start": 109, | ||||
|                   "end": 111, | ||||
|                   "start": 103, | ||||
|                   "type": "ArrayExpression", | ||||
|                   "type": "ArrayExpression" | ||||
|                 }, | ||||
|                 { | ||||
|                   "end": 120, | ||||
|                   "start": 119, | ||||
|                   "end": 114, | ||||
|                   "start": 113, | ||||
|                   "type": "PipeSubstitution", | ||||
|                   "type": "PipeSubstitution" | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 108, | ||||
|                 "end": 102, | ||||
|                 "name": "line", | ||||
|                 "start": 104, | ||||
|                 "start": 98, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 121, | ||||
|               "start": 104, | ||||
|               "end": 115, | ||||
|               "start": 98, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             }, | ||||
|             { | ||||
|               "arguments": [ | ||||
|                 { | ||||
|                   "end": 139, | ||||
|                   "end": 133, | ||||
|                   "raw": "10", | ||||
|                   "start": 137, | ||||
|                   "start": 131, | ||||
|                   "type": "Literal", | ||||
|                   "type": "Literal", | ||||
|                   "value": 10.0 | ||||
|                 }, | ||||
|                 { | ||||
|                   "end": 142, | ||||
|                   "start": 141, | ||||
|                   "end": 136, | ||||
|                   "start": 135, | ||||
|                   "type": "PipeSubstitution", | ||||
|                   "type": "PipeSubstitution" | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 136, | ||||
|                 "end": 130, | ||||
|                 "name": "extrude", | ||||
|                 "start": 129, | ||||
|                 "start": 123, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 143, | ||||
|               "start": 129, | ||||
|               "end": 137, | ||||
|               "start": 123, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             } | ||||
|           ], | ||||
|           "end": 143, | ||||
|           "start": 18, | ||||
|           "end": 137, | ||||
|           "start": 12, | ||||
|           "type": "PipeExpression", | ||||
|           "type": "PipeExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 143, | ||||
|       "end": 137, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 144, | ||||
|   "end": 138, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,37 +6,37 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 17, | ||||
|         "end": 11, | ||||
|         "id": { | ||||
|           "end": 8, | ||||
|           "end": 2, | ||||
|           "name": "sg", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "argument": { | ||||
|             "end": 17, | ||||
|             "end": 11, | ||||
|             "name": "scale", | ||||
|             "start": 12, | ||||
|             "start": 6, | ||||
|             "type": "Identifier", | ||||
|             "type": "Identifier" | ||||
|           }, | ||||
|           "end": 17, | ||||
|           "end": 11, | ||||
|           "operator": "-", | ||||
|           "start": 11, | ||||
|           "start": 5, | ||||
|           "type": "UnaryExpression", | ||||
|           "type": "UnaryExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 17, | ||||
|       "end": 11, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 17, | ||||
|   "end": 11, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,29 +6,29 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 23, | ||||
|         "end": 17, | ||||
|         "id": { | ||||
|           "end": 13, | ||||
|           "end": 7, | ||||
|           "name": "myArray", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "end": 23, | ||||
|           "end": 17, | ||||
|           "endElement": { | ||||
|             "end": 22, | ||||
|             "end": 16, | ||||
|             "raw": "10", | ||||
|             "start": 20, | ||||
|             "start": 14, | ||||
|             "type": "Literal", | ||||
|             "type": "Literal", | ||||
|             "value": 10.0 | ||||
|           }, | ||||
|           "endInclusive": true, | ||||
|           "start": 16, | ||||
|           "start": 10, | ||||
|           "startElement": { | ||||
|             "end": 18, | ||||
|             "end": 12, | ||||
|             "raw": "0", | ||||
|             "start": 17, | ||||
|             "start": 11, | ||||
|             "type": "Literal", | ||||
|             "type": "Literal", | ||||
|             "value": 0.0 | ||||
| @ -36,16 +36,16 @@ expression: actual | ||||
|           "type": "ArrayRangeExpression", | ||||
|           "type": "ArrayRangeExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 23, | ||||
|       "end": 17, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 23, | ||||
|   "end": 17, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,11 +6,11 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 165, | ||||
|         "end": 159, | ||||
|         "id": { | ||||
|           "end": 14, | ||||
|           "end": 8, | ||||
|           "name": "mySketch", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
| @ -20,36 +20,36 @@ expression: actual | ||||
|                 { | ||||
|                   "elements": [ | ||||
|                     { | ||||
|                       "end": 33, | ||||
|                       "end": 27, | ||||
|                       "raw": "0", | ||||
|                       "start": 32, | ||||
|                       "start": 26, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 0.0 | ||||
|                     }, | ||||
|                     { | ||||
|                       "end": 35, | ||||
|                       "end": 29, | ||||
|                       "raw": "0", | ||||
|                       "start": 34, | ||||
|                       "start": 28, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 0.0 | ||||
|                     } | ||||
|                   ], | ||||
|                   "end": 36, | ||||
|                   "start": 31, | ||||
|                   "end": 30, | ||||
|                   "start": 25, | ||||
|                   "type": "ArrayExpression", | ||||
|                   "type": "ArrayExpression" | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 30, | ||||
|                 "end": 24, | ||||
|                 "name": "startSketchAt", | ||||
|                 "start": 17, | ||||
|                 "start": 11, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 37, | ||||
|               "start": 17, | ||||
|               "end": 31, | ||||
|               "start": 11, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             }, | ||||
| @ -58,49 +58,49 @@ expression: actual | ||||
|                 { | ||||
|                   "elements": [ | ||||
|                     { | ||||
|                       "end": 58, | ||||
|                       "end": 52, | ||||
|                       "raw": "0", | ||||
|                       "start": 57, | ||||
|                       "start": 51, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 0.0 | ||||
|                     }, | ||||
|                     { | ||||
|                       "end": 61, | ||||
|                       "end": 55, | ||||
|                       "raw": "1", | ||||
|                       "start": 60, | ||||
|                       "start": 54, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 1.0 | ||||
|                     } | ||||
|                   ], | ||||
|                   "end": 62, | ||||
|                   "start": 56, | ||||
|                   "end": 56, | ||||
|                   "start": 50, | ||||
|                   "type": "ArrayExpression", | ||||
|                   "type": "ArrayExpression" | ||||
|                 }, | ||||
|                 { | ||||
|                   "end": 65, | ||||
|                   "start": 64, | ||||
|                   "end": 59, | ||||
|                   "start": 58, | ||||
|                   "type": "PipeSubstitution", | ||||
|                   "type": "PipeSubstitution" | ||||
|                 }, | ||||
|                 { | ||||
|                   "end": 74, | ||||
|                   "start": 67, | ||||
|                   "end": 68, | ||||
|                   "start": 61, | ||||
|                   "type": "TagDeclarator", | ||||
|                   "type": "TagDeclarator", | ||||
|                   "value": "myPath" | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 55, | ||||
|                 "end": 49, | ||||
|                 "name": "lineTo", | ||||
|                 "start": 49, | ||||
|                 "start": 43, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 75, | ||||
|               "start": 49, | ||||
|               "end": 69, | ||||
|               "start": 43, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             }, | ||||
| @ -109,42 +109,42 @@ expression: actual | ||||
|                 { | ||||
|                   "elements": [ | ||||
|                     { | ||||
|                       "end": 96, | ||||
|                       "end": 90, | ||||
|                       "raw": "1", | ||||
|                       "start": 95, | ||||
|                       "start": 89, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 1.0 | ||||
|                     }, | ||||
|                     { | ||||
|                       "end": 99, | ||||
|                       "end": 93, | ||||
|                       "raw": "1", | ||||
|                       "start": 98, | ||||
|                       "start": 92, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 1.0 | ||||
|                     } | ||||
|                   ], | ||||
|                   "end": 100, | ||||
|                   "start": 94, | ||||
|                   "end": 94, | ||||
|                   "start": 88, | ||||
|                   "type": "ArrayExpression", | ||||
|                   "type": "ArrayExpression" | ||||
|                 }, | ||||
|                 { | ||||
|                   "end": 103, | ||||
|                   "start": 102, | ||||
|                   "end": 97, | ||||
|                   "start": 96, | ||||
|                   "type": "PipeSubstitution", | ||||
|                   "type": "PipeSubstitution" | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 93, | ||||
|                 "end": 87, | ||||
|                 "name": "lineTo", | ||||
|                 "start": 87, | ||||
|                 "start": 81, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 104, | ||||
|               "start": 87, | ||||
|               "end": 98, | ||||
|               "start": 81, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             }, | ||||
| @ -153,88 +153,88 @@ expression: actual | ||||
|                 { | ||||
|                   "elements": [ | ||||
|                     { | ||||
|                       "end": 125, | ||||
|                       "end": 119, | ||||
|                       "raw": "1", | ||||
|                       "start": 124, | ||||
|                       "start": 118, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 1.0 | ||||
|                     }, | ||||
|                     { | ||||
|                       "end": 128, | ||||
|                       "end": 122, | ||||
|                       "raw": "0", | ||||
|                       "start": 127, | ||||
|                       "start": 121, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 0.0 | ||||
|                     } | ||||
|                   ], | ||||
|                   "end": 129, | ||||
|                   "start": 123, | ||||
|                   "end": 123, | ||||
|                   "start": 117, | ||||
|                   "type": "ArrayExpression", | ||||
|                   "type": "ArrayExpression" | ||||
|                 }, | ||||
|                 { | ||||
|                   "end": 132, | ||||
|                   "start": 131, | ||||
|                   "end": 126, | ||||
|                   "start": 125, | ||||
|                   "type": "PipeSubstitution", | ||||
|                   "type": "PipeSubstitution" | ||||
|                 }, | ||||
|                 { | ||||
|                   "end": 144, | ||||
|                   "start": 134, | ||||
|                   "end": 138, | ||||
|                   "start": 128, | ||||
|                   "type": "TagDeclarator", | ||||
|                   "type": "TagDeclarator", | ||||
|                   "value": "rightPath" | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 122, | ||||
|                 "end": 116, | ||||
|                 "name": "lineTo", | ||||
|                 "start": 116, | ||||
|                 "start": 110, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 145, | ||||
|               "start": 116, | ||||
|               "end": 139, | ||||
|               "start": 110, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             }, | ||||
|             { | ||||
|               "arguments": [ | ||||
|                 { | ||||
|                   "end": 164, | ||||
|                   "start": 163, | ||||
|                   "end": 158, | ||||
|                   "start": 157, | ||||
|                   "type": "PipeSubstitution", | ||||
|                   "type": "PipeSubstitution" | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 162, | ||||
|                 "end": 156, | ||||
|                 "name": "close", | ||||
|                 "start": 157, | ||||
|                 "start": 151, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 165, | ||||
|               "start": 157, | ||||
|               "end": 159, | ||||
|               "start": 151, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             } | ||||
|           ], | ||||
|           "end": 165, | ||||
|           "start": 17, | ||||
|           "end": 159, | ||||
|           "start": 11, | ||||
|           "type": "PipeExpression", | ||||
|           "type": "PipeExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 165, | ||||
|       "end": 159, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 165, | ||||
|   "end": 159, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,11 +6,11 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 70, | ||||
|         "end": 64, | ||||
|         "id": { | ||||
|           "end": 14, | ||||
|           "end": 8, | ||||
|           "name": "mySketch", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
| @ -20,36 +20,36 @@ expression: actual | ||||
|                 { | ||||
|                   "elements": [ | ||||
|                     { | ||||
|                       "end": 33, | ||||
|                       "end": 27, | ||||
|                       "raw": "0", | ||||
|                       "start": 32, | ||||
|                       "start": 26, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 0.0 | ||||
|                     }, | ||||
|                     { | ||||
|                       "end": 35, | ||||
|                       "end": 29, | ||||
|                       "raw": "0", | ||||
|                       "start": 34, | ||||
|                       "start": 28, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 0.0 | ||||
|                     } | ||||
|                   ], | ||||
|                   "end": 36, | ||||
|                   "start": 31, | ||||
|                   "end": 30, | ||||
|                   "start": 25, | ||||
|                   "type": "ArrayExpression", | ||||
|                   "type": "ArrayExpression" | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 30, | ||||
|                 "end": 24, | ||||
|                 "name": "startSketchAt", | ||||
|                 "start": 17, | ||||
|                 "start": 11, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 37, | ||||
|               "start": 17, | ||||
|               "end": 31, | ||||
|               "start": 11, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             }, | ||||
| @ -58,81 +58,81 @@ expression: actual | ||||
|                 { | ||||
|                   "elements": [ | ||||
|                     { | ||||
|                       "end": 50, | ||||
|                       "end": 44, | ||||
|                       "raw": "1", | ||||
|                       "start": 49, | ||||
|                       "start": 43, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 1.0 | ||||
|                     }, | ||||
|                     { | ||||
|                       "end": 53, | ||||
|                       "end": 47, | ||||
|                       "raw": "1", | ||||
|                       "start": 52, | ||||
|                       "start": 46, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 1.0 | ||||
|                     } | ||||
|                   ], | ||||
|                   "end": 54, | ||||
|                   "start": 48, | ||||
|                   "end": 48, | ||||
|                   "start": 42, | ||||
|                   "type": "ArrayExpression", | ||||
|                   "type": "ArrayExpression" | ||||
|                 }, | ||||
|                 { | ||||
|                   "end": 57, | ||||
|                   "start": 56, | ||||
|                   "end": 51, | ||||
|                   "start": 50, | ||||
|                   "type": "PipeSubstitution", | ||||
|                   "type": "PipeSubstitution" | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 47, | ||||
|                 "end": 41, | ||||
|                 "name": "lineTo", | ||||
|                 "start": 41, | ||||
|                 "start": 35, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 58, | ||||
|               "start": 41, | ||||
|               "end": 52, | ||||
|               "start": 35, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             }, | ||||
|             { | ||||
|               "arguments": [ | ||||
|                 { | ||||
|                   "end": 69, | ||||
|                   "start": 68, | ||||
|                   "end": 63, | ||||
|                   "start": 62, | ||||
|                   "type": "PipeSubstitution", | ||||
|                   "type": "PipeSubstitution" | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 67, | ||||
|                 "end": 61, | ||||
|                 "name": "close", | ||||
|                 "start": 62, | ||||
|                 "start": 56, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 70, | ||||
|               "start": 62, | ||||
|               "end": 64, | ||||
|               "start": 56, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             } | ||||
|           ], | ||||
|           "end": 70, | ||||
|           "start": 17, | ||||
|           "end": 64, | ||||
|           "start": 11, | ||||
|           "type": "PipeExpression", | ||||
|           "type": "PipeExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 70, | ||||
|       "end": 64, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 70, | ||||
|   "end": 64, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,44 +6,44 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 30, | ||||
|         "end": 24, | ||||
|         "id": { | ||||
|           "end": 11, | ||||
|           "end": 5, | ||||
|           "name": "myBox", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "arguments": [ | ||||
|             { | ||||
|               "end": 29, | ||||
|               "end": 23, | ||||
|               "name": "p", | ||||
|               "start": 28, | ||||
|               "start": 22, | ||||
|               "type": "Identifier", | ||||
|               "type": "Identifier" | ||||
|             } | ||||
|           ], | ||||
|           "callee": { | ||||
|             "end": 27, | ||||
|             "end": 21, | ||||
|             "name": "startSketchAt", | ||||
|             "start": 14, | ||||
|             "start": 8, | ||||
|             "type": "Identifier" | ||||
|           }, | ||||
|           "end": 30, | ||||
|           "start": 14, | ||||
|           "end": 24, | ||||
|           "start": 8, | ||||
|           "type": "CallExpression", | ||||
|           "type": "CallExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 30, | ||||
|       "end": 24, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 30, | ||||
|   "end": 24, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,11 +6,11 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 29, | ||||
|         "end": 23, | ||||
|         "id": { | ||||
|           "end": 11, | ||||
|           "end": 5, | ||||
|           "name": "myBox", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
| @ -18,69 +18,69 @@ expression: actual | ||||
|             { | ||||
|               "arguments": [ | ||||
|                 { | ||||
|                   "end": 17, | ||||
|                   "end": 11, | ||||
|                   "raw": "1", | ||||
|                   "start": 16, | ||||
|                   "start": 10, | ||||
|                   "type": "Literal", | ||||
|                   "type": "Literal", | ||||
|                   "value": 1.0 | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 15, | ||||
|                 "end": 9, | ||||
|                 "name": "f", | ||||
|                 "start": 14, | ||||
|                 "start": 8, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 18, | ||||
|               "start": 14, | ||||
|               "end": 12, | ||||
|               "start": 8, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             }, | ||||
|             { | ||||
|               "arguments": [ | ||||
|                 { | ||||
|                   "end": 25, | ||||
|                   "end": 19, | ||||
|                   "raw": "2", | ||||
|                   "start": 24, | ||||
|                   "start": 18, | ||||
|                   "type": "Literal", | ||||
|                   "type": "Literal", | ||||
|                   "value": 2.0 | ||||
|                 }, | ||||
|                 { | ||||
|                   "end": 28, | ||||
|                   "start": 27, | ||||
|                   "end": 22, | ||||
|                   "start": 21, | ||||
|                   "type": "PipeSubstitution", | ||||
|                   "type": "PipeSubstitution" | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 23, | ||||
|                 "end": 17, | ||||
|                 "name": "g", | ||||
|                 "start": 22, | ||||
|                 "start": 16, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 29, | ||||
|               "start": 22, | ||||
|               "end": 23, | ||||
|               "start": 16, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             } | ||||
|           ], | ||||
|           "end": 29, | ||||
|           "start": 14, | ||||
|           "end": 23, | ||||
|           "start": 8, | ||||
|           "type": "PipeExpression", | ||||
|           "type": "PipeExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 29, | ||||
|       "end": 23, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 29, | ||||
|   "end": 23, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,11 +6,11 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 49, | ||||
|         "end": 43, | ||||
|         "id": { | ||||
|           "end": 11, | ||||
|           "end": 5, | ||||
|           "name": "myBox", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
| @ -18,21 +18,21 @@ expression: actual | ||||
|             { | ||||
|               "arguments": [ | ||||
|                 { | ||||
|                   "end": 29, | ||||
|                   "end": 23, | ||||
|                   "name": "p", | ||||
|                   "start": 28, | ||||
|                   "start": 22, | ||||
|                   "type": "Identifier", | ||||
|                   "type": "Identifier" | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 27, | ||||
|                 "end": 21, | ||||
|                 "name": "startSketchAt", | ||||
|                 "start": 14, | ||||
|                 "start": 8, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 30, | ||||
|               "start": 14, | ||||
|               "end": 24, | ||||
|               "start": 8, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             }, | ||||
| @ -41,60 +41,60 @@ expression: actual | ||||
|                 { | ||||
|                   "elements": [ | ||||
|                     { | ||||
|                       "end": 41, | ||||
|                       "end": 35, | ||||
|                       "raw": "0", | ||||
|                       "start": 40, | ||||
|                       "start": 34, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 0.0 | ||||
|                     }, | ||||
|                     { | ||||
|                       "end": 44, | ||||
|                       "end": 38, | ||||
|                       "name": "l", | ||||
|                       "start": 43, | ||||
|                       "start": 37, | ||||
|                       "type": "Identifier", | ||||
|                       "type": "Identifier" | ||||
|                     } | ||||
|                   ], | ||||
|                   "end": 45, | ||||
|                   "start": 39, | ||||
|                   "end": 39, | ||||
|                   "start": 33, | ||||
|                   "type": "ArrayExpression", | ||||
|                   "type": "ArrayExpression" | ||||
|                 }, | ||||
|                 { | ||||
|                   "end": 48, | ||||
|                   "start": 47, | ||||
|                   "end": 42, | ||||
|                   "start": 41, | ||||
|                   "type": "PipeSubstitution", | ||||
|                   "type": "PipeSubstitution" | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 38, | ||||
|                 "end": 32, | ||||
|                 "name": "line", | ||||
|                 "start": 34, | ||||
|                 "start": 28, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 49, | ||||
|               "start": 34, | ||||
|               "end": 43, | ||||
|               "start": 28, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             } | ||||
|           ], | ||||
|           "end": 49, | ||||
|           "start": 14, | ||||
|           "end": 43, | ||||
|           "start": 8, | ||||
|           "type": "PipeExpression", | ||||
|           "type": "PipeExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 49, | ||||
|       "end": 43, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 49, | ||||
|   "end": 43, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,11 +6,11 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 37, | ||||
|         "end": 31, | ||||
|         "id": { | ||||
|           "end": 14, | ||||
|           "end": 8, | ||||
|           "name": "mySketch", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
| @ -18,49 +18,49 @@ expression: actual | ||||
|             { | ||||
|               "elements": [ | ||||
|                 { | ||||
|                   "end": 33, | ||||
|                   "end": 27, | ||||
|                   "raw": "0", | ||||
|                   "start": 32, | ||||
|                   "start": 26, | ||||
|                   "type": "Literal", | ||||
|                   "type": "Literal", | ||||
|                   "value": 0.0 | ||||
|                 }, | ||||
|                 { | ||||
|                   "end": 35, | ||||
|                   "end": 29, | ||||
|                   "raw": "0", | ||||
|                   "start": 34, | ||||
|                   "start": 28, | ||||
|                   "type": "Literal", | ||||
|                   "type": "Literal", | ||||
|                   "value": 0.0 | ||||
|                 } | ||||
|               ], | ||||
|               "end": 36, | ||||
|               "start": 31, | ||||
|               "end": 30, | ||||
|               "start": 25, | ||||
|               "type": "ArrayExpression", | ||||
|               "type": "ArrayExpression" | ||||
|             } | ||||
|           ], | ||||
|           "callee": { | ||||
|             "end": 30, | ||||
|             "end": 24, | ||||
|             "name": "startSketchAt", | ||||
|             "start": 17, | ||||
|             "start": 11, | ||||
|             "type": "Identifier" | ||||
|           }, | ||||
|           "end": 37, | ||||
|           "start": 17, | ||||
|           "end": 31, | ||||
|           "start": 11, | ||||
|           "type": "CallExpression", | ||||
|           "type": "CallExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 37, | ||||
|       "end": 31, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 37, | ||||
|   "end": 31, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,19 +6,19 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 36, | ||||
|         "end": 30, | ||||
|         "id": { | ||||
|           "end": 11, | ||||
|           "end": 5, | ||||
|           "name": "myVar", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "arguments": [ | ||||
|             { | ||||
|               "end": 19, | ||||
|               "end": 13, | ||||
|               "raw": "5", | ||||
|               "start": 18, | ||||
|               "start": 12, | ||||
|               "type": "Literal", | ||||
|               "type": "Literal", | ||||
|               "value": 5.0 | ||||
| @ -27,61 +27,61 @@ expression: actual | ||||
|               "argument": { | ||||
|                 "arguments": [ | ||||
|                   { | ||||
|                     "end": 31, | ||||
|                     "end": 25, | ||||
|                     "raw": "5", | ||||
|                     "start": 30, | ||||
|                     "start": 24, | ||||
|                     "type": "Literal", | ||||
|                     "type": "Literal", | ||||
|                     "value": 5.0 | ||||
|                   }, | ||||
|                   { | ||||
|                     "end": 34, | ||||
|                     "end": 28, | ||||
|                     "raw": "4", | ||||
|                     "start": 33, | ||||
|                     "start": 27, | ||||
|                     "type": "Literal", | ||||
|                     "type": "Literal", | ||||
|                     "value": 4.0 | ||||
|                   } | ||||
|                 ], | ||||
|                 "callee": { | ||||
|                   "end": 29, | ||||
|                   "end": 23, | ||||
|                   "name": "legLen", | ||||
|                   "start": 23, | ||||
|                   "start": 17, | ||||
|                   "type": "Identifier" | ||||
|                 }, | ||||
|                 "end": 35, | ||||
|                 "start": 23, | ||||
|                 "end": 29, | ||||
|                 "start": 17, | ||||
|                 "type": "CallExpression", | ||||
|                 "type": "CallExpression" | ||||
|               }, | ||||
|               "end": 35, | ||||
|               "end": 29, | ||||
|               "operator": "-", | ||||
|               "start": 22, | ||||
|               "start": 16, | ||||
|               "type": "UnaryExpression", | ||||
|               "type": "UnaryExpression" | ||||
|             } | ||||
|           ], | ||||
|           "callee": { | ||||
|             "end": 17, | ||||
|             "end": 11, | ||||
|             "name": "min", | ||||
|             "start": 14, | ||||
|             "start": 8, | ||||
|             "type": "Identifier" | ||||
|           }, | ||||
|           "end": 36, | ||||
|           "start": 14, | ||||
|           "end": 30, | ||||
|           "start": 8, | ||||
|           "type": "CallExpression", | ||||
|           "type": "CallExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 36, | ||||
|       "end": 30, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 36, | ||||
|   "end": 30, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,11 +6,11 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 132, | ||||
|         "end": 126, | ||||
|         "id": { | ||||
|           "end": 16, | ||||
|           "end": 10, | ||||
|           "name": "sketch001", | ||||
|           "start": 7, | ||||
|           "start": 1, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
| @ -18,53 +18,53 @@ expression: actual | ||||
|             { | ||||
|               "arguments": [ | ||||
|                 { | ||||
|                   "end": 37, | ||||
|                   "end": 31, | ||||
|                   "raw": "'XY'", | ||||
|                   "start": 33, | ||||
|                   "start": 27, | ||||
|                   "type": "Literal", | ||||
|                   "type": "Literal", | ||||
|                   "value": "XY" | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 32, | ||||
|                 "end": 26, | ||||
|                 "name": "startSketchOn", | ||||
|                 "start": 19, | ||||
|                 "start": 13, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 38, | ||||
|               "start": 19, | ||||
|               "end": 32, | ||||
|               "start": 13, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             }, | ||||
|             { | ||||
|               "arguments": [ | ||||
|                 { | ||||
|                   "end": 131, | ||||
|                   "start": 130, | ||||
|                   "end": 125, | ||||
|                   "start": 124, | ||||
|                   "type": "PipeSubstitution", | ||||
|                   "type": "PipeSubstitution" | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 129, | ||||
|                 "end": 123, | ||||
|                 "name": "startProfileAt", | ||||
|                 "start": 115, | ||||
|                 "start": 109, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 132, | ||||
|               "start": 115, | ||||
|               "end": 126, | ||||
|               "start": 109, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             } | ||||
|           ], | ||||
|           "end": 132, | ||||
|           "end": 126, | ||||
|           "nonCodeMeta": { | ||||
|             "nonCodeNodes": { | ||||
|               "0": [ | ||||
|                 { | ||||
|                   "end": 52, | ||||
|                   "start": 41, | ||||
|                   "end": 46, | ||||
|                   "start": 35, | ||||
|                   "type": "NonCodeNode", | ||||
|                   "value": { | ||||
|                     "type": "blockComment", | ||||
| @ -73,8 +73,8 @@ expression: actual | ||||
|                   } | ||||
|                 }, | ||||
|                 { | ||||
|                   "end": 74, | ||||
|                   "start": 55, | ||||
|                   "end": 68, | ||||
|                   "start": 49, | ||||
|                   "type": "NonCodeNode", | ||||
|                   "value": { | ||||
|                     "type": "blockComment", | ||||
| @ -83,8 +83,8 @@ expression: actual | ||||
|                   } | ||||
|                 }, | ||||
|                 { | ||||
|                   "end": 98, | ||||
|                   "start": 77, | ||||
|                   "end": 92, | ||||
|                   "start": 71, | ||||
|                   "type": "NonCodeNode", | ||||
|                   "value": { | ||||
|                     "type": "blockComment", | ||||
| @ -93,8 +93,8 @@ expression: actual | ||||
|                   } | ||||
|                 }, | ||||
|                 { | ||||
|                   "end": 109, | ||||
|                   "start": 101, | ||||
|                   "end": 103, | ||||
|                   "start": 95, | ||||
|                   "type": "NonCodeNode", | ||||
|                   "value": { | ||||
|                     "type": "blockComment", | ||||
| @ -106,20 +106,20 @@ expression: actual | ||||
|             }, | ||||
|             "startNodes": [] | ||||
|           }, | ||||
|           "start": 19, | ||||
|           "start": 13, | ||||
|           "type": "PipeExpression", | ||||
|           "type": "PipeExpression" | ||||
|         }, | ||||
|         "start": 7, | ||||
|         "start": 1, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 132, | ||||
|       "end": 126, | ||||
|       "kind": "const", | ||||
|       "start": 1, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 133, | ||||
|   "end": 127, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,91 +6,91 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 31, | ||||
|         "end": 25, | ||||
|         "id": { | ||||
|           "end": 11, | ||||
|           "end": 5, | ||||
|           "name": "my14", | ||||
|           "start": 7, | ||||
|           "start": 1, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "end": 31, | ||||
|           "end": 25, | ||||
|           "left": { | ||||
|             "end": 19, | ||||
|             "end": 13, | ||||
|             "left": { | ||||
|               "end": 15, | ||||
|               "end": 9, | ||||
|               "raw": "4", | ||||
|               "start": 14, | ||||
|               "start": 8, | ||||
|               "type": "Literal", | ||||
|               "type": "Literal", | ||||
|               "value": 4.0 | ||||
|             }, | ||||
|             "operator": "^", | ||||
|             "right": { | ||||
|               "end": 19, | ||||
|               "end": 13, | ||||
|               "raw": "2", | ||||
|               "start": 18, | ||||
|               "start": 12, | ||||
|               "type": "Literal", | ||||
|               "type": "Literal", | ||||
|               "value": 2.0 | ||||
|             }, | ||||
|             "start": 14, | ||||
|             "start": 8, | ||||
|             "type": "BinaryExpression", | ||||
|             "type": "BinaryExpression" | ||||
|           }, | ||||
|           "operator": "-", | ||||
|           "right": { | ||||
|             "end": 31, | ||||
|             "end": 25, | ||||
|             "left": { | ||||
|               "end": 27, | ||||
|               "end": 21, | ||||
|               "left": { | ||||
|                 "end": 23, | ||||
|                 "end": 17, | ||||
|                 "raw": "3", | ||||
|                 "start": 22, | ||||
|                 "start": 16, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 3.0 | ||||
|               }, | ||||
|               "operator": "^", | ||||
|               "right": { | ||||
|                 "end": 27, | ||||
|                 "end": 21, | ||||
|                 "raw": "2", | ||||
|                 "start": 26, | ||||
|                 "start": 20, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 2.0 | ||||
|               }, | ||||
|               "start": 22, | ||||
|               "start": 16, | ||||
|               "type": "BinaryExpression", | ||||
|               "type": "BinaryExpression" | ||||
|             }, | ||||
|             "operator": "*", | ||||
|             "right": { | ||||
|               "end": 31, | ||||
|               "end": 25, | ||||
|               "raw": "2", | ||||
|               "start": 30, | ||||
|               "start": 24, | ||||
|               "type": "Literal", | ||||
|               "type": "Literal", | ||||
|               "value": 2.0 | ||||
|             }, | ||||
|             "start": 22, | ||||
|             "start": 16, | ||||
|             "type": "BinaryExpression", | ||||
|             "type": "BinaryExpression" | ||||
|           }, | ||||
|           "start": 14, | ||||
|           "start": 8, | ||||
|           "type": "BinaryExpression", | ||||
|           "type": "BinaryExpression" | ||||
|         }, | ||||
|         "start": 7, | ||||
|         "start": 1, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 31, | ||||
|       "end": 25, | ||||
|       "kind": "const", | ||||
|       "start": 1, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 32, | ||||
|   "end": 26, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,79 +6,79 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 74, | ||||
|         "end": 68, | ||||
|         "id": { | ||||
|           "end": 7, | ||||
|           "end": 1, | ||||
|           "name": "x", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "cond": { | ||||
|             "end": 17, | ||||
|             "end": 11, | ||||
|             "raw": "true", | ||||
|             "start": 13, | ||||
|             "start": 7, | ||||
|             "type": "Literal", | ||||
|             "type": "Literal", | ||||
|             "value": true | ||||
|           }, | ||||
|           "digest": null, | ||||
|           "else_ifs": [], | ||||
|           "end": 74, | ||||
|           "end": 68, | ||||
|           "final_else": { | ||||
|             "body": [ | ||||
|               { | ||||
|                 "end": 64, | ||||
|                 "end": 58, | ||||
|                 "expression": { | ||||
|                   "end": 64, | ||||
|                   "end": 58, | ||||
|                   "raw": "4", | ||||
|                   "start": 63, | ||||
|                   "start": 57, | ||||
|                   "type": "Literal", | ||||
|                   "type": "Literal", | ||||
|                   "value": 4.0 | ||||
|                 }, | ||||
|                 "start": 63, | ||||
|                 "start": 57, | ||||
|                 "type": "ExpressionStatement", | ||||
|                 "type": "ExpressionStatement" | ||||
|               } | ||||
|             ], | ||||
|             "end": 73, | ||||
|             "start": 63 | ||||
|             "end": 67, | ||||
|             "start": 57 | ||||
|           }, | ||||
|           "start": 10, | ||||
|           "start": 4, | ||||
|           "then_val": { | ||||
|             "body": [ | ||||
|               { | ||||
|                 "end": 33, | ||||
|                 "end": 27, | ||||
|                 "expression": { | ||||
|                   "end": 33, | ||||
|                   "end": 27, | ||||
|                   "raw": "3", | ||||
|                   "start": 32, | ||||
|                   "start": 26, | ||||
|                   "type": "Literal", | ||||
|                   "type": "Literal", | ||||
|                   "value": 3.0 | ||||
|                 }, | ||||
|                 "start": 32, | ||||
|                 "start": 26, | ||||
|                 "type": "ExpressionStatement", | ||||
|                 "type": "ExpressionStatement" | ||||
|               } | ||||
|             ], | ||||
|             "end": 42, | ||||
|             "start": 32 | ||||
|             "end": 36, | ||||
|             "start": 26 | ||||
|           }, | ||||
|           "type": "IfExpression", | ||||
|           "type": "IfExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 74, | ||||
|       "end": 68, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 74, | ||||
|   "end": 68, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,18 +6,18 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 121, | ||||
|         "end": 115, | ||||
|         "id": { | ||||
|           "end": 7, | ||||
|           "end": 1, | ||||
|           "name": "x", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "cond": { | ||||
|             "end": 17, | ||||
|             "end": 11, | ||||
|             "raw": "true", | ||||
|             "start": 13, | ||||
|             "start": 7, | ||||
|             "type": "Literal", | ||||
|             "type": "Literal", | ||||
|             "value": true | ||||
| @ -28,105 +28,105 @@ expression: actual | ||||
|               "cond": { | ||||
|                 "arguments": [ | ||||
|                   { | ||||
|                     "end": 63, | ||||
|                     "end": 57, | ||||
|                     "name": "radius", | ||||
|                     "start": 57, | ||||
|                     "start": 51, | ||||
|                     "type": "Identifier", | ||||
|                     "type": "Identifier" | ||||
|                   } | ||||
|                 ], | ||||
|                 "callee": { | ||||
|                   "end": 56, | ||||
|                   "end": 50, | ||||
|                   "name": "func", | ||||
|                   "start": 52, | ||||
|                   "start": 46, | ||||
|                   "type": "Identifier" | ||||
|                 }, | ||||
|                 "end": 64, | ||||
|                 "start": 52, | ||||
|                 "end": 58, | ||||
|                 "start": 46, | ||||
|                 "type": "CallExpression", | ||||
|                 "type": "CallExpression" | ||||
|               }, | ||||
|               "digest": null, | ||||
|               "end": 90, | ||||
|               "start": 44, | ||||
|               "end": 84, | ||||
|               "start": 38, | ||||
|               "then_val": { | ||||
|                 "body": [ | ||||
|                   { | ||||
|                     "end": 80, | ||||
|                     "end": 74, | ||||
|                     "expression": { | ||||
|                       "end": 80, | ||||
|                       "end": 74, | ||||
|                       "raw": "4", | ||||
|                       "start": 79, | ||||
|                       "start": 73, | ||||
|                       "type": "Literal", | ||||
|                       "type": "Literal", | ||||
|                       "value": 4.0 | ||||
|                     }, | ||||
|                     "start": 79, | ||||
|                     "start": 73, | ||||
|                     "type": "ExpressionStatement", | ||||
|                     "type": "ExpressionStatement" | ||||
|                   } | ||||
|                 ], | ||||
|                 "end": 89, | ||||
|                 "start": 65 | ||||
|                 "end": 83, | ||||
|                 "start": 59 | ||||
|               }, | ||||
|               "type": "ElseIf" | ||||
|             } | ||||
|           ], | ||||
|           "end": 121, | ||||
|           "end": 115, | ||||
|           "final_else": { | ||||
|             "body": [ | ||||
|               { | ||||
|                 "end": 111, | ||||
|                 "end": 105, | ||||
|                 "expression": { | ||||
|                   "end": 111, | ||||
|                   "end": 105, | ||||
|                   "raw": "5", | ||||
|                   "start": 110, | ||||
|                   "start": 104, | ||||
|                   "type": "Literal", | ||||
|                   "type": "Literal", | ||||
|                   "value": 5.0 | ||||
|                 }, | ||||
|                 "start": 110, | ||||
|                 "start": 104, | ||||
|                 "type": "ExpressionStatement", | ||||
|                 "type": "ExpressionStatement" | ||||
|               } | ||||
|             ], | ||||
|             "end": 120, | ||||
|             "start": 110 | ||||
|             "end": 114, | ||||
|             "start": 104 | ||||
|           }, | ||||
|           "start": 10, | ||||
|           "start": 4, | ||||
|           "then_val": { | ||||
|             "body": [ | ||||
|               { | ||||
|                 "end": 33, | ||||
|                 "end": 27, | ||||
|                 "expression": { | ||||
|                   "end": 33, | ||||
|                   "end": 27, | ||||
|                   "raw": "3", | ||||
|                   "start": 32, | ||||
|                   "start": 26, | ||||
|                   "type": "Literal", | ||||
|                   "type": "Literal", | ||||
|                   "value": 3.0 | ||||
|                 }, | ||||
|                 "start": 32, | ||||
|                 "start": 26, | ||||
|                 "type": "ExpressionStatement", | ||||
|                 "type": "ExpressionStatement" | ||||
|               } | ||||
|             ], | ||||
|             "end": 42, | ||||
|             "start": 32 | ||||
|             "end": 36, | ||||
|             "start": 26 | ||||
|           }, | ||||
|           "type": "IfExpression", | ||||
|           "type": "IfExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 121, | ||||
|       "end": 115, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 121, | ||||
|   "end": 115, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,85 +6,85 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 42, | ||||
|         "end": 36, | ||||
|         "id": { | ||||
|           "end": 9, | ||||
|           "end": 3, | ||||
|           "name": "obj", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "end": 42, | ||||
|           "end": 36, | ||||
|           "properties": [ | ||||
|             { | ||||
|               "end": 30, | ||||
|               "end": 24, | ||||
|               "key": { | ||||
|                 "end": 19, | ||||
|                 "end": 13, | ||||
|                 "name": "center", | ||||
|                 "start": 13, | ||||
|                 "start": 7, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "start": 13, | ||||
|               "start": 7, | ||||
|               "type": "ObjectProperty", | ||||
|               "value": { | ||||
|                 "elements": [ | ||||
|                   { | ||||
|                     "end": 25, | ||||
|                     "end": 19, | ||||
|                     "raw": "10", | ||||
|                     "start": 23, | ||||
|                     "start": 17, | ||||
|                     "type": "Literal", | ||||
|                     "type": "Literal", | ||||
|                     "value": 10.0 | ||||
|                   }, | ||||
|                   { | ||||
|                     "end": 29, | ||||
|                     "end": 23, | ||||
|                     "raw": "10", | ||||
|                     "start": 27, | ||||
|                     "start": 21, | ||||
|                     "type": "Literal", | ||||
|                     "type": "Literal", | ||||
|                     "value": 10.0 | ||||
|                   } | ||||
|                 ], | ||||
|                 "end": 30, | ||||
|                 "start": 22, | ||||
|                 "end": 24, | ||||
|                 "start": 16, | ||||
|                 "type": "ArrayExpression", | ||||
|                 "type": "ArrayExpression" | ||||
|               } | ||||
|             }, | ||||
|             { | ||||
|               "end": 41, | ||||
|               "end": 35, | ||||
|               "key": { | ||||
|                 "end": 38, | ||||
|                 "end": 32, | ||||
|                 "name": "radius", | ||||
|                 "start": 32, | ||||
|                 "start": 26, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "start": 32, | ||||
|               "start": 26, | ||||
|               "type": "ObjectProperty", | ||||
|               "value": { | ||||
|                 "end": 41, | ||||
|                 "end": 35, | ||||
|                 "raw": "5", | ||||
|                 "start": 40, | ||||
|                 "start": 34, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 5.0 | ||||
|               } | ||||
|             } | ||||
|           ], | ||||
|           "start": 12, | ||||
|           "start": 6, | ||||
|           "type": "ObjectExpression", | ||||
|           "type": "ObjectExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 42, | ||||
|       "end": 36, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 42, | ||||
|   "end": 36, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,11 +6,11 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 35, | ||||
|         "end": 29, | ||||
|         "id": { | ||||
|           "end": 11, | ||||
|           "end": 5, | ||||
|           "name": "myVar", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
| @ -19,69 +19,69 @@ expression: actual | ||||
|               "argument": { | ||||
|                 "arguments": [ | ||||
|                   { | ||||
|                     "end": 27, | ||||
|                     "end": 21, | ||||
|                     "raw": "5", | ||||
|                     "start": 26, | ||||
|                     "start": 20, | ||||
|                     "type": "Literal", | ||||
|                     "type": "Literal", | ||||
|                     "value": 5.0 | ||||
|                   }, | ||||
|                   { | ||||
|                     "end": 30, | ||||
|                     "end": 24, | ||||
|                     "raw": "4", | ||||
|                     "start": 29, | ||||
|                     "start": 23, | ||||
|                     "type": "Literal", | ||||
|                     "type": "Literal", | ||||
|                     "value": 4.0 | ||||
|                   } | ||||
|                 ], | ||||
|                 "callee": { | ||||
|                   "end": 25, | ||||
|                   "end": 19, | ||||
|                   "name": "legLen", | ||||
|                   "start": 19, | ||||
|                   "start": 13, | ||||
|                   "type": "Identifier" | ||||
|                 }, | ||||
|                 "end": 31, | ||||
|                 "start": 19, | ||||
|                 "end": 25, | ||||
|                 "start": 13, | ||||
|                 "type": "CallExpression", | ||||
|                 "type": "CallExpression" | ||||
|               }, | ||||
|               "end": 31, | ||||
|               "end": 25, | ||||
|               "operator": "-", | ||||
|               "start": 18, | ||||
|               "start": 12, | ||||
|               "type": "UnaryExpression", | ||||
|               "type": "UnaryExpression" | ||||
|             }, | ||||
|             { | ||||
|               "end": 34, | ||||
|               "end": 28, | ||||
|               "raw": "5", | ||||
|               "start": 33, | ||||
|               "start": 27, | ||||
|               "type": "Literal", | ||||
|               "type": "Literal", | ||||
|               "value": 5.0 | ||||
|             } | ||||
|           ], | ||||
|           "callee": { | ||||
|             "end": 17, | ||||
|             "end": 11, | ||||
|             "name": "min", | ||||
|             "start": 14, | ||||
|             "start": 8, | ||||
|             "type": "Identifier" | ||||
|           }, | ||||
|           "end": 35, | ||||
|           "start": 14, | ||||
|           "end": 29, | ||||
|           "start": 8, | ||||
|           "type": "CallExpression", | ||||
|           "type": "CallExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 35, | ||||
|       "end": 29, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 35, | ||||
|   "end": 29, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,82 +6,82 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 36, | ||||
|         "end": 30, | ||||
|         "id": { | ||||
|           "end": 11, | ||||
|           "end": 5, | ||||
|           "name": "myVar", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "body": [ | ||||
|             { | ||||
|               "end": 19, | ||||
|               "end": 13, | ||||
|               "left": { | ||||
|                 "end": 15, | ||||
|                 "end": 9, | ||||
|                 "raw": "5", | ||||
|                 "start": 14, | ||||
|                 "start": 8, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 5.0 | ||||
|               }, | ||||
|               "operator": "+", | ||||
|               "right": { | ||||
|                 "end": 19, | ||||
|                 "end": 13, | ||||
|                 "raw": "6", | ||||
|                 "start": 18, | ||||
|                 "start": 12, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 6.0 | ||||
|               }, | ||||
|               "start": 14, | ||||
|               "start": 8, | ||||
|               "type": "BinaryExpression", | ||||
|               "type": "BinaryExpression" | ||||
|             }, | ||||
|             { | ||||
|               "arguments": [ | ||||
|                 { | ||||
|                   "end": 32, | ||||
|                   "end": 26, | ||||
|                   "raw": "45", | ||||
|                   "start": 30, | ||||
|                   "start": 24, | ||||
|                   "type": "Literal", | ||||
|                   "type": "Literal", | ||||
|                   "value": 45.0 | ||||
|                 }, | ||||
|                 { | ||||
|                   "end": 35, | ||||
|                   "start": 34, | ||||
|                   "end": 29, | ||||
|                   "start": 28, | ||||
|                   "type": "PipeSubstitution", | ||||
|                   "type": "PipeSubstitution" | ||||
|                 } | ||||
|               ], | ||||
|               "callee": { | ||||
|                 "end": 29, | ||||
|                 "end": 23, | ||||
|                 "name": "myFunc", | ||||
|                 "start": 23, | ||||
|                 "start": 17, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "end": 36, | ||||
|               "start": 23, | ||||
|               "end": 30, | ||||
|               "start": 17, | ||||
|               "type": "CallExpression", | ||||
|               "type": "CallExpression" | ||||
|             } | ||||
|           ], | ||||
|           "end": 36, | ||||
|           "start": 14, | ||||
|           "end": 30, | ||||
|           "start": 8, | ||||
|           "type": "PipeExpression", | ||||
|           "type": "PipeExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 36, | ||||
|       "end": 30, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 36, | ||||
|   "end": 30, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,51 +6,51 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 27, | ||||
|         "end": 21, | ||||
|         "id": { | ||||
|           "end": 7, | ||||
|           "end": 1, | ||||
|           "name": "x", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "end": 27, | ||||
|           "end": 21, | ||||
|           "left": { | ||||
|             "argument": { | ||||
|               "end": 15, | ||||
|               "end": 9, | ||||
|               "name": "leg2", | ||||
|               "start": 11, | ||||
|               "start": 5, | ||||
|               "type": "Identifier", | ||||
|               "type": "Identifier" | ||||
|             }, | ||||
|             "end": 15, | ||||
|             "end": 9, | ||||
|             "operator": "-", | ||||
|             "start": 10, | ||||
|             "start": 4, | ||||
|             "type": "UnaryExpression", | ||||
|             "type": "UnaryExpression" | ||||
|           }, | ||||
|           "operator": "+", | ||||
|           "right": { | ||||
|             "end": 27, | ||||
|             "end": 21, | ||||
|             "name": "thickness", | ||||
|             "start": 18, | ||||
|             "start": 12, | ||||
|             "type": "Identifier", | ||||
|             "type": "Identifier" | ||||
|           }, | ||||
|           "start": 10, | ||||
|           "start": 4, | ||||
|           "type": "BinaryExpression", | ||||
|           "type": "BinaryExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 27, | ||||
|       "end": 21, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 27, | ||||
|   "end": 21, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,38 +6,38 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 11, | ||||
|         "end": 5, | ||||
|         "id": { | ||||
|           "end": 7, | ||||
|           "end": 1, | ||||
|           "name": "x", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "end": 11, | ||||
|           "end": 5, | ||||
|           "raw": "1", | ||||
|           "start": 10, | ||||
|           "start": 4, | ||||
|           "type": "Literal", | ||||
|           "type": "Literal", | ||||
|           "value": 1.0 | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 11, | ||||
|       "end": 5, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 40, | ||||
|   "end": 34, | ||||
|   "nonCodeMeta": { | ||||
|     "nonCodeNodes": { | ||||
|       "0": [ | ||||
|         { | ||||
|           "end": 40, | ||||
|           "start": 11, | ||||
|           "end": 34, | ||||
|           "start": 5, | ||||
|           "type": "NonCodeNode", | ||||
|           "value": { | ||||
|             "type": "inlineComment", | ||||
|  | ||||
| @ -6,63 +6,63 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 26, | ||||
|         "end": 20, | ||||
|         "id": { | ||||
|           "end": 9, | ||||
|           "end": 3, | ||||
|           "name": "obj", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "end": 26, | ||||
|           "end": 20, | ||||
|           "properties": [ | ||||
|             { | ||||
|               "end": 18, | ||||
|               "end": 12, | ||||
|               "key": { | ||||
|                 "end": 15, | ||||
|                 "end": 9, | ||||
|                 "name": "a", | ||||
|                 "start": 14, | ||||
|                 "start": 8, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "start": 14, | ||||
|               "start": 8, | ||||
|               "type": "ObjectProperty", | ||||
|               "value": { | ||||
|                 "end": 18, | ||||
|                 "end": 12, | ||||
|                 "raw": "1", | ||||
|                 "start": 17, | ||||
|                 "start": 11, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 1.0 | ||||
|               } | ||||
|             }, | ||||
|             { | ||||
|               "end": 24, | ||||
|               "end": 18, | ||||
|               "key": { | ||||
|                 "end": 21, | ||||
|                 "end": 15, | ||||
|                 "name": "b", | ||||
|                 "start": 20, | ||||
|                 "start": 14, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "start": 20, | ||||
|               "start": 14, | ||||
|               "type": "ObjectProperty", | ||||
|               "value": { | ||||
|                 "end": 24, | ||||
|                 "end": 18, | ||||
|                 "raw": "2", | ||||
|                 "start": 23, | ||||
|                 "start": 17, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 2.0 | ||||
|               } | ||||
|             } | ||||
|           ], | ||||
|           "start": 12, | ||||
|           "start": 6, | ||||
|           "type": "ObjectExpression", | ||||
|           "type": "ObjectExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 26, | ||||
|       "end": 20, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
| @ -70,19 +70,19 @@ expression: actual | ||||
|     }, | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 55, | ||||
|         "id": { | ||||
|         "end": 43, | ||||
|         "id": { | ||||
|           "end": 31, | ||||
|           "name": "height", | ||||
|           "start": 37, | ||||
|           "start": 25, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "end": 55, | ||||
|           "end": 43, | ||||
|           "left": { | ||||
|             "end": 47, | ||||
|             "end": 35, | ||||
|             "raw": "1", | ||||
|             "start": 46, | ||||
|             "start": 34, | ||||
|             "type": "Literal", | ||||
|             "type": "Literal", | ||||
|             "value": 1.0 | ||||
| @ -90,39 +90,39 @@ expression: actual | ||||
|           "operator": "-", | ||||
|           "right": { | ||||
|             "computed": false, | ||||
|             "end": 55, | ||||
|             "end": 43, | ||||
|             "object": { | ||||
|               "end": 53, | ||||
|               "end": 41, | ||||
|               "name": "obj", | ||||
|               "start": 50, | ||||
|               "start": 38, | ||||
|               "type": "Identifier", | ||||
|               "type": "Identifier" | ||||
|             }, | ||||
|             "property": { | ||||
|               "end": 55, | ||||
|               "end": 43, | ||||
|               "name": "a", | ||||
|               "start": 54, | ||||
|               "start": 42, | ||||
|               "type": "Identifier", | ||||
|               "type": "Identifier" | ||||
|             }, | ||||
|             "start": 50, | ||||
|             "start": 38, | ||||
|             "type": "MemberExpression", | ||||
|             "type": "MemberExpression" | ||||
|           }, | ||||
|           "start": 46, | ||||
|           "start": 34, | ||||
|           "type": "BinaryExpression", | ||||
|           "type": "BinaryExpression" | ||||
|         }, | ||||
|         "start": 37, | ||||
|         "start": 25, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 55, | ||||
|       "end": 43, | ||||
|       "kind": "const", | ||||
|       "start": 31, | ||||
|       "start": 25, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 55, | ||||
|   "end": 43, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,63 +6,63 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 26, | ||||
|         "end": 20, | ||||
|         "id": { | ||||
|           "end": 9, | ||||
|           "end": 3, | ||||
|           "name": "obj", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "end": 26, | ||||
|           "end": 20, | ||||
|           "properties": [ | ||||
|             { | ||||
|               "end": 18, | ||||
|               "end": 12, | ||||
|               "key": { | ||||
|                 "end": 15, | ||||
|                 "end": 9, | ||||
|                 "name": "a", | ||||
|                 "start": 14, | ||||
|                 "start": 8, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "start": 14, | ||||
|               "start": 8, | ||||
|               "type": "ObjectProperty", | ||||
|               "value": { | ||||
|                 "end": 18, | ||||
|                 "end": 12, | ||||
|                 "raw": "1", | ||||
|                 "start": 17, | ||||
|                 "start": 11, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 1.0 | ||||
|               } | ||||
|             }, | ||||
|             { | ||||
|               "end": 24, | ||||
|               "end": 18, | ||||
|               "key": { | ||||
|                 "end": 21, | ||||
|                 "end": 15, | ||||
|                 "name": "b", | ||||
|                 "start": 20, | ||||
|                 "start": 14, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "start": 20, | ||||
|               "start": 14, | ||||
|               "type": "ObjectProperty", | ||||
|               "value": { | ||||
|                 "end": 24, | ||||
|                 "end": 18, | ||||
|                 "raw": "2", | ||||
|                 "start": 23, | ||||
|                 "start": 17, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 2.0 | ||||
|               } | ||||
|             } | ||||
|           ], | ||||
|           "start": 12, | ||||
|           "start": 6, | ||||
|           "type": "ObjectExpression", | ||||
|           "type": "ObjectExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 26, | ||||
|       "end": 20, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
| @ -70,19 +70,19 @@ expression: actual | ||||
|     }, | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 59, | ||||
|         "end": 47, | ||||
|         "id": { | ||||
|           "end": 44, | ||||
|           "end": 32, | ||||
|           "name": "height", | ||||
|           "start": 38, | ||||
|           "start": 26, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "end": 59, | ||||
|           "end": 47, | ||||
|           "left": { | ||||
|             "end": 48, | ||||
|             "end": 36, | ||||
|             "raw": "1", | ||||
|             "start": 47, | ||||
|             "start": 35, | ||||
|             "type": "Literal", | ||||
|             "type": "Literal", | ||||
|             "value": 1.0 | ||||
| @ -90,40 +90,40 @@ expression: actual | ||||
|           "operator": "-", | ||||
|           "right": { | ||||
|             "computed": false, | ||||
|             "end": 59, | ||||
|             "end": 47, | ||||
|             "object": { | ||||
|               "end": 54, | ||||
|               "end": 42, | ||||
|               "name": "obj", | ||||
|               "start": 51, | ||||
|               "start": 39, | ||||
|               "type": "Identifier", | ||||
|               "type": "Identifier" | ||||
|             }, | ||||
|             "property": { | ||||
|               "end": 58, | ||||
|               "end": 46, | ||||
|               "raw": "\"a\"", | ||||
|               "start": 55, | ||||
|               "start": 43, | ||||
|               "type": "Literal", | ||||
|               "type": "Literal", | ||||
|               "value": "a" | ||||
|             }, | ||||
|             "start": 51, | ||||
|             "start": 39, | ||||
|             "type": "MemberExpression", | ||||
|             "type": "MemberExpression" | ||||
|           }, | ||||
|           "start": 47, | ||||
|           "start": 35, | ||||
|           "type": "BinaryExpression", | ||||
|           "type": "BinaryExpression" | ||||
|         }, | ||||
|         "start": 38, | ||||
|         "start": 26, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 59, | ||||
|       "end": 47, | ||||
|       "kind": "const", | ||||
|       "start": 32, | ||||
|       "start": 26, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 59, | ||||
|   "end": 47, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,63 +6,63 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 26, | ||||
|         "end": 20, | ||||
|         "id": { | ||||
|           "end": 9, | ||||
|           "end": 3, | ||||
|           "name": "obj", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "end": 26, | ||||
|           "end": 20, | ||||
|           "properties": [ | ||||
|             { | ||||
|               "end": 18, | ||||
|               "end": 12, | ||||
|               "key": { | ||||
|                 "end": 15, | ||||
|                 "end": 9, | ||||
|                 "name": "a", | ||||
|                 "start": 14, | ||||
|                 "start": 8, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "start": 14, | ||||
|               "start": 8, | ||||
|               "type": "ObjectProperty", | ||||
|               "value": { | ||||
|                 "end": 18, | ||||
|                 "end": 12, | ||||
|                 "raw": "1", | ||||
|                 "start": 17, | ||||
|                 "start": 11, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 1.0 | ||||
|               } | ||||
|             }, | ||||
|             { | ||||
|               "end": 24, | ||||
|               "end": 18, | ||||
|               "key": { | ||||
|                 "end": 21, | ||||
|                 "end": 15, | ||||
|                 "name": "b", | ||||
|                 "start": 20, | ||||
|                 "start": 14, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "start": 20, | ||||
|               "start": 14, | ||||
|               "type": "ObjectProperty", | ||||
|               "value": { | ||||
|                 "end": 24, | ||||
|                 "end": 18, | ||||
|                 "raw": "2", | ||||
|                 "start": 23, | ||||
|                 "start": 17, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 2.0 | ||||
|               } | ||||
|             } | ||||
|           ], | ||||
|           "start": 12, | ||||
|           "start": 6, | ||||
|           "type": "ObjectExpression", | ||||
|           "type": "ObjectExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 26, | ||||
|       "end": 20, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
| @ -70,60 +70,60 @@ expression: actual | ||||
|     }, | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 58, | ||||
|         "end": 46, | ||||
|         "id": { | ||||
|           "end": 43, | ||||
|           "end": 31, | ||||
|           "name": "height", | ||||
|           "start": 37, | ||||
|           "start": 25, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "end": 58, | ||||
|           "end": 46, | ||||
|           "left": { | ||||
|             "computed": false, | ||||
|             "end": 54, | ||||
|             "end": 42, | ||||
|             "object": { | ||||
|               "end": 49, | ||||
|               "end": 37, | ||||
|               "name": "obj", | ||||
|               "start": 46, | ||||
|               "start": 34, | ||||
|               "type": "Identifier", | ||||
|               "type": "Identifier" | ||||
|             }, | ||||
|             "property": { | ||||
|               "end": 53, | ||||
|               "end": 41, | ||||
|               "raw": "\"a\"", | ||||
|               "start": 50, | ||||
|               "start": 38, | ||||
|               "type": "Literal", | ||||
|               "type": "Literal", | ||||
|               "value": "a" | ||||
|             }, | ||||
|             "start": 46, | ||||
|             "start": 34, | ||||
|             "type": "MemberExpression", | ||||
|             "type": "MemberExpression" | ||||
|           }, | ||||
|           "operator": "-", | ||||
|           "right": { | ||||
|             "end": 58, | ||||
|             "end": 46, | ||||
|             "raw": "1", | ||||
|             "start": 57, | ||||
|             "start": 45, | ||||
|             "type": "Literal", | ||||
|             "type": "Literal", | ||||
|             "value": 1.0 | ||||
|           }, | ||||
|           "start": 46, | ||||
|           "start": 34, | ||||
|           "type": "BinaryExpression", | ||||
|           "type": "BinaryExpression" | ||||
|         }, | ||||
|         "start": 37, | ||||
|         "start": 25, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 58, | ||||
|       "end": 46, | ||||
|       "kind": "const", | ||||
|       "start": 31, | ||||
|       "start": 25, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 58, | ||||
|   "end": 46, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,63 +6,63 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 26, | ||||
|         "end": 20, | ||||
|         "id": { | ||||
|           "end": 9, | ||||
|           "end": 3, | ||||
|           "name": "obj", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "end": 26, | ||||
|           "end": 20, | ||||
|           "properties": [ | ||||
|             { | ||||
|               "end": 18, | ||||
|               "end": 12, | ||||
|               "key": { | ||||
|                 "end": 15, | ||||
|                 "end": 9, | ||||
|                 "name": "a", | ||||
|                 "start": 14, | ||||
|                 "start": 8, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "start": 14, | ||||
|               "start": 8, | ||||
|               "type": "ObjectProperty", | ||||
|               "value": { | ||||
|                 "end": 18, | ||||
|                 "end": 12, | ||||
|                 "raw": "1", | ||||
|                 "start": 17, | ||||
|                 "start": 11, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 1.0 | ||||
|               } | ||||
|             }, | ||||
|             { | ||||
|               "end": 24, | ||||
|               "end": 18, | ||||
|               "key": { | ||||
|                 "end": 21, | ||||
|                 "end": 15, | ||||
|                 "name": "b", | ||||
|                 "start": 20, | ||||
|                 "start": 14, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "start": 20, | ||||
|               "start": 14, | ||||
|               "type": "ObjectProperty", | ||||
|               "value": { | ||||
|                 "end": 24, | ||||
|                 "end": 18, | ||||
|                 "raw": "2", | ||||
|                 "start": 23, | ||||
|                 "start": 17, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 2.0 | ||||
|               } | ||||
|             } | ||||
|           ], | ||||
|           "start": 12, | ||||
|           "start": 6, | ||||
|           "type": "ObjectExpression", | ||||
|           "type": "ObjectExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 26, | ||||
|       "end": 20, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
| @ -70,21 +70,21 @@ expression: actual | ||||
|     }, | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 63, | ||||
|         "end": 51, | ||||
|         "id": { | ||||
|           "end": 43, | ||||
|           "end": 31, | ||||
|           "name": "height", | ||||
|           "start": 37, | ||||
|           "start": 25, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "elements": [ | ||||
|             { | ||||
|               "end": 59, | ||||
|               "end": 47, | ||||
|               "left": { | ||||
|                 "end": 48, | ||||
|                 "end": 36, | ||||
|                 "raw": "1", | ||||
|                 "start": 47, | ||||
|                 "start": 35, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 1.0 | ||||
| @ -92,54 +92,54 @@ expression: actual | ||||
|               "operator": "-", | ||||
|               "right": { | ||||
|                 "computed": false, | ||||
|                 "end": 59, | ||||
|                 "end": 47, | ||||
|                 "object": { | ||||
|                   "end": 54, | ||||
|                   "end": 42, | ||||
|                   "name": "obj", | ||||
|                   "start": 51, | ||||
|                   "start": 39, | ||||
|                   "type": "Identifier", | ||||
|                   "type": "Identifier" | ||||
|                 }, | ||||
|                 "property": { | ||||
|                   "end": 58, | ||||
|                   "end": 46, | ||||
|                   "raw": "\"a\"", | ||||
|                   "start": 55, | ||||
|                   "start": 43, | ||||
|                   "type": "Literal", | ||||
|                   "type": "Literal", | ||||
|                   "value": "a" | ||||
|                 }, | ||||
|                 "start": 51, | ||||
|                 "start": 39, | ||||
|                 "type": "MemberExpression", | ||||
|                 "type": "MemberExpression" | ||||
|               }, | ||||
|               "start": 47, | ||||
|               "start": 35, | ||||
|               "type": "BinaryExpression", | ||||
|               "type": "BinaryExpression" | ||||
|             }, | ||||
|             { | ||||
|               "end": 62, | ||||
|               "end": 50, | ||||
|               "raw": "0", | ||||
|               "start": 61, | ||||
|               "start": 49, | ||||
|               "type": "Literal", | ||||
|               "type": "Literal", | ||||
|               "value": 0.0 | ||||
|             } | ||||
|           ], | ||||
|           "end": 63, | ||||
|           "start": 46, | ||||
|           "end": 51, | ||||
|           "start": 34, | ||||
|           "type": "ArrayExpression", | ||||
|           "type": "ArrayExpression" | ||||
|         }, | ||||
|         "start": 37, | ||||
|         "start": 25, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 63, | ||||
|       "end": 51, | ||||
|       "kind": "const", | ||||
|       "start": 31, | ||||
|       "start": 25, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 63, | ||||
|   "end": 51, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -0,0 +1,76 @@ | ||||
| --- | ||||
| source: kcl/src/parsing/parser.rs | ||||
| expression: actual | ||||
| snapshot_kind: text | ||||
| --- | ||||
| { | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 35, | ||||
|         "id": { | ||||
|           "end": 6, | ||||
|           "name": "foo", | ||||
|           "start": 3, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "body": { | ||||
|             "body": [ | ||||
|               { | ||||
|                 "argument": { | ||||
|                   "end": 33, | ||||
|                   "raw": "1", | ||||
|                   "start": 32, | ||||
|                   "type": "Literal", | ||||
|                   "type": "Literal", | ||||
|                   "value": 1.0 | ||||
|                 }, | ||||
|                 "end": 33, | ||||
|                 "start": 25, | ||||
|                 "type": "ReturnStatement", | ||||
|                 "type": "ReturnStatement" | ||||
|               } | ||||
|             ], | ||||
|             "end": 35, | ||||
|             "start": 23 | ||||
|           }, | ||||
|           "end": 35, | ||||
|           "params": [ | ||||
|             { | ||||
|               "type": "Parameter", | ||||
|               "identifier": { | ||||
|                 "end": 8, | ||||
|                 "name": "x", | ||||
|                 "start": 7, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "type_": { | ||||
|                 "type": "Primitive", | ||||
|                 "type": "Number" | ||||
|               }, | ||||
|               "default_value": { | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 2.0, | ||||
|                 "raw": "2" | ||||
|               } | ||||
|             } | ||||
|           ], | ||||
|           "start": 6, | ||||
|           "type": "FunctionExpression", | ||||
|           "type": "FunctionExpression" | ||||
|         }, | ||||
|         "start": 3, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 35, | ||||
|       "kind": "fn", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 35, | ||||
|   "start": 0 | ||||
| } | ||||
| @ -0,0 +1,72 @@ | ||||
| --- | ||||
| source: kcl/src/parsing/parser.rs | ||||
| expression: actual | ||||
| snapshot_kind: text | ||||
| --- | ||||
| { | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 27, | ||||
|         "id": { | ||||
|           "end": 6, | ||||
|           "name": "foo", | ||||
|           "start": 3, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "body": { | ||||
|             "body": [ | ||||
|               { | ||||
|                 "argument": { | ||||
|                   "end": 25, | ||||
|                   "raw": "1", | ||||
|                   "start": 24, | ||||
|                   "type": "Literal", | ||||
|                   "type": "Literal", | ||||
|                   "value": 1.0 | ||||
|                 }, | ||||
|                 "end": 25, | ||||
|                 "start": 17, | ||||
|                 "type": "ReturnStatement", | ||||
|                 "type": "ReturnStatement" | ||||
|               } | ||||
|             ], | ||||
|             "end": 27, | ||||
|             "start": 15 | ||||
|           }, | ||||
|           "end": 27, | ||||
|           "params": [ | ||||
|             { | ||||
|               "type": "Parameter", | ||||
|               "identifier": { | ||||
|                 "end": 8, | ||||
|                 "name": "x", | ||||
|                 "start": 7, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "default_value": { | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 2.0, | ||||
|                 "raw": "2" | ||||
|               } | ||||
|             } | ||||
|           ], | ||||
|           "start": 6, | ||||
|           "type": "FunctionExpression", | ||||
|           "type": "FunctionExpression" | ||||
|         }, | ||||
|         "start": 3, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 27, | ||||
|       "kind": "fn", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 27, | ||||
|   "start": 0 | ||||
| } | ||||
| @ -6,63 +6,63 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 26, | ||||
|         "end": 20, | ||||
|         "id": { | ||||
|           "end": 9, | ||||
|           "end": 3, | ||||
|           "name": "obj", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "end": 26, | ||||
|           "end": 20, | ||||
|           "properties": [ | ||||
|             { | ||||
|               "end": 18, | ||||
|               "end": 12, | ||||
|               "key": { | ||||
|                 "end": 15, | ||||
|                 "end": 9, | ||||
|                 "name": "a", | ||||
|                 "start": 14, | ||||
|                 "start": 8, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "start": 14, | ||||
|               "start": 8, | ||||
|               "type": "ObjectProperty", | ||||
|               "value": { | ||||
|                 "end": 18, | ||||
|                 "end": 12, | ||||
|                 "raw": "1", | ||||
|                 "start": 17, | ||||
|                 "start": 11, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 1.0 | ||||
|               } | ||||
|             }, | ||||
|             { | ||||
|               "end": 24, | ||||
|               "end": 18, | ||||
|               "key": { | ||||
|                 "end": 21, | ||||
|                 "end": 15, | ||||
|                 "name": "b", | ||||
|                 "start": 20, | ||||
|                 "start": 14, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "start": 20, | ||||
|               "start": 14, | ||||
|               "type": "ObjectProperty", | ||||
|               "value": { | ||||
|                 "end": 24, | ||||
|                 "end": 18, | ||||
|                 "raw": "2", | ||||
|                 "start": 23, | ||||
|                 "start": 17, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 2.0 | ||||
|               } | ||||
|             } | ||||
|           ], | ||||
|           "start": 12, | ||||
|           "start": 6, | ||||
|           "type": "ObjectExpression", | ||||
|           "type": "ObjectExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 26, | ||||
|       "end": 20, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
| @ -70,76 +70,76 @@ expression: actual | ||||
|     }, | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 63, | ||||
|         "end": 51, | ||||
|         "id": { | ||||
|           "end": 43, | ||||
|           "end": 31, | ||||
|           "name": "height", | ||||
|           "start": 37, | ||||
|           "start": 25, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "elements": [ | ||||
|             { | ||||
|               "end": 59, | ||||
|               "end": 47, | ||||
|               "left": { | ||||
|                 "computed": false, | ||||
|                 "end": 55, | ||||
|                 "end": 43, | ||||
|                 "object": { | ||||
|                   "end": 50, | ||||
|                   "end": 38, | ||||
|                   "name": "obj", | ||||
|                   "start": 47, | ||||
|                   "start": 35, | ||||
|                   "type": "Identifier", | ||||
|                   "type": "Identifier" | ||||
|                 }, | ||||
|                 "property": { | ||||
|                   "end": 54, | ||||
|                   "end": 42, | ||||
|                   "raw": "\"a\"", | ||||
|                   "start": 51, | ||||
|                   "start": 39, | ||||
|                   "type": "Literal", | ||||
|                   "type": "Literal", | ||||
|                   "value": "a" | ||||
|                 }, | ||||
|                 "start": 47, | ||||
|                 "start": 35, | ||||
|                 "type": "MemberExpression", | ||||
|                 "type": "MemberExpression" | ||||
|               }, | ||||
|               "operator": "-", | ||||
|               "right": { | ||||
|                 "end": 59, | ||||
|                 "end": 47, | ||||
|                 "raw": "1", | ||||
|                 "start": 58, | ||||
|                 "start": 46, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 1.0 | ||||
|               }, | ||||
|               "start": 47, | ||||
|               "start": 35, | ||||
|               "type": "BinaryExpression", | ||||
|               "type": "BinaryExpression" | ||||
|             }, | ||||
|             { | ||||
|               "end": 62, | ||||
|               "end": 50, | ||||
|               "raw": "0", | ||||
|               "start": 61, | ||||
|               "start": 49, | ||||
|               "type": "Literal", | ||||
|               "type": "Literal", | ||||
|               "value": 0.0 | ||||
|             } | ||||
|           ], | ||||
|           "end": 63, | ||||
|           "start": 46, | ||||
|           "end": 51, | ||||
|           "start": 34, | ||||
|           "type": "ArrayExpression", | ||||
|           "type": "ArrayExpression" | ||||
|         }, | ||||
|         "start": 37, | ||||
|         "start": 25, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 63, | ||||
|       "end": 51, | ||||
|       "kind": "const", | ||||
|       "start": 31, | ||||
|       "start": 25, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 63, | ||||
|   "end": 51, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,63 +6,63 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 26, | ||||
|         "end": 20, | ||||
|         "id": { | ||||
|           "end": 9, | ||||
|           "end": 3, | ||||
|           "name": "obj", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "end": 26, | ||||
|           "end": 20, | ||||
|           "properties": [ | ||||
|             { | ||||
|               "end": 18, | ||||
|               "end": 12, | ||||
|               "key": { | ||||
|                 "end": 15, | ||||
|                 "end": 9, | ||||
|                 "name": "a", | ||||
|                 "start": 14, | ||||
|                 "start": 8, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "start": 14, | ||||
|               "start": 8, | ||||
|               "type": "ObjectProperty", | ||||
|               "value": { | ||||
|                 "end": 18, | ||||
|                 "end": 12, | ||||
|                 "raw": "1", | ||||
|                 "start": 17, | ||||
|                 "start": 11, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 1.0 | ||||
|               } | ||||
|             }, | ||||
|             { | ||||
|               "end": 24, | ||||
|               "end": 18, | ||||
|               "key": { | ||||
|                 "end": 21, | ||||
|                 "end": 15, | ||||
|                 "name": "b", | ||||
|                 "start": 20, | ||||
|                 "start": 14, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "start": 20, | ||||
|               "start": 14, | ||||
|               "type": "ObjectProperty", | ||||
|               "value": { | ||||
|                 "end": 24, | ||||
|                 "end": 18, | ||||
|                 "raw": "2", | ||||
|                 "start": 23, | ||||
|                 "start": 17, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 2.0 | ||||
|               } | ||||
|             } | ||||
|           ], | ||||
|           "start": 12, | ||||
|           "start": 6, | ||||
|           "type": "ObjectExpression", | ||||
|           "type": "ObjectExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 26, | ||||
|       "end": 20, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
| @ -70,76 +70,76 @@ expression: actual | ||||
|     }, | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 62, | ||||
|         "end": 50, | ||||
|         "id": { | ||||
|           "end": 43, | ||||
|           "end": 31, | ||||
|           "name": "height", | ||||
|           "start": 37, | ||||
|           "start": 25, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "elements": [ | ||||
|             { | ||||
|               "end": 58, | ||||
|               "end": 46, | ||||
|               "left": { | ||||
|                 "computed": false, | ||||
|                 "end": 55, | ||||
|                 "end": 43, | ||||
|                 "object": { | ||||
|                   "end": 50, | ||||
|                   "end": 38, | ||||
|                   "name": "obj", | ||||
|                   "start": 47, | ||||
|                   "start": 35, | ||||
|                   "type": "Identifier", | ||||
|                   "type": "Identifier" | ||||
|                 }, | ||||
|                 "property": { | ||||
|                   "end": 54, | ||||
|                   "end": 42, | ||||
|                   "raw": "\"a\"", | ||||
|                   "start": 51, | ||||
|                   "start": 39, | ||||
|                   "type": "Literal", | ||||
|                   "type": "Literal", | ||||
|                   "value": "a" | ||||
|                 }, | ||||
|                 "start": 47, | ||||
|                 "start": 35, | ||||
|                 "type": "MemberExpression", | ||||
|                 "type": "MemberExpression" | ||||
|               }, | ||||
|               "operator": "-", | ||||
|               "right": { | ||||
|                 "end": 58, | ||||
|                 "end": 46, | ||||
|                 "raw": "1", | ||||
|                 "start": 57, | ||||
|                 "start": 45, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 1.0 | ||||
|               }, | ||||
|               "start": 47, | ||||
|               "start": 35, | ||||
|               "type": "BinaryExpression", | ||||
|               "type": "BinaryExpression" | ||||
|             }, | ||||
|             { | ||||
|               "end": 61, | ||||
|               "end": 49, | ||||
|               "raw": "0", | ||||
|               "start": 60, | ||||
|               "start": 48, | ||||
|               "type": "Literal", | ||||
|               "type": "Literal", | ||||
|               "value": 0.0 | ||||
|             } | ||||
|           ], | ||||
|           "end": 62, | ||||
|           "start": 46, | ||||
|           "end": 50, | ||||
|           "start": 34, | ||||
|           "type": "ArrayExpression", | ||||
|           "type": "ArrayExpression" | ||||
|         }, | ||||
|         "start": 37, | ||||
|         "start": 25, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 62, | ||||
|       "end": 50, | ||||
|       "kind": "const", | ||||
|       "start": 31, | ||||
|       "start": 25, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 62, | ||||
|   "end": 50, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,19 +6,19 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 24, | ||||
|         "end": 18, | ||||
|         "id": { | ||||
|           "end": 12, | ||||
|           "end": 6, | ||||
|           "name": "height", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "end": 24, | ||||
|           "end": 18, | ||||
|           "left": { | ||||
|             "end": 16, | ||||
|             "end": 10, | ||||
|             "raw": "1", | ||||
|             "start": 15, | ||||
|             "start": 9, | ||||
|             "type": "Literal", | ||||
|             "type": "Literal", | ||||
|             "value": 1.0 | ||||
| @ -26,39 +26,39 @@ expression: actual | ||||
|           "operator": "-", | ||||
|           "right": { | ||||
|             "computed": false, | ||||
|             "end": 24, | ||||
|             "end": 18, | ||||
|             "object": { | ||||
|               "end": 22, | ||||
|               "end": 16, | ||||
|               "name": "obj", | ||||
|               "start": 19, | ||||
|               "start": 13, | ||||
|               "type": "Identifier", | ||||
|               "type": "Identifier" | ||||
|             }, | ||||
|             "property": { | ||||
|               "end": 24, | ||||
|               "end": 18, | ||||
|               "name": "a", | ||||
|               "start": 23, | ||||
|               "start": 17, | ||||
|               "type": "Identifier", | ||||
|               "type": "Identifier" | ||||
|             }, | ||||
|             "start": 19, | ||||
|             "start": 13, | ||||
|             "type": "MemberExpression", | ||||
|             "type": "MemberExpression" | ||||
|           }, | ||||
|           "start": 15, | ||||
|           "start": 9, | ||||
|           "type": "BinaryExpression", | ||||
|           "type": "BinaryExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 24, | ||||
|       "end": 18, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 24, | ||||
|   "end": 18, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,61 +6,61 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 21, | ||||
|         "end": 15, | ||||
|         "id": { | ||||
|           "end": 9, | ||||
|           "end": 3, | ||||
|           "name": "six", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "end": 21, | ||||
|           "end": 15, | ||||
|           "left": { | ||||
|             "end": 17, | ||||
|             "end": 11, | ||||
|             "left": { | ||||
|               "end": 13, | ||||
|               "end": 7, | ||||
|               "raw": "1", | ||||
|               "start": 12, | ||||
|               "start": 6, | ||||
|               "type": "Literal", | ||||
|               "type": "Literal", | ||||
|               "value": 1.0 | ||||
|             }, | ||||
|             "operator": "+", | ||||
|             "right": { | ||||
|               "end": 17, | ||||
|               "end": 11, | ||||
|               "raw": "2", | ||||
|               "start": 16, | ||||
|               "start": 10, | ||||
|               "type": "Literal", | ||||
|               "type": "Literal", | ||||
|               "value": 2.0 | ||||
|             }, | ||||
|             "start": 12, | ||||
|             "start": 6, | ||||
|             "type": "BinaryExpression", | ||||
|             "type": "BinaryExpression" | ||||
|           }, | ||||
|           "operator": "+", | ||||
|           "right": { | ||||
|             "end": 21, | ||||
|             "end": 15, | ||||
|             "raw": "3", | ||||
|             "start": 20, | ||||
|             "start": 14, | ||||
|             "type": "Literal", | ||||
|             "type": "Literal", | ||||
|             "value": 3.0 | ||||
|           }, | ||||
|           "start": 12, | ||||
|           "start": 6, | ||||
|           "type": "BinaryExpression", | ||||
|           "type": "BinaryExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 21, | ||||
|       "end": 15, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 21, | ||||
|   "end": 15, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,61 +6,61 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 22, | ||||
|         "end": 16, | ||||
|         "id": { | ||||
|           "end": 10, | ||||
|           "end": 4, | ||||
|           "name": "five", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "end": 22, | ||||
|           "end": 16, | ||||
|           "left": { | ||||
|             "end": 18, | ||||
|             "end": 12, | ||||
|             "left": { | ||||
|               "end": 14, | ||||
|               "end": 8, | ||||
|               "raw": "3", | ||||
|               "start": 13, | ||||
|               "start": 7, | ||||
|               "type": "Literal", | ||||
|               "type": "Literal", | ||||
|               "value": 3.0 | ||||
|             }, | ||||
|             "operator": "*", | ||||
|             "right": { | ||||
|               "end": 18, | ||||
|               "end": 12, | ||||
|               "raw": "1", | ||||
|               "start": 17, | ||||
|               "start": 11, | ||||
|               "type": "Literal", | ||||
|               "type": "Literal", | ||||
|               "value": 1.0 | ||||
|             }, | ||||
|             "start": 13, | ||||
|             "start": 7, | ||||
|             "type": "BinaryExpression", | ||||
|             "type": "BinaryExpression" | ||||
|           }, | ||||
|           "operator": "+", | ||||
|           "right": { | ||||
|             "end": 22, | ||||
|             "end": 16, | ||||
|             "raw": "2", | ||||
|             "start": 21, | ||||
|             "start": 15, | ||||
|             "type": "Literal", | ||||
|             "type": "Literal", | ||||
|             "value": 2.0 | ||||
|           }, | ||||
|           "start": 13, | ||||
|           "start": 7, | ||||
|           "type": "BinaryExpression", | ||||
|           "type": "BinaryExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 22, | ||||
|       "end": 16, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 22, | ||||
|   "end": 16, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,61 +6,61 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 30, | ||||
|         "end": 24, | ||||
|         "id": { | ||||
|           "end": 12, | ||||
|           "end": 6, | ||||
|           "name": "height", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "elements": [ | ||||
|             { | ||||
|               "computed": false, | ||||
|               "end": 25, | ||||
|               "end": 19, | ||||
|               "object": { | ||||
|                 "end": 20, | ||||
|                 "end": 14, | ||||
|                 "name": "obj", | ||||
|                 "start": 17, | ||||
|                 "start": 11, | ||||
|                 "type": "Identifier", | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "property": { | ||||
|                 "end": 24, | ||||
|                 "end": 18, | ||||
|                 "raw": "\"a\"", | ||||
|                 "start": 21, | ||||
|                 "start": 15, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": "a" | ||||
|               }, | ||||
|               "start": 17, | ||||
|               "start": 11, | ||||
|               "type": "MemberExpression", | ||||
|               "type": "MemberExpression" | ||||
|             }, | ||||
|             { | ||||
|               "end": 28, | ||||
|               "end": 22, | ||||
|               "raw": "0", | ||||
|               "start": 27, | ||||
|               "start": 21, | ||||
|               "type": "Literal", | ||||
|               "type": "Literal", | ||||
|               "value": 0.0 | ||||
|             } | ||||
|           ], | ||||
|           "end": 30, | ||||
|           "start": 15, | ||||
|           "end": 24, | ||||
|           "start": 9, | ||||
|           "type": "ArrayExpression", | ||||
|           "type": "ArrayExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 30, | ||||
|       "end": 24, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 30, | ||||
|   "end": 24, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,63 +6,63 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 26, | ||||
|         "end": 20, | ||||
|         "id": { | ||||
|           "end": 9, | ||||
|           "end": 3, | ||||
|           "name": "obj", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "end": 26, | ||||
|           "end": 20, | ||||
|           "properties": [ | ||||
|             { | ||||
|               "end": 18, | ||||
|               "end": 12, | ||||
|               "key": { | ||||
|                 "end": 15, | ||||
|                 "end": 9, | ||||
|                 "name": "a", | ||||
|                 "start": 14, | ||||
|                 "start": 8, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "start": 14, | ||||
|               "start": 8, | ||||
|               "type": "ObjectProperty", | ||||
|               "value": { | ||||
|                 "end": 18, | ||||
|                 "end": 12, | ||||
|                 "raw": "1", | ||||
|                 "start": 17, | ||||
|                 "start": 11, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 1.0 | ||||
|               } | ||||
|             }, | ||||
|             { | ||||
|               "end": 24, | ||||
|               "end": 18, | ||||
|               "key": { | ||||
|                 "end": 21, | ||||
|                 "end": 15, | ||||
|                 "name": "b", | ||||
|                 "start": 20, | ||||
|                 "start": 14, | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "start": 20, | ||||
|               "start": 14, | ||||
|               "type": "ObjectProperty", | ||||
|               "value": { | ||||
|                 "end": 24, | ||||
|                 "end": 18, | ||||
|                 "raw": "2", | ||||
|                 "start": 23, | ||||
|                 "start": 17, | ||||
|                 "type": "Literal", | ||||
|                 "type": "Literal", | ||||
|                 "value": 2.0 | ||||
|               } | ||||
|             } | ||||
|           ], | ||||
|           "start": 12, | ||||
|           "start": 6, | ||||
|           "type": "ObjectExpression", | ||||
|           "type": "ObjectExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 26, | ||||
|       "end": 20, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
| @ -70,45 +70,45 @@ expression: actual | ||||
|     }, | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 54, | ||||
|         "end": 42, | ||||
|         "id": { | ||||
|           "end": 43, | ||||
|           "end": 31, | ||||
|           "name": "height", | ||||
|           "start": 37, | ||||
|           "start": 25, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "computed": false, | ||||
|           "end": 54, | ||||
|           "end": 42, | ||||
|           "object": { | ||||
|             "end": 49, | ||||
|             "end": 37, | ||||
|             "name": "obj", | ||||
|             "start": 46, | ||||
|             "start": 34, | ||||
|             "type": "Identifier", | ||||
|             "type": "Identifier" | ||||
|           }, | ||||
|           "property": { | ||||
|             "end": 53, | ||||
|             "end": 41, | ||||
|             "raw": "\"a\"", | ||||
|             "start": 50, | ||||
|             "start": 38, | ||||
|             "type": "Literal", | ||||
|             "type": "Literal", | ||||
|             "value": "a" | ||||
|           }, | ||||
|           "start": 46, | ||||
|           "start": 34, | ||||
|           "type": "MemberExpression", | ||||
|           "type": "MemberExpression" | ||||
|         }, | ||||
|         "start": 37, | ||||
|         "start": 25, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 54, | ||||
|       "end": 42, | ||||
|       "kind": "const", | ||||
|       "start": 31, | ||||
|       "start": 25, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 54, | ||||
|   "end": 42, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,59 +6,59 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 27, | ||||
|         "end": 21, | ||||
|         "id": { | ||||
|           "end": 10, | ||||
|           "end": 4, | ||||
|           "name": "prop", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "computed": true, | ||||
|           "end": 27, | ||||
|           "end": 21, | ||||
|           "object": { | ||||
|             "computed": false, | ||||
|             "end": 22, | ||||
|             "end": 16, | ||||
|             "object": { | ||||
|               "end": 15, | ||||
|               "end": 9, | ||||
|               "name": "yo", | ||||
|               "start": 13, | ||||
|               "start": 7, | ||||
|               "type": "Identifier", | ||||
|               "type": "Identifier" | ||||
|             }, | ||||
|             "property": { | ||||
|               "end": 21, | ||||
|               "end": 15, | ||||
|               "raw": "\"one\"", | ||||
|               "start": 16, | ||||
|               "start": 10, | ||||
|               "type": "Literal", | ||||
|               "type": "Literal", | ||||
|               "value": "one" | ||||
|             }, | ||||
|             "start": 13, | ||||
|             "start": 7, | ||||
|             "type": "MemberExpression", | ||||
|             "type": "MemberExpression" | ||||
|           }, | ||||
|           "property": { | ||||
|             "end": 26, | ||||
|             "end": 20, | ||||
|             "name": "two", | ||||
|             "start": 23, | ||||
|             "start": 17, | ||||
|             "type": "Identifier", | ||||
|             "type": "Identifier" | ||||
|           }, | ||||
|           "start": 13, | ||||
|           "start": 7, | ||||
|           "type": "MemberExpression", | ||||
|           "type": "MemberExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 27, | ||||
|       "end": 21, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 27, | ||||
|   "end": 21, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,44 +6,44 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 17, | ||||
|         "end": 11, | ||||
|         "id": { | ||||
|           "end": 9, | ||||
|           "end": 3, | ||||
|           "name": "pt1", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "computed": true, | ||||
|           "end": 17, | ||||
|           "end": 11, | ||||
|           "object": { | ||||
|             "end": 14, | ||||
|             "end": 8, | ||||
|             "name": "b1", | ||||
|             "start": 12, | ||||
|             "start": 6, | ||||
|             "type": "Identifier", | ||||
|             "type": "Identifier" | ||||
|           }, | ||||
|           "property": { | ||||
|             "end": 16, | ||||
|             "end": 10, | ||||
|             "name": "x", | ||||
|             "start": 15, | ||||
|             "start": 9, | ||||
|             "type": "Identifier", | ||||
|             "type": "Identifier" | ||||
|           }, | ||||
|           "start": 12, | ||||
|           "start": 6, | ||||
|           "type": "MemberExpression", | ||||
|           "type": "MemberExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 17, | ||||
|       "end": 11, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 17, | ||||
|   "end": 11, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,86 +6,86 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 34, | ||||
|         "end": 28, | ||||
|         "id": { | ||||
|           "end": 10, | ||||
|           "end": 4, | ||||
|           "name": "prop", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "computed": false, | ||||
|           "end": 34, | ||||
|           "object": { | ||||
|             "computed": false, | ||||
|             "end": 29, | ||||
|           "end": 28, | ||||
|           "object": { | ||||
|             "computed": false, | ||||
|             "end": 23, | ||||
|             "object": { | ||||
|               "computed": false, | ||||
|                 "end": 19, | ||||
|               "end": 17, | ||||
|               "object": { | ||||
|                   "end": 15, | ||||
|                 "computed": false, | ||||
|                 "end": 13, | ||||
|                 "object": { | ||||
|                   "end": 9, | ||||
|                   "name": "yo", | ||||
|                   "start": 13, | ||||
|                   "start": 7, | ||||
|                   "type": "Identifier", | ||||
|                   "type": "Identifier" | ||||
|                 }, | ||||
|                 "property": { | ||||
|                   "end": 19, | ||||
|                   "end": 13, | ||||
|                   "name": "one", | ||||
|                   "start": 16, | ||||
|                   "start": 10, | ||||
|                   "type": "Identifier", | ||||
|                   "type": "Identifier" | ||||
|                 }, | ||||
|                 "start": 13, | ||||
|                 "start": 7, | ||||
|                 "type": "MemberExpression", | ||||
|                 "type": "MemberExpression" | ||||
|               }, | ||||
|               "property": { | ||||
|                 "end": 17, | ||||
|                 "name": "two", | ||||
|                 "start": 14, | ||||
|                 "type": "Identifier", | ||||
|                 "type": "Identifier" | ||||
|               }, | ||||
|               "start": 7, | ||||
|               "type": "MemberExpression", | ||||
|               "type": "MemberExpression" | ||||
|             }, | ||||
|             "property": { | ||||
|               "end": 23, | ||||
|                 "name": "two", | ||||
|                 "start": 20, | ||||
|               "name": "three", | ||||
|               "start": 18, | ||||
|               "type": "Identifier", | ||||
|               "type": "Identifier" | ||||
|             }, | ||||
|               "start": 13, | ||||
|             "start": 7, | ||||
|             "type": "MemberExpression", | ||||
|             "type": "MemberExpression" | ||||
|           }, | ||||
|           "property": { | ||||
|               "end": 29, | ||||
|               "name": "three", | ||||
|             "end": 28, | ||||
|             "name": "four", | ||||
|             "start": 24, | ||||
|             "type": "Identifier", | ||||
|             "type": "Identifier" | ||||
|           }, | ||||
|             "start": 13, | ||||
|           "start": 7, | ||||
|           "type": "MemberExpression", | ||||
|           "type": "MemberExpression" | ||||
|         }, | ||||
|           "property": { | ||||
|             "end": 34, | ||||
|             "name": "four", | ||||
|             "start": 30, | ||||
|             "type": "Identifier", | ||||
|             "type": "Identifier" | ||||
|           }, | ||||
|           "start": 13, | ||||
|           "type": "MemberExpression", | ||||
|           "type": "MemberExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 34, | ||||
|       "end": 28, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 34, | ||||
|   "end": 28, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,45 +6,45 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 17, | ||||
|         "end": 11, | ||||
|         "id": { | ||||
|           "end": 9, | ||||
|           "end": 3, | ||||
|           "name": "pt1", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "computed": false, | ||||
|           "end": 17, | ||||
|           "end": 11, | ||||
|           "object": { | ||||
|             "end": 14, | ||||
|             "end": 8, | ||||
|             "name": "b1", | ||||
|             "start": 12, | ||||
|             "start": 6, | ||||
|             "type": "Identifier", | ||||
|             "type": "Identifier" | ||||
|           }, | ||||
|           "property": { | ||||
|             "end": 16, | ||||
|             "end": 10, | ||||
|             "raw": "0", | ||||
|             "start": 15, | ||||
|             "start": 9, | ||||
|             "type": "Literal", | ||||
|             "type": "Literal", | ||||
|             "value": 0.0 | ||||
|           }, | ||||
|           "start": 12, | ||||
|           "start": 6, | ||||
|           "type": "MemberExpression", | ||||
|           "type": "MemberExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 17, | ||||
|       "end": 11, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 17, | ||||
|   "end": 11, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
| @ -6,45 +6,45 @@ expression: actual | ||||
|   "body": [ | ||||
|     { | ||||
|       "declaration": { | ||||
|         "end": 22, | ||||
|         "end": 16, | ||||
|         "id": { | ||||
|           "end": 9, | ||||
|           "end": 3, | ||||
|           "name": "pt1", | ||||
|           "start": 6, | ||||
|           "start": 0, | ||||
|           "type": "Identifier" | ||||
|         }, | ||||
|         "init": { | ||||
|           "computed": false, | ||||
|           "end": 22, | ||||
|           "end": 16, | ||||
|           "object": { | ||||
|             "end": 14, | ||||
|             "end": 8, | ||||
|             "name": "b1", | ||||
|             "start": 12, | ||||
|             "start": 6, | ||||
|             "type": "Identifier", | ||||
|             "type": "Identifier" | ||||
|           }, | ||||
|           "property": { | ||||
|             "end": 21, | ||||
|             "end": 15, | ||||
|             "raw": "'zero'", | ||||
|             "start": 15, | ||||
|             "start": 9, | ||||
|             "type": "Literal", | ||||
|             "type": "Literal", | ||||
|             "value": "zero" | ||||
|           }, | ||||
|           "start": 12, | ||||
|           "start": 6, | ||||
|           "type": "MemberExpression", | ||||
|           "type": "MemberExpression" | ||||
|         }, | ||||
|         "start": 6, | ||||
|         "start": 0, | ||||
|         "type": "VariableDeclarator" | ||||
|       }, | ||||
|       "end": 22, | ||||
|       "end": 16, | ||||
|       "kind": "const", | ||||
|       "start": 0, | ||||
|       "type": "VariableDeclaration", | ||||
|       "type": "VariableDeclaration" | ||||
|     } | ||||
|   ], | ||||
|   "end": 22, | ||||
|   "end": 16, | ||||
|   "start": 0 | ||||
| } | ||||
|  | ||||
