Compare commits
	
		
			30 Commits
		
	
	
		
			achalmers/
			...
			franknoiro
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| f70847c407 | |||
| 441d957228 | |||
| 9e57034873 | |||
| eb96d6539c | |||
| 513c76ecc8 | |||
| 51d9449280 | |||
| 6366bc4766 | |||
| 7a21918223 | |||
| 8072f1db63 | |||
| 18e1855fa9 | |||
| 7be53c7d4a | |||
| 2bf20988ef | |||
| 1495cc6d18 | |||
| f876e6ca3c | |||
| 60a0c811ab | |||
| cab0c1e6a1 | |||
| 417d720b22 | |||
| 77293952c0 | |||
| ea3d604b73 | |||
| 023a659491 | |||
| dd3a2b14f9 | |||
| 9510849a3a | |||
| 5b208356b4 | |||
| 84209e764e | |||
| 2b85e7abd6 | |||
| 9155a5efc8 | |||
| 2ccc27112a | |||
| ee160b67f4 | |||
| 5117b6f5d6 | |||
| bc8a7a364d | 
							
								
								
									
										22
									
								
								.github/workflows/build-apps.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -362,6 +362,17 @@ jobs: | ||||
|       - name: List artifacts | ||||
|         run: "ls -R out" | ||||
|  | ||||
|       - 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, | ||||
|           # see https://github.com/actions/checkout/issues/1471 | ||||
|           git fetch --prune --unshallow --tags | ||||
|           export TAG="nightly-${VERSION}" | ||||
|           export PREVIOUS_TAG=$(git describe --tags --match="nightly-v[0-9]*" --abbrev=0) | ||||
|           export NOTES=$(./scripts/get-nightly-changelog.sh) | ||||
|           yarn files:set-notes | ||||
|  | ||||
|       - name: Authenticate to Google Cloud | ||||
|         if: ${{ env.IS_NIGHTLY == 'true' }} | ||||
|         uses: 'google-github-actions/auth@v2.1.7' | ||||
| @ -382,3 +393,14 @@ jobs: | ||||
|           glob: '*' | ||||
|           parent: false | ||||
|           destination: 'dl.kittycad.io/releases/modeling-app/nightly' | ||||
|  | ||||
|       - name: Tag nightly commit | ||||
|         if: ${{ env.IS_NIGHTLY == 'true' }} | ||||
|         uses: actions/github-script@v7 | ||||
|         with: | ||||
|           script: | | ||||
|             const { VERSION } = process.env           | ||||
|             const { owner, repo } = context.repo | ||||
|             const { sha } = context | ||||
|             const ref = `refs/tags/nightly-${VERSION}` | ||||
|             github.rest.git.createRef({ owner, repo, sha, ref }) | ||||
|  | ||||
| @ -99,7 +99,7 @@ yarn tron:start | ||||
|  | ||||
| This will start the application and hot-reload on changes. | ||||
|  | ||||
| Devtools can be opened with the usual Cmd/Ctrl-Shift-I. | ||||
| Devtools can be opened with the usual Cmd-Opt-I (Mac) or Ctrl-Shift-I (Linux and Windows). | ||||
|  | ||||
| To build, run `yarn tron:package`. | ||||
|  | ||||
|  | ||||
| @ -518,7 +518,10 @@ test.describe('Editor tests', () => { | ||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
|   }) | ||||
|  | ||||
|   test('error with 2 source ranges gets 2 diagnostics', async ({ page }) => { | ||||
|   // TODO currently multiple source ranges are not supported | ||||
|   test.skip('error with 2 source ranges gets 2 diagnostics', async ({ | ||||
|     page, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|  | ||||
| @ -6,6 +6,7 @@ export class ToolbarFixture { | ||||
|   public page: Page | ||||
|  | ||||
|   extrudeButton!: Locator | ||||
|   loftButton!: Locator | ||||
|   offsetPlaneButton!: Locator | ||||
|   startSketchBtn!: Locator | ||||
|   lineBtn!: Locator | ||||
| @ -26,6 +27,7 @@ export class ToolbarFixture { | ||||
|   reConstruct = (page: Page) => { | ||||
|     this.page = page | ||||
|     this.extrudeButton = page.getByTestId('extrude') | ||||
|     this.loftButton = page.getByTestId('loft') | ||||
|     this.offsetPlaneButton = page.getByTestId('plane-offset') | ||||
|     this.startSketchBtn = page.getByTestId('sketch') | ||||
|     this.lineBtn = page.getByTestId('line') | ||||
|  | ||||
| @ -677,3 +677,94 @@ test(`Offset plane point-and-click`, async ({ | ||||
|     await scene.expectPixelColor([74, 74, 74], testPoint, 15) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| const loftPointAndClickCases = [ | ||||
|   { shouldPreselect: true }, | ||||
|   { shouldPreselect: false }, | ||||
| ] | ||||
| loftPointAndClickCases.forEach(({ shouldPreselect }) => { | ||||
|   test(`Loft point-and-click (preselected sketches: ${shouldPreselect})`, async ({ | ||||
|     app, | ||||
|     page, | ||||
|     scene, | ||||
|     editor, | ||||
|     toolbar, | ||||
|     cmdBar, | ||||
|   }) => { | ||||
|     const initialCode = `sketch001 = startSketchOn('XZ') | ||||
|     |> circle({ center = [0, 0], radius = 30 }, %) | ||||
|     plane001 = offsetPlane('XZ', 50) | ||||
|     sketch002 = startSketchOn(plane001) | ||||
|     |> circle({ center = [0, 0], radius = 20 }, %) | ||||
| ` | ||||
|     await app.initialise(initialCode) | ||||
|  | ||||
|     // One dumb hardcoded screen pixel value | ||||
|     const testPoint = { x: 575, y: 200 } | ||||
|     const [clickOnSketch1] = scene.makeMouseHelpers(testPoint.x, testPoint.y) | ||||
|     const [clickOnSketch2] = scene.makeMouseHelpers( | ||||
|       testPoint.x, | ||||
|       testPoint.y + 80 | ||||
|     ) | ||||
|     const loftDeclaration = 'loft001 = loft([sketch001, sketch002])' | ||||
|  | ||||
|     await test.step(`Look for the white of the sketch001 shape`, async () => { | ||||
|       await scene.expectPixelColor([254, 254, 254], testPoint, 15) | ||||
|     }) | ||||
|  | ||||
|     async function selectSketches() { | ||||
|       await clickOnSketch1() | ||||
|       await page.keyboard.down('Shift') | ||||
|       await clickOnSketch2() | ||||
|       await app.page.waitForTimeout(500) | ||||
|       await page.keyboard.up('Shift') | ||||
|     } | ||||
|  | ||||
|     if (!shouldPreselect) { | ||||
|       await test.step(`Go through the command bar flow without preselected sketches`, async () => { | ||||
|         await toolbar.loftButton.click() | ||||
|         await cmdBar.expectState({ | ||||
|           stage: 'arguments', | ||||
|           currentArgKey: 'selection', | ||||
|           currentArgValue: '', | ||||
|           headerArguments: { Selection: '' }, | ||||
|           highlightedHeaderArg: 'selection', | ||||
|           commandName: 'Loft', | ||||
|         }) | ||||
|         await selectSketches() | ||||
|         await cmdBar.progressCmdBar() | ||||
|         await cmdBar.expectState({ | ||||
|           stage: 'review', | ||||
|           headerArguments: { Selection: '2 faces' }, | ||||
|           commandName: 'Loft', | ||||
|         }) | ||||
|         await cmdBar.progressCmdBar() | ||||
|       }) | ||||
|     } else { | ||||
|       await test.step(`Preselect the two sketches`, async () => { | ||||
|         await selectSketches() | ||||
|       }) | ||||
|  | ||||
|       await test.step(`Go through the command bar flow with preselected sketches`, async () => { | ||||
|         await toolbar.loftButton.click() | ||||
|         await cmdBar.progressCmdBar() | ||||
|         await cmdBar.expectState({ | ||||
|           stage: 'review', | ||||
|           headerArguments: { Selection: '2 faces' }, | ||||
|           commandName: 'Loft', | ||||
|         }) | ||||
|         await cmdBar.progressCmdBar() | ||||
|       }) | ||||
|     } | ||||
|  | ||||
|     await test.step(`Confirm code is added to the editor, scene has changed`, async () => { | ||||
|       await editor.expectEditor.toContain(loftDeclaration) | ||||
|       await editor.expectState({ | ||||
|         diagnostics: [], | ||||
|         activeLines: [loftDeclaration], | ||||
|         highlightedCode: '', | ||||
|       }) | ||||
|       await scene.expectPixelColor([89, 89, 89], testPoint, 15) | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| @ -550,7 +550,7 @@ sketch001 = startSketchAt([-0, -0]) | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     // Constants and locators | ||||
|     const planeColor: [number, number, number] = [170, 220, 170] | ||||
|     const planeColor: [number, number, number] = [161, 220, 155] | ||||
|     const bgColor: [number, number, number] = [27, 27, 27] | ||||
|     const middlePixelIsColor = async (color: [number, number, number]) => { | ||||
|       return u.getGreatestPixDiff({ x: 600, y: 250 }, color) | ||||
|  | ||||
| @ -7,6 +7,8 @@ try { | ||||
|     .split('\n') | ||||
|     .filter((line) => line && line.length > 1) | ||||
|     .forEach((line) => { | ||||
|       // Allow line comments. | ||||
|       if (line.trimStart().startsWith('#')) return | ||||
|       const [key, value] = line.split('=') | ||||
|       // prefer env vars over secrets file | ||||
|       secrets[key] = process.env[key] || (value as any).replaceAll('"', '') | ||||
|  | ||||
| @ -943,6 +943,110 @@ sketch002 = startSketchOn(extrude001, 'END') | ||||
| `.replace(/\s/g, '') | ||||
|     ) | ||||
|   }) | ||||
|  | ||||
|   /* TODO: once we fix bug turn on. | ||||
|    test('empty-scene default-planes act as expected when spaces in file', async ({ | ||||
|     page, | ||||
|     browserName, | ||||
|   }) => { | ||||
|  | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|  | ||||
|     await u.openDebugPanel() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
|     await u.closeDebugPanel() | ||||
|  | ||||
|     const XYPlanePoint = { x: 774, y: 116 } as const | ||||
|     const unHoveredColor: [number, number, number] = [47, 47, 93] | ||||
|     expect( | ||||
|       await u.getGreatestPixDiff(XYPlanePoint, unHoveredColor) | ||||
|     ).toBeLessThan(8) | ||||
|  | ||||
|     await page.mouse.move(XYPlanePoint.x, XYPlanePoint.y) | ||||
|     await page.waitForTimeout(200) | ||||
|  | ||||
|     // color should not change for having been hovered | ||||
|     expect( | ||||
|       await u.getGreatestPixDiff(XYPlanePoint, unHoveredColor) | ||||
|     ).toBeLessThan(8) | ||||
|  | ||||
|     await u.openAndClearDebugPanel() | ||||
|  | ||||
|     // Fill with spaces | ||||
|     await u.codeLocator.fill(`                | ||||
| `) | ||||
|  | ||||
|     await u.openDebugPanel() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
|     await u.closeDebugPanel() | ||||
|  | ||||
|     expect( | ||||
|       await u.getGreatestPixDiff(XYPlanePoint, unHoveredColor) | ||||
|     ).toBeLessThan(8) | ||||
|  | ||||
|     await page.mouse.move(XYPlanePoint.x, XYPlanePoint.y) | ||||
|     await page.waitForTimeout(200) | ||||
|  | ||||
|     // color should not change for having been hovered | ||||
|     expect( | ||||
|       await u.getGreatestPixDiff(XYPlanePoint, unHoveredColor) | ||||
|     ).toBeLessThan(8) | ||||
|   }) | ||||
|  | ||||
|   test('empty-scene default-planes act as expected when only code comments in file', async ({ | ||||
|     page, | ||||
|     browserName, | ||||
|   }) => { | ||||
|  | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|  | ||||
|     await u.openDebugPanel() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
|     await u.closeDebugPanel() | ||||
|  | ||||
|     const XYPlanePoint = { x: 774, y: 116 } as const | ||||
|     const unHoveredColor: [number, number, number] = [47, 47, 93] | ||||
|     expect( | ||||
|       await u.getGreatestPixDiff(XYPlanePoint, unHoveredColor) | ||||
|     ).toBeLessThan(8) | ||||
|  | ||||
|     await page.mouse.move(XYPlanePoint.x, XYPlanePoint.y) | ||||
|     await page.waitForTimeout(200) | ||||
|  | ||||
|     // color should not change for having been hovered | ||||
|     expect( | ||||
|       await u.getGreatestPixDiff(XYPlanePoint, unHoveredColor) | ||||
|     ).toBeLessThan(8) | ||||
|  | ||||
|     await u.openAndClearDebugPanel() | ||||
|  | ||||
|     // Fill with spaces | ||||
|     await u.codeLocator.fill(`// this is a code comments ya nerds | ||||
| `) | ||||
|  | ||||
|     await u.openDebugPanel() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
|     await u.closeDebugPanel() | ||||
|  | ||||
|     expect( | ||||
|       await u.getGreatestPixDiff(XYPlanePoint, unHoveredColor) | ||||
|     ).toBeLessThan(8) | ||||
|  | ||||
|     await page.mouse.move(XYPlanePoint.x, XYPlanePoint.y) | ||||
|     await page.waitForTimeout(200) | ||||
|  | ||||
|     // color should not change for having been hovered | ||||
|     expect( | ||||
|       await u.getGreatestPixDiff(XYPlanePoint, unHoveredColor) | ||||
|     ).toBeLessThan(8) | ||||
|   })*/ | ||||
|  | ||||
|   test('empty-scene default-planes act as expected', async ({ | ||||
|     page, | ||||
|     browserName, | ||||
|  | ||||
| Before Width: | Height: | Size: 33 KiB After Width: | Height: | Size: 33 KiB | 
| Before Width: | Height: | Size: 34 KiB After Width: | Height: | Size: 34 KiB | 
| Before Width: | Height: | Size: 53 KiB After Width: | Height: | Size: 50 KiB | 
| Before Width: | Height: | Size: 54 KiB After Width: | Height: | Size: 51 KiB | 
| Before Width: | Height: | Size: 40 KiB After Width: | Height: | Size: 40 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: 39 KiB After Width: | Height: | Size: 39 KiB | 
| @ -23,7 +23,7 @@ test.describe('Test toggling perspective', () => { | ||||
|       y: screenHeight * 0.4, | ||||
|     } | ||||
|     const backgroundColor: [number, number, number] = [29, 29, 29] | ||||
|     const xzPlaneColor: [number, number, number] = [50, 50, 99] | ||||
|     const xzPlaneColor: [number, number, number] = [82, 55, 96] | ||||
|     const locationToHaveColor = async (color: [number, number, number]) => { | ||||
|       return u.getGreatestPixDiff(checkedScreenLocation, color) | ||||
|     } | ||||
|  | ||||
							
								
								
									
										5
									
								
								scripts/get-nightly-changelog.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @ -0,0 +1,5 @@ | ||||
| #!/bin/bash | ||||
| echo "## What's Changed" | ||||
| git log ${PREVIOUS_TAG}..HEAD --oneline --pretty=format:%s | grep -v Bump | awk '{print "* "toupper(substr($0,0,1))substr($0,2)}' | ||||
| echo "" | ||||
| echo "**Full Changelog**: https://github.com/KittyCAD/modeling-app/compare/${PREVIOUS_TAG}...${TAG}" | ||||
							
								
								
									
										65
									
								
								src/App.tsx
									
									
									
									
									
								
							
							
						
						| @ -3,7 +3,7 @@ import { useHotKeyListener } from './hooks/useHotKeyListener' | ||||
| import { Stream } from './components/Stream' | ||||
| import { AppHeader } from './components/AppHeader' | ||||
| import { useHotkeys } from 'react-hotkeys-hook' | ||||
| import { useLoaderData, useNavigate } from 'react-router-dom' | ||||
| import { useLoaderData, useLocation, useNavigate } from 'react-router-dom' | ||||
| import { type IndexLoaderData } from 'lib/types' | ||||
| import { PATHS } from 'lib/paths' | ||||
| import { useSettingsAuthContext } from 'hooks/useSettingsAuthContext' | ||||
| @ -22,7 +22,14 @@ import Gizmo from 'components/Gizmo' | ||||
| import { CoreDumpManager } from 'lib/coredump' | ||||
| import { UnitsMenu } from 'components/UnitsMenu' | ||||
| import { CameraProjectionToggle } from 'components/CameraProjectionToggle' | ||||
| import { homeDefaultStatusBarItems } from 'components/statusBar/homeDefaultStatusBarItems' | ||||
| import { StatusBar } from 'components/StatusBar' | ||||
| import { useModelStateStatus } from 'components/ModelStateIndicator' | ||||
| import { useNetworkHealthStatus } from 'components/NetworkHealthIndicator' | ||||
| import { useModelingContext } from 'hooks/useModelingContext' | ||||
| import { xStateValueToString } from 'lib/xStateValueToString' | ||||
| import { maybeWriteToDisk } from 'lib/telemetry' | ||||
| import { useNetworkMachineStatus } from 'components/NetworkMachineIndicator' | ||||
| maybeWriteToDisk() | ||||
|   .then(() => {}) | ||||
|   .catch(() => {}) | ||||
| @ -31,11 +38,10 @@ export function App() { | ||||
|   const { project, file } = useLoaderData() as IndexLoaderData | ||||
|   useRefreshSettings(PATHS.FILE + 'SETTINGS') | ||||
|   const navigate = useNavigate() | ||||
|   const location = useLocation() | ||||
|   const filePath = useAbsoluteFilePath() | ||||
|   const { onProjectOpen } = useLspContext() | ||||
|   // We need the ref for the outermost div so we can screenshot the app for | ||||
|   // the coredump. | ||||
|   const ref = useRef<HTMLDivElement>(null) | ||||
|   const { state: modelingState, streamRef } = useModelingContext() | ||||
|  | ||||
|   const projectName = project?.name || null | ||||
|   const projectPath = project?.path || null | ||||
| @ -77,21 +83,44 @@ export function App() { | ||||
|   useEngineConnectionSubscriptions() | ||||
|  | ||||
|   return ( | ||||
|     <div className="relative h-full flex flex-col" ref={ref}> | ||||
|       <AppHeader | ||||
|         className={'transition-opacity transition-duration-75 ' + paneOpacity} | ||||
|         project={{ project, file }} | ||||
|         enableMenu={true} | ||||
|     <div className="h-screen flex flex-col overflow-hidden select-none"> | ||||
|       <div className="relative flex flex-1 flex-col" ref={streamRef}> | ||||
|         <AppHeader | ||||
|           className={'transition-opacity transition-duration-75 ' + paneOpacity} | ||||
|           project={{ project, file }} | ||||
|           enableMenu={true} | ||||
|         /> | ||||
|         <ModalContainer /> | ||||
|         <ModelingSidebar paneOpacity={paneOpacity} /> | ||||
|         <Stream /> | ||||
|         {/* <CamToggle /> */} | ||||
|         <LowerRightControls coreDumpManager={coreDumpManager}> | ||||
|           <UnitsMenu /> | ||||
|           <Gizmo /> | ||||
|           <CameraProjectionToggle /> | ||||
|         </LowerRightControls> | ||||
|       </div> | ||||
|       <StatusBar | ||||
|         globalItems={[ | ||||
|           useNetworkHealthStatus(), | ||||
|           useNetworkMachineStatus(), | ||||
|           ...homeDefaultStatusBarItems({ coreDumpManager, location }), | ||||
|         ]} | ||||
|         localItems={[ | ||||
|           { | ||||
|             id: 'modeling-state', | ||||
|             element: 'text', | ||||
|             label: | ||||
|               modelingState.value instanceof Object | ||||
|                 ? xStateValueToString(modelingState.value) ?? '' | ||||
|                 : modelingState.value, | ||||
|             toolTip: { | ||||
|               children: 'The current state of the modeler', | ||||
|             }, | ||||
|           }, | ||||
|           useModelStateStatus(), | ||||
|         ]} | ||||
|       /> | ||||
|       <ModalContainer /> | ||||
|       <ModelingSidebar paneOpacity={paneOpacity} /> | ||||
|       <Stream /> | ||||
|       {/* <CamToggle /> */} | ||||
|       <LowerRightControls coreDumpManager={coreDumpManager}> | ||||
|         <UnitsMenu /> | ||||
|         <Gizmo /> | ||||
|         <CameraProjectionToggle /> | ||||
|       </LowerRightControls> | ||||
|     </div> | ||||
|   ) | ||||
| } | ||||
|  | ||||
| @ -155,7 +155,6 @@ export class CameraControls { | ||||
|       this.camera.zoom = camProps.zoom || 1 | ||||
|     } | ||||
|     this.camera.updateProjectionMatrix() | ||||
|     console.log('doing this thing', camProps) | ||||
|     this.update(true) | ||||
|   } | ||||
|  | ||||
| @ -273,14 +272,26 @@ export class CameraControls { | ||||
|         camSettings.center.y, | ||||
|         camSettings.center.z | ||||
|       ) | ||||
|       const quat = new Quaternion( | ||||
|       const orientation = new Quaternion( | ||||
|         camSettings.orientation.x, | ||||
|         camSettings.orientation.y, | ||||
|         camSettings.orientation.z, | ||||
|         camSettings.orientation.w | ||||
|       ).invert() | ||||
|  | ||||
|       this.camera.up.copy(new Vector3(0, 1, 0).applyQuaternion(quat)) | ||||
|       const newUp = new Vector3( | ||||
|         camSettings.up.x, | ||||
|         camSettings.up.y, | ||||
|         camSettings.up.z | ||||
|       ) | ||||
|       this.camera.quaternion.set( | ||||
|         orientation.x, | ||||
|         orientation.y, | ||||
|         orientation.z, | ||||
|         orientation.w | ||||
|       ) | ||||
|       this.camera.up.copy(newUp) | ||||
|       this.camera.updateProjectionMatrix() | ||||
|       if (this.camera instanceof PerspectiveCamera && camSettings.ortho) { | ||||
|         this.useOrthographicCamera() | ||||
|       } | ||||
| @ -1164,7 +1175,7 @@ export class CameraControls { | ||||
|       this.camera.updateProjectionMatrix() | ||||
|     } | ||||
|  | ||||
|     if (this.syncDirection === 'clientToEngine' || forceUpdate) | ||||
|     if (this.syncDirection === 'clientToEngine' || forceUpdate) { | ||||
|       this.throttledUpdateEngineCamera({ | ||||
|         quaternion: this.camera.quaternion, | ||||
|         position: this.camera.position, | ||||
| @ -1172,6 +1183,7 @@ export class CameraControls { | ||||
|         isPerspective: this.isPerspective, | ||||
|         target: this.target, | ||||
|       }) | ||||
|     } | ||||
|     this.deferReactUpdate(this.reactCameraProperties) | ||||
|     Object.values(this._camChangeCallbacks).forEach((cb) => cb()) | ||||
|   } | ||||
|  | ||||
| @ -29,6 +29,9 @@ import { | ||||
|   Expr, | ||||
|   parse, | ||||
|   recast, | ||||
|   defaultSourceRange, | ||||
|   resultIsOk, | ||||
|   ProgramMemory, | ||||
| } from 'lang/wasm' | ||||
| import { CustomIcon, CustomIconName } from 'components/CustomIcon' | ||||
| import { ConstrainInfo } from 'lang/std/stdTypes' | ||||
| @ -412,14 +415,15 @@ export async function deleteSegment({ | ||||
|   if (err(modifiedAst)) return Promise.reject(modifiedAst) | ||||
|  | ||||
|   const newCode = recast(modifiedAst) | ||||
|   modifiedAst = parse(newCode) | ||||
|   if (err(modifiedAst)) return Promise.reject(modifiedAst) | ||||
|   const pResult = parse(newCode) | ||||
|   if (err(pResult) || !resultIsOk(pResult)) return Promise.reject(pResult) | ||||
|   modifiedAst = pResult.program | ||||
|  | ||||
|   const testExecute = await executeAst({ | ||||
|     ast: modifiedAst, | ||||
|     idGenerator: kclManager.execState.idGenerator, | ||||
|     useFakeExecutor: true, | ||||
|     engineCommandManager: engineCommandManager, | ||||
|     // We make sure to send an empty program memory to denote we mean mock mode. | ||||
|     programMemoryOverride: ProgramMemory.empty(), | ||||
|   }) | ||||
|   if (testExecute.errors.length) { | ||||
|     toast.error('Segment tag used outside of current Sketch. Could not delete.') | ||||
| @ -590,7 +594,9 @@ const ConstraintSymbol = ({ | ||||
|   if (err(_node)) return | ||||
|   const node = _node.node | ||||
|  | ||||
|   const range: SourceRange = node ? [node.start, node.end] : [0, 0] | ||||
|   const range: SourceRange = node | ||||
|     ? [node.start, node.end, true] | ||||
|     : defaultSourceRange() | ||||
|  | ||||
|   if (_type === 'intersectionTag') return null | ||||
|  | ||||
| @ -612,7 +618,7 @@ const ConstraintSymbol = ({ | ||||
|           editorManager.setHighlightRange([range]) | ||||
|         }} | ||||
|         onMouseLeave={() => { | ||||
|           editorManager.setHighlightRange([[0, 0]]) | ||||
|           editorManager.setHighlightRange([defaultSourceRange()]) | ||||
|         }} | ||||
|         // disabled={isConstrained || !convertToVarEnabled} | ||||
|         // disabled={implicitDesc} TODO why does this change styles that are hard to override? | ||||
| @ -627,10 +633,12 @@ const ConstraintSymbol = ({ | ||||
|             }) | ||||
|           } else if (isConstrained) { | ||||
|             try { | ||||
|               const parsed = parse(recast(kclManager.ast)) | ||||
|               if (trap(parsed)) return Promise.reject(parsed) | ||||
|               const pResult = parse(recast(kclManager.ast)) | ||||
|               if (trap(pResult) || !resultIsOk(pResult)) | ||||
|                 return Promise.reject(pResult) | ||||
|  | ||||
|               const _node1 = getNodeFromPath<CallExpression>( | ||||
|                 parsed, | ||||
|                 pResult.program!, | ||||
|                 pathToNode, | ||||
|                 'CallExpression', | ||||
|                 true | ||||
|  | ||||
| @ -48,6 +48,9 @@ import { | ||||
|   VariableDeclarator, | ||||
|   sketchFromKclValue, | ||||
|   sketchFromKclValueOptional, | ||||
|   defaultSourceRange, | ||||
|   sourceRangeFromRust, | ||||
|   resultIsOk, | ||||
| } from 'lang/wasm' | ||||
| import { | ||||
|   engineCommandManager, | ||||
| @ -495,10 +498,9 @@ export class SceneEntities { | ||||
|  | ||||
|     const { execState } = await executeAst({ | ||||
|       ast: truncatedAst, | ||||
|       useFakeExecutor: true, | ||||
|       engineCommandManager: this.engineCommandManager, | ||||
|       // We make sure to send an empty program memory to denote we mean mock mode. | ||||
|       programMemoryOverride, | ||||
|       idGenerator: kclManager.execState.idGenerator, | ||||
|     }) | ||||
|     const programMemory = execState.memory | ||||
|     const sketch = sketchFromPathToNode({ | ||||
| @ -530,7 +532,7 @@ export class SceneEntities { | ||||
|  | ||||
|     const segPathToNode = getNodePathFromSourceRange( | ||||
|       maybeModdedAst, | ||||
|       sketch.start.__geoMeta.sourceRange | ||||
|       sourceRangeFromRust(sketch.start.__geoMeta.sourceRange) | ||||
|     ) | ||||
|     if (sketch?.paths?.[0]?.type !== 'Circle') { | ||||
|       const _profileStart = createProfileStartHandle({ | ||||
| @ -552,7 +554,7 @@ export class SceneEntities { | ||||
|     sketch.paths.forEach((segment, index) => { | ||||
|       let segPathToNode = getNodePathFromSourceRange( | ||||
|         maybeModdedAst, | ||||
|         segment.__geoMeta.sourceRange | ||||
|         sourceRangeFromRust(segment.__geoMeta.sourceRange) | ||||
|       ) | ||||
|       if ( | ||||
|         draftExpressionsIndices && | ||||
| @ -561,12 +563,12 @@ export class SceneEntities { | ||||
|         const previousSegment = sketch.paths[index - 1] || sketch.start | ||||
|         const previousSegmentPathToNode = getNodePathFromSourceRange( | ||||
|           maybeModdedAst, | ||||
|           previousSegment.__geoMeta.sourceRange | ||||
|           sourceRangeFromRust(previousSegment.__geoMeta.sourceRange) | ||||
|         ) | ||||
|         const bodyIndex = previousSegmentPathToNode[1][0] | ||||
|         segPathToNode = getNodePathFromSourceRange( | ||||
|           truncatedAst, | ||||
|           segment.__geoMeta.sourceRange | ||||
|           sourceRangeFromRust(segment.__geoMeta.sourceRange) | ||||
|         ) | ||||
|         segPathToNode[1][0] = bodyIndex | ||||
|       } | ||||
| @ -575,7 +577,10 @@ export class SceneEntities { | ||||
|         index <= draftExpressionsIndices.end && | ||||
|         index >= draftExpressionsIndices.start | ||||
|       const isSelected = selectionRanges?.graphSelections.some((selection) => | ||||
|         isOverlap(selection?.codeRef?.range, segment.__geoMeta.sourceRange) | ||||
|         isOverlap( | ||||
|           selection?.codeRef?.range, | ||||
|           sourceRangeFromRust(segment.__geoMeta.sourceRange) | ||||
|         ) | ||||
|       ) | ||||
|  | ||||
|       let seg: Group | ||||
| @ -657,13 +662,11 @@ export class SceneEntities { | ||||
|   } | ||||
|   updateAstAndRejigSketch = async ( | ||||
|     sketchPathToNode: PathToNode, | ||||
|     modifiedAst: Node<Program> | Error, | ||||
|     modifiedAst: Node<Program>, | ||||
|     forward: [number, number, number], | ||||
|     up: [number, number, number], | ||||
|     origin: [number, number, number] | ||||
|   ) => { | ||||
|     if (err(modifiedAst)) return modifiedAst | ||||
|  | ||||
|     const nextAst = await kclManager.updateAst(modifiedAst, false) | ||||
|     await this.tearDownSketch({ removeAxis: false }) | ||||
|     sceneInfra.resetMouseListeners() | ||||
| @ -721,8 +724,9 @@ export class SceneEntities { | ||||
|       pathToNode: sketchPathToNode, | ||||
|     }) | ||||
|     if (trap(mod)) return Promise.reject(mod) | ||||
|     const modifiedAst = parse(recast(mod.modifiedAst)) | ||||
|     if (trap(modifiedAst)) return Promise.reject(modifiedAst) | ||||
|     const pResult = parse(recast(mod.modifiedAst)) | ||||
|     if (trap(pResult) || !resultIsOk(pResult)) return Promise.reject(pResult) | ||||
|     const modifiedAst = pResult.program | ||||
|  | ||||
|     const draftExpressionsIndices = { start: index, end: index } | ||||
|  | ||||
| @ -914,9 +918,9 @@ export class SceneEntities { | ||||
|       ...getRectangleCallExpressions(rectangleOrigin, tags), | ||||
|     ]) | ||||
|  | ||||
|     let _recastAst = parse(recast(_ast)) | ||||
|     if (trap(_recastAst)) return Promise.reject(_recastAst) | ||||
|     _ast = _recastAst | ||||
|     const pResult = parse(recast(_ast)) | ||||
|     if (trap(pResult) || !resultIsOk(pResult)) return Promise.reject(pResult) | ||||
|     _ast = pResult.program | ||||
|  | ||||
|     const { programMemoryOverride, truncatedAst } = await this.setupSketch({ | ||||
|       sketchPathToNode, | ||||
| @ -950,10 +954,9 @@ export class SceneEntities { | ||||
|  | ||||
|         const { execState } = await executeAst({ | ||||
|           ast: truncatedAst, | ||||
|           useFakeExecutor: true, | ||||
|           engineCommandManager: this.engineCommandManager, | ||||
|           // We make sure to send an empty program memory to denote we mean mock mode. | ||||
|           programMemoryOverride, | ||||
|           idGenerator: kclManager.execState.idGenerator, | ||||
|         }) | ||||
|         const programMemory = execState.memory | ||||
|         this.sceneProgramMemory = programMemory | ||||
| @ -998,9 +1001,10 @@ export class SceneEntities { | ||||
|         updateRectangleSketch(sketchInit, x, y, tags[0]) | ||||
|  | ||||
|         const newCode = recast(_ast) | ||||
|         let _recastAst = parse(newCode) | ||||
|         if (trap(_recastAst)) return | ||||
|         _ast = _recastAst | ||||
|         const pResult = parse(newCode) | ||||
|         if (trap(pResult) || !resultIsOk(pResult)) | ||||
|           return Promise.reject(pResult) | ||||
|         _ast = pResult.program | ||||
|  | ||||
|         // Update the primary AST and unequip the rectangle tool | ||||
|         await kclManager.executeAstMock(_ast) | ||||
| @ -1013,10 +1017,9 @@ export class SceneEntities { | ||||
|  | ||||
|         const { execState } = await executeAst({ | ||||
|           ast: _ast, | ||||
|           useFakeExecutor: true, | ||||
|           engineCommandManager: this.engineCommandManager, | ||||
|           // We make sure to send an empty program memory to denote we mean mock mode. | ||||
|           programMemoryOverride, | ||||
|           idGenerator: kclManager.execState.idGenerator, | ||||
|         }) | ||||
|         const programMemory = execState.memory | ||||
|  | ||||
| @ -1071,9 +1074,9 @@ export class SceneEntities { | ||||
|       ...getRectangleCallExpressions(rectangleOrigin, tags), | ||||
|     ]) | ||||
|  | ||||
|     let _recastAst = parse(recast(_ast)) | ||||
|     if (trap(_recastAst)) return Promise.reject(_recastAst) | ||||
|     _ast = _recastAst | ||||
|     const pResult = parse(recast(_ast)) | ||||
|     if (trap(pResult) || !resultIsOk(pResult)) return Promise.reject(pResult) | ||||
|     _ast = pResult.program | ||||
|  | ||||
|     const { programMemoryOverride, truncatedAst } = await this.setupSketch({ | ||||
|       sketchPathToNode, | ||||
| @ -1114,10 +1117,9 @@ export class SceneEntities { | ||||
|  | ||||
|         const { execState } = await executeAst({ | ||||
|           ast: truncatedAst, | ||||
|           useFakeExecutor: true, | ||||
|           engineCommandManager: this.engineCommandManager, | ||||
|           // We make sure to send an empty program memory to denote we mean mock mode. | ||||
|           programMemoryOverride, | ||||
|           idGenerator: kclManager.execState.idGenerator, | ||||
|         }) | ||||
|         const programMemory = execState.memory | ||||
|         this.sceneProgramMemory = programMemory | ||||
| @ -1165,9 +1167,10 @@ export class SceneEntities { | ||||
|             rectangleOrigin[1] | ||||
|           ) | ||||
|  | ||||
|           let _recastAst = parse(recast(_ast)) | ||||
|           if (trap(_recastAst)) return | ||||
|           _ast = _recastAst | ||||
|           const pResult = parse(recast(_ast)) | ||||
|           if (trap(pResult) || !resultIsOk(pResult)) | ||||
|             return Promise.reject(pResult) | ||||
|           _ast = pResult.program | ||||
|  | ||||
|           // Update the primary AST and unequip the rectangle tool | ||||
|           await kclManager.executeAstMock(_ast) | ||||
| @ -1180,10 +1183,9 @@ export class SceneEntities { | ||||
|  | ||||
|           const { execState } = await executeAst({ | ||||
|             ast: _ast, | ||||
|             useFakeExecutor: true, | ||||
|             engineCommandManager: this.engineCommandManager, | ||||
|             // We make sure to send an empty program memory to denote we mean mock mode. | ||||
|             programMemoryOverride, | ||||
|             idGenerator: kclManager.execState.idGenerator, | ||||
|           }) | ||||
|           const programMemory = execState.memory | ||||
|  | ||||
| @ -1241,9 +1243,9 @@ export class SceneEntities { | ||||
|       ]), | ||||
|     ]) | ||||
|  | ||||
|     let _recastAst = parse(recast(_ast)) | ||||
|     if (trap(_recastAst)) return Promise.reject(_recastAst) | ||||
|     _ast = _recastAst | ||||
|     const pResult = parse(recast(_ast)) | ||||
|     if (trap(pResult) || !resultIsOk(pResult)) return Promise.reject(pResult) | ||||
|     _ast = pResult.program | ||||
|  | ||||
|     // do a quick mock execution to get the program memory up-to-date | ||||
|     await kclManager.executeAstMock(_ast) | ||||
| @ -1299,10 +1301,9 @@ export class SceneEntities { | ||||
|  | ||||
|         const { execState } = await executeAst({ | ||||
|           ast: modded, | ||||
|           useFakeExecutor: true, | ||||
|           engineCommandManager: this.engineCommandManager, | ||||
|           // We make sure to send an empty program memory to denote we mean mock mode. | ||||
|           programMemoryOverride, | ||||
|           idGenerator: kclManager.execState.idGenerator, | ||||
|         }) | ||||
|         const programMemory = execState.memory | ||||
|         this.sceneProgramMemory = programMemory | ||||
| @ -1365,9 +1366,10 @@ export class SceneEntities { | ||||
|  | ||||
|           const newCode = recast(modded) | ||||
|           if (err(newCode)) return | ||||
|           let _recastAst = parse(newCode) | ||||
|           if (trap(_recastAst)) return Promise.reject(_recastAst) | ||||
|           _ast = _recastAst | ||||
|           const pResult = parse(newCode) | ||||
|           if (trap(pResult) || !resultIsOk(pResult)) | ||||
|             return Promise.reject(pResult) | ||||
|           _ast = pResult.program | ||||
|  | ||||
|           // Update the primary AST and unequip the rectangle tool | ||||
|           await kclManager.executeAstMock(_ast) | ||||
| @ -1660,7 +1662,7 @@ export class SceneEntities { | ||||
|         kclManager.programMemory, | ||||
|         { | ||||
|           type: 'sourceRange', | ||||
|           sourceRange: [node.start, node.end], | ||||
|           sourceRange: [node.start, node.end, true], | ||||
|         }, | ||||
|         getChangeSketchInput() | ||||
|       ) | ||||
| @ -1683,10 +1685,9 @@ export class SceneEntities { | ||||
|         codeManager.updateCodeEditor(code) | ||||
|       const { execState } = await executeAst({ | ||||
|         ast: truncatedAst, | ||||
|         useFakeExecutor: true, | ||||
|         engineCommandManager: this.engineCommandManager, | ||||
|         // We make sure to send an empty program memory to denote we mean mock mode. | ||||
|         programMemoryOverride, | ||||
|         idGenerator: kclManager.execState.idGenerator, | ||||
|       }) | ||||
|       const programMemory = execState.memory | ||||
|       this.sceneProgramMemory = programMemory | ||||
| @ -1750,7 +1751,7 @@ export class SceneEntities { | ||||
|   ): (() => SegmentOverlayPayload | null) => { | ||||
|     const segPathToNode = getNodePathFromSourceRange( | ||||
|       modifiedAst, | ||||
|       segment.__geoMeta.sourceRange | ||||
|       sourceRangeFromRust(segment.__geoMeta.sourceRange) | ||||
|     ) | ||||
|     const sgPaths = sketch.paths | ||||
|     const originalPathToNodeStr = JSON.stringify(segPathToNode) | ||||
| @ -1901,8 +1902,10 @@ export class SceneEntities { | ||||
|           SEGMENT_BODIES_PLUS_PROFILE_START | ||||
|         ) | ||||
|         if (parent?.userData?.pathToNode) { | ||||
|           const updatedAst = parse(recast(kclManager.ast)) | ||||
|           if (trap(updatedAst)) return | ||||
|           const pResult = parse(recast(kclManager.ast)) | ||||
|           if (trap(pResult) || !resultIsOk(pResult)) | ||||
|             return Promise.reject(pResult) | ||||
|           const updatedAst = pResult.program | ||||
|           const _node = getNodeFromPath<Node<CallExpression>>( | ||||
|             updatedAst, | ||||
|             parent.userData.pathToNode, | ||||
| @ -1910,7 +1913,7 @@ export class SceneEntities { | ||||
|           ) | ||||
|           if (trap(_node, { suppress: true })) return | ||||
|           const node = _node.node | ||||
|           editorManager.setHighlightRange([[node.start, node.end]]) | ||||
|           editorManager.setHighlightRange([[node.start, node.end, true]]) | ||||
|           const yellow = 0xffff00 | ||||
|           colorSegment(selected, yellow) | ||||
|           const extraSegmentGroup = parent.getObjectByName(EXTRA_SEGMENT_HANDLE) | ||||
| @ -1955,10 +1958,10 @@ export class SceneEntities { | ||||
|             }) | ||||
|           return | ||||
|         } | ||||
|         editorManager.setHighlightRange([[0, 0]]) | ||||
|         editorManager.setHighlightRange([defaultSourceRange()]) | ||||
|       }, | ||||
|       onMouseLeave: ({ selected, ...rest }: OnMouseEnterLeaveArgs) => { | ||||
|         editorManager.setHighlightRange([[0, 0]]) | ||||
|         editorManager.setHighlightRange([defaultSourceRange()]) | ||||
|         const parent = getParentGroup( | ||||
|           selected, | ||||
|           SEGMENT_BODIES_PLUS_PROFILE_START | ||||
| @ -2087,8 +2090,10 @@ function prepareTruncatedMemoryAndAst( | ||||
|     ).body.push(newSegment) | ||||
|     // update source ranges to section we just added. | ||||
|     // hacks like this wouldn't be needed if the AST put pathToNode info in memory/sketch segments | ||||
|     const updatedSrcRangeAst = parse(recast(_ast)) // get source ranges correct since unfortunately we still rely on them | ||||
|     if (err(updatedSrcRangeAst)) return updatedSrcRangeAst | ||||
|     const pResult = parse(recast(_ast)) // get source ranges correct since unfortunately we still rely on them | ||||
|     if (trap(pResult) || !resultIsOk(pResult)) | ||||
|       return Error('Unexpected compilation error') | ||||
|     const updatedSrcRangeAst = pResult.program | ||||
|  | ||||
|     const lastPipeItem = ( | ||||
|       (updatedSrcRangeAst.body[bodyIndex] as VariableDeclaration) | ||||
|  | ||||
| @ -5,6 +5,7 @@ import { useEffect, useRef, useState } from 'react' | ||||
| import { trap } from 'lib/trap' | ||||
| import { codeToIdSelections } from 'lib/selections' | ||||
| import { codeRefFromRange } from 'lang/std/artifactGraph' | ||||
| import { defaultSourceRange } from 'lang/wasm' | ||||
|  | ||||
| export function AstExplorer() { | ||||
|   const { context } = useModelingContext() | ||||
| @ -46,7 +47,7 @@ export function AstExplorer() { | ||||
|       <div | ||||
|         className="h-full relative" | ||||
|         onMouseLeave={(e) => { | ||||
|           editorManager.setHighlightRange([[0, 0]]) | ||||
|           editorManager.setHighlightRange([defaultSourceRange()]) | ||||
|         }} | ||||
|       > | ||||
|         <pre className="text-xs"> | ||||
| @ -115,15 +116,19 @@ function DisplayObj({ | ||||
|         hasCursor ? 'bg-violet-100/80 dark:bg-violet-100/25' : '' | ||||
|       }`} | ||||
|       onMouseEnter={(e) => { | ||||
|         editorManager.setHighlightRange([[obj?.start || 0, obj.end]]) | ||||
|         editorManager.setHighlightRange([[obj?.start || 0, obj.end, true]]) | ||||
|         e.stopPropagation() | ||||
|       }} | ||||
|       onMouseMove={(e) => { | ||||
|         e.stopPropagation() | ||||
|         editorManager.setHighlightRange([[obj?.start || 0, obj.end]]) | ||||
|         editorManager.setHighlightRange([[obj?.start || 0, obj.end, true]]) | ||||
|       }} | ||||
|       onClick={(e) => { | ||||
|         const range: [number, number] = [obj?.start || 0, obj.end || 0] | ||||
|         const range: [number, number, boolean] = [ | ||||
|           obj?.start || 0, | ||||
|           obj.end || 0, | ||||
|           true, | ||||
|         ] | ||||
|         const idInfo = codeToIdSelections([ | ||||
|           { codeRef: codeRefFromRange(range, kclManager.ast) }, | ||||
|         ])[0] | ||||
|  | ||||
| @ -1,5 +1,11 @@ | ||||
| import { useEffect, useState, useRef } from 'react' | ||||
| import { parse, BinaryPart, Expr, ProgramMemory } from '../lang/wasm' | ||||
| import { | ||||
|   parse, | ||||
|   BinaryPart, | ||||
|   Expr, | ||||
|   ProgramMemory, | ||||
|   resultIsOk, | ||||
| } from '../lang/wasm' | ||||
| import { | ||||
|   createIdentifier, | ||||
|   createLiteral, | ||||
| @ -141,8 +147,9 @@ export function useCalc({ | ||||
|   useEffect(() => { | ||||
|     try { | ||||
|       const code = `const __result__ = ${value}` | ||||
|       const ast = parse(code) | ||||
|       if (trap(ast)) return | ||||
|       const pResult = parse(code) | ||||
|       if (trap(pResult) || !resultIsOk(pResult)) return | ||||
|       const ast = pResult.program | ||||
|       const _programMem: ProgramMemory = ProgramMemory.empty() | ||||
|       for (const { key, value } of availableVarInfo.variables) { | ||||
|         const error = _programMem.set(key, { | ||||
| @ -156,9 +163,8 @@ export function useCalc({ | ||||
|       executeAst({ | ||||
|         ast, | ||||
|         engineCommandManager, | ||||
|         useFakeExecutor: true, | ||||
|         // We make sure to send an empty program memory to denote we mean mock mode. | ||||
|         programMemoryOverride: kclManager.programMemory.clone(), | ||||
|         idGenerator: kclManager.execState.idGenerator, | ||||
|       }).then(({ execState }) => { | ||||
|         const resultDeclaration = ast.body.find( | ||||
|           (a) => | ||||
|  | ||||
| @ -636,6 +636,16 @@ const CustomIconMap = { | ||||
|       /> | ||||
|     </svg> | ||||
|   ), | ||||
|   loading: ( | ||||
|     <svg viewBox="0 0 20 20" fill="none" xmlns="http://www.w3.org/2000/svg"> | ||||
|       <path | ||||
|         fillRule="evenodd" | ||||
|         clipRule="evenodd" | ||||
|         d="M12.5001 6.25839C11.76 5.76392 10.89 5.5 10 5.5V4.5C11.0878 4.5 12.1512 4.82257 13.0556 5.42692C13.9601 6.03126 14.6651 6.89025 15.0813 7.89524C15.4976 8.90023 15.6065 10.0061 15.3943 11.073C15.1821 12.1399 14.6583 13.1199 13.8891 13.8891C13.1199 14.6583 12.1399 15.1821 11.073 15.3943C10.0061 15.6065 8.90023 15.4976 7.89524 15.0813C6.89025 14.6651 6.03126 13.9601 5.42692 13.0556C4.82257 12.1512 4.5 11.0878 4.5 10H5.5C5.5 10.89 5.76392 11.76 6.25839 12.5001C6.75285 13.2401 7.45566 13.8169 8.27792 14.1575C9.10019 14.4981 10.005 14.5872 10.8779 14.4135C11.7508 14.2399 12.5526 13.8113 13.182 13.182C13.8113 12.5526 14.2399 11.7508 14.4135 10.8779C14.5872 10.005 14.4981 9.10019 14.1575 8.27792C13.8169 7.45566 13.2401 6.75285 12.5001 6.25839Z" | ||||
|         fill="currentColor" | ||||
|       /> | ||||
|     </svg> | ||||
|   ), | ||||
|   lockClosed: ( | ||||
|     <svg | ||||
|       viewBox="0 0 20 20" | ||||
|  | ||||
| @ -1,141 +1,14 @@ | ||||
| import { APP_VERSION } from 'routes/Settings' | ||||
| import { CustomIcon } from 'components/CustomIcon' | ||||
| import Tooltip from 'components/Tooltip' | ||||
| import { PATHS } from 'lib/paths' | ||||
| import { NetworkHealthIndicator } from 'components/NetworkHealthIndicator' | ||||
| import { HelpMenu } from './HelpMenu' | ||||
| import { Link, useLocation } from 'react-router-dom' | ||||
| import { useAbsoluteFilePath } from 'hooks/useAbsoluteFilePath' | ||||
| import { coreDump } from 'lang/wasm' | ||||
| import toast from 'react-hot-toast' | ||||
| import { CoreDumpManager } from 'lib/coredump' | ||||
| import openWindow, { openExternalBrowserIfDesktop } from 'lib/openWindow' | ||||
| import { NetworkMachineIndicator } from './NetworkMachineIndicator' | ||||
| import { ModelStateIndicator } from './ModelStateIndicator' | ||||
| import { reportRejection } from 'lib/trap' | ||||
|  | ||||
| export function LowerRightControls({ | ||||
|   children, | ||||
|   coreDumpManager, | ||||
| }: { | ||||
|   children?: React.ReactNode | ||||
|   coreDumpManager?: CoreDumpManager | ||||
| }) { | ||||
|   const location = useLocation() | ||||
|   const filePath = useAbsoluteFilePath() | ||||
|  | ||||
|   const linkOverrideClassName = | ||||
|     '!text-chalkboard-70 hover:!text-chalkboard-80 dark:!text-chalkboard-40 dark:hover:!text-chalkboard-30' | ||||
|  | ||||
|   function reportbug(event: { | ||||
|     preventDefault: () => void | ||||
|     stopPropagation: () => void | ||||
|   }) { | ||||
|     event?.preventDefault() | ||||
|     event?.stopPropagation() | ||||
|  | ||||
|     if (!coreDumpManager) { | ||||
|       // open default reporting option | ||||
|       openWindow( | ||||
|         'https://github.com/KittyCAD/modeling-app/issues/new/choose' | ||||
|       ).catch(reportRejection) | ||||
|     } else { | ||||
|       toast | ||||
|         .promise( | ||||
|           coreDump(coreDumpManager, true), | ||||
|           { | ||||
|             loading: 'Preparing bug report...', | ||||
|             success: 'Bug report opened in new window', | ||||
|             error: 'Unable to export a core dump. Using default reporting.', | ||||
|           }, | ||||
|           { | ||||
|             success: { | ||||
|               // Note: this extended duration is especially important for Playwright e2e testing | ||||
|               // default duration is 2000 - https://react-hot-toast.com/docs/toast#default-durations | ||||
|               duration: 6000, | ||||
|             }, | ||||
|           } | ||||
|         ) | ||||
|         .catch((err: Error) => { | ||||
|           if (err) { | ||||
|             openWindow( | ||||
|               'https://github.com/KittyCAD/modeling-app/issues/new/choose' | ||||
|             ).catch(reportRejection) | ||||
|           } | ||||
|         }) | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   return ( | ||||
|     <section className="fixed bottom-2 right-2 flex flex-col items-end gap-3 pointer-events-none"> | ||||
|     <section className="absolute bottom-2 right-2 flex flex-col items-end gap-3 pointer-events-none"> | ||||
|       {children} | ||||
|       <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}`} | ||||
|           target="_blank" | ||||
|           rel="noopener noreferrer" | ||||
|           className={'!no-underline font-mono text-xs ' + linkOverrideClassName} | ||||
|         > | ||||
|           v{APP_VERSION} | ||||
|         </a> | ||||
|         <a | ||||
|           onClick={reportbug} | ||||
|           href="https://github.com/KittyCAD/modeling-app/issues/new/choose" | ||||
|           target="_blank" | ||||
|           rel="noopener noreferrer" | ||||
|         > | ||||
|           <CustomIcon | ||||
|             name="bug" | ||||
|             className={`w-5 h-5 ${linkOverrideClassName}`} | ||||
|           /> | ||||
|           <Tooltip position="top" contentClassName="text-xs"> | ||||
|             Report a bug | ||||
|           </Tooltip> | ||||
|         </a> | ||||
|         <Link | ||||
|           to={ | ||||
|             location.pathname.includes(PATHS.FILE) | ||||
|               ? filePath + PATHS.TELEMETRY + '?tab=project' | ||||
|               : PATHS.HOME + PATHS.TELEMETRY | ||||
|           } | ||||
|           data-testid="telemetry-link" | ||||
|         > | ||||
|           <CustomIcon | ||||
|             name="stopwatch" | ||||
|             className={`w-5 h-5 ${linkOverrideClassName}`} | ||||
|           /> | ||||
|           <span className="sr-only">Telemetry</span> | ||||
|           <Tooltip position="top" contentClassName="text-xs"> | ||||
|             Telemetry | ||||
|           </Tooltip> | ||||
|         </Link> | ||||
|         <Link | ||||
|           to={ | ||||
|             location.pathname.includes(PATHS.FILE) | ||||
|               ? filePath + PATHS.SETTINGS + '?tab=project' | ||||
|               : PATHS.HOME + PATHS.SETTINGS | ||||
|           } | ||||
|           data-testid="settings-link" | ||||
|         > | ||||
|           <CustomIcon | ||||
|             name="settings" | ||||
|             className={`w-5 h-5 ${linkOverrideClassName}`} | ||||
|           /> | ||||
|           <span className="sr-only">Settings</span> | ||||
|           <Tooltip position="top" contentClassName="text-xs"> | ||||
|             Settings | ||||
|           </Tooltip> | ||||
|         </Link> | ||||
|         <NetworkMachineIndicator className={linkOverrideClassName} /> | ||||
|         {!location.pathname.startsWith(PATHS.HOME) && ( | ||||
|           <NetworkHealthIndicator /> | ||||
|         )} | ||||
|         <HelpMenu /> | ||||
|       </menu> | ||||
|     </section> | ||||
|   ) | ||||
| } | ||||
|  | ||||
| @ -1,6 +1,39 @@ | ||||
| import { useEngineCommands } from './EngineCommands' | ||||
| import { Spinner } from './Spinner' | ||||
| import { CustomIcon } from './CustomIcon' | ||||
| import { StatusBarItemType } from './statusBar/statusBarTypes' | ||||
|  | ||||
| export const useModelStateStatus = (): StatusBarItemType => { | ||||
|   const [commands] = useEngineCommands() | ||||
|   const lastCommandType = commands[commands.length - 1]?.type | ||||
|  | ||||
|   let icon: StatusBarItemType['icon'] = 'loading' | ||||
|   const baseDataTestId = 'model-state-indicator' | ||||
|   let dataTestId = baseDataTestId | ||||
|  | ||||
|   if (lastCommandType === 'receive-reliable') { | ||||
|     icon = 'checkmark' | ||||
|     dataTestId = `${baseDataTestId}-receive-reliable` | ||||
|   } else if (lastCommandType === 'execution-done') { | ||||
|     icon = 'checkmark' | ||||
|     dataTestId = `${baseDataTestId}-execution-done` | ||||
|   } else if (lastCommandType === 'export-done') { | ||||
|     icon = 'checkmark' | ||||
|     dataTestId = `${baseDataTestId}-export-done` | ||||
|   } | ||||
|  | ||||
|   return { | ||||
|     id: 'model-state-indicator', | ||||
|     label: '', | ||||
|     icon, | ||||
|     toolTip: { | ||||
|       children: 'Model state indicator', | ||||
|     }, | ||||
|     element: 'button', | ||||
|     onClick: () => {}, | ||||
|     'data-testid': dataTestId, | ||||
|   } | ||||
| } | ||||
|  | ||||
| export const ModelStateIndicator = () => { | ||||
|   const [commands] = useEngineCommands() | ||||
|  | ||||
| @ -50,6 +50,7 @@ import { | ||||
|   isSketchPipe, | ||||
|   Selections, | ||||
|   updateSelections, | ||||
|   canLoftSelection, | ||||
| } from 'lib/selections' | ||||
| import { applyConstraintIntersect } from './Toolbar/Intersect' | ||||
| import { applyConstraintAbsDistance } from './Toolbar/SetAbsDistance' | ||||
| @ -66,7 +67,7 @@ import { | ||||
|   sketchOnOffsetPlane, | ||||
|   startSketchOnDefault, | ||||
| } from 'lang/modifyAst' | ||||
| import { Program, parse, recast } from 'lang/wasm' | ||||
| import { Program, parse, recast, resultIsOk } from 'lang/wasm' | ||||
| import { | ||||
|   doesSceneHaveSweepableSketch, | ||||
|   getNodePathFromSourceRange, | ||||
| @ -98,6 +99,7 @@ type MachineContext<T extends AnyStateMachine> = { | ||||
|   state: StateFrom<T> | ||||
|   context: ContextFrom<T> | ||||
|   send: Prop<Actor<T>, 'send'> | ||||
|   streamRef: React.RefObject<HTMLDivElement> | ||||
| } | ||||
|  | ||||
| export const ModelingMachineContext = createContext( | ||||
| @ -569,6 +571,21 @@ export const ModelingMachineProvider = ({ | ||||
|           if (err(canSweep)) return false | ||||
|           return canSweep | ||||
|         }, | ||||
|         'has valid loft selection': ({ context: { selectionRanges } }) => { | ||||
|           const hasNoSelection = | ||||
|             selectionRanges.graphSelections.length === 0 || | ||||
|             isRangeBetweenCharacters(selectionRanges) || | ||||
|             isSelectionLastLine(selectionRanges, codeManager.code) | ||||
|  | ||||
|           if (hasNoSelection) { | ||||
|             const count = 2 | ||||
|             return doesSceneHaveSweepableSketch(kclManager.ast, count) | ||||
|           } | ||||
|  | ||||
|           const canLoft = canLoftSelection(selectionRanges) | ||||
|           if (err(canLoft)) return false | ||||
|           return canLoft | ||||
|         }, | ||||
|         'has valid selection for deletion': ({ | ||||
|           context: { selectionRanges }, | ||||
|         }) => { | ||||
| @ -596,15 +613,11 @@ export const ModelingMachineProvider = ({ | ||||
|           ) | ||||
|         }, | ||||
|         'Has exportable geometry': () => { | ||||
|           if ( | ||||
|             kclManager.kclErrors.length === 0 && | ||||
|             kclManager.ast.body.length > 0 | ||||
|           ) | ||||
|           if (!kclManager.hasErrors() && kclManager.ast.body.length > 0) | ||||
|             return true | ||||
|           else { | ||||
|             let errorMessage = 'Unable to Export ' | ||||
|             if (kclManager.kclErrors.length > 0) | ||||
|               errorMessage += 'due to KCL Errors' | ||||
|             if (kclManager.hasErrors()) errorMessage += 'due to KCL Errors' | ||||
|             else if (kclManager.ast.body.length === 0) | ||||
|               errorMessage += 'due to Empty Scene' | ||||
|             console.error(errorMessage) | ||||
| @ -722,7 +735,11 @@ export const ModelingMachineProvider = ({ | ||||
|                 constraint: 'setHorzDistance', | ||||
|                 selectionRanges, | ||||
|               }) | ||||
|             const _modifiedAst = parse(recast(modifiedAst)) | ||||
|             const pResult = parse(recast(modifiedAst)) | ||||
|             if (trap(pResult) || !resultIsOk(pResult)) | ||||
|               return Promise.reject(new Error('Unexpected compilation error')) | ||||
|             const _modifiedAst = pResult.program | ||||
|  | ||||
|             if (!sketchDetails) | ||||
|               return Promise.reject(new Error('No sketch details')) | ||||
|             const updatedPathToNode = updatePathToNodeFromMap( | ||||
| @ -763,7 +780,10 @@ export const ModelingMachineProvider = ({ | ||||
|                 constraint: 'setVertDistance', | ||||
|                 selectionRanges, | ||||
|               }) | ||||
|             const _modifiedAst = parse(recast(modifiedAst)) | ||||
|             const pResult = parse(recast(modifiedAst)) | ||||
|             if (trap(pResult) || !resultIsOk(pResult)) | ||||
|               return Promise.reject(new Error('Unexpected compilation error')) | ||||
|             const _modifiedAst = pResult.program | ||||
|             if (!sketchDetails) | ||||
|               return Promise.reject(new Error('No sketch details')) | ||||
|             const updatedPathToNode = updatePathToNodeFromMap( | ||||
| @ -811,7 +831,10 @@ export const ModelingMachineProvider = ({ | ||||
|                   selectionRanges, | ||||
|                   angleOrLength: 'setAngle', | ||||
|                 })) | ||||
|             const _modifiedAst = parse(recast(modifiedAst)) | ||||
|             const pResult = parse(recast(modifiedAst)) | ||||
|             if (trap(pResult) || !resultIsOk(pResult)) | ||||
|               return Promise.reject(new Error('Unexpected compilation error')) | ||||
|             const _modifiedAst = pResult.program | ||||
|             if (err(_modifiedAst)) return Promise.reject(_modifiedAst) | ||||
|  | ||||
|             if (!sketchDetails) | ||||
| @ -853,7 +876,10 @@ export const ModelingMachineProvider = ({ | ||||
|               await applyConstraintAngleLength({ | ||||
|                 selectionRanges, | ||||
|               }) | ||||
|             const _modifiedAst = parse(recast(modifiedAst)) | ||||
|             const pResult = parse(recast(modifiedAst)) | ||||
|             if (trap(pResult) || !resultIsOk(pResult)) | ||||
|               return Promise.reject(new Error('Unexpected compilation error')) | ||||
|             const _modifiedAst = pResult.program | ||||
|             if (!sketchDetails) | ||||
|               return Promise.reject(new Error('No sketch details')) | ||||
|             const updatedPathToNode = updatePathToNodeFromMap( | ||||
| @ -893,7 +919,10 @@ export const ModelingMachineProvider = ({ | ||||
|               await applyConstraintIntersect({ | ||||
|                 selectionRanges, | ||||
|               }) | ||||
|             const _modifiedAst = parse(recast(modifiedAst)) | ||||
|             const pResult = parse(recast(modifiedAst)) | ||||
|             if (trap(pResult) || !resultIsOk(pResult)) | ||||
|               return Promise.reject(new Error('Unexpected compilation error')) | ||||
|             const _modifiedAst = pResult.program | ||||
|             if (!sketchDetails) | ||||
|               return Promise.reject(new Error('No sketch details')) | ||||
|             const updatedPathToNode = updatePathToNodeFromMap( | ||||
| @ -934,7 +963,10 @@ export const ModelingMachineProvider = ({ | ||||
|                 constraint: 'xAbs', | ||||
|                 selectionRanges, | ||||
|               }) | ||||
|             const _modifiedAst = parse(recast(modifiedAst)) | ||||
|             const pResult = parse(recast(modifiedAst)) | ||||
|             if (trap(pResult) || !resultIsOk(pResult)) | ||||
|               return Promise.reject(new Error('Unexpected compilation error')) | ||||
|             const _modifiedAst = pResult.program | ||||
|             if (!sketchDetails) | ||||
|               return Promise.reject(new Error('No sketch details')) | ||||
|             const updatedPathToNode = updatePathToNodeFromMap( | ||||
| @ -975,7 +1007,10 @@ export const ModelingMachineProvider = ({ | ||||
|                 constraint: 'yAbs', | ||||
|                 selectionRanges, | ||||
|               }) | ||||
|             const _modifiedAst = parse(recast(modifiedAst)) | ||||
|             const pResult = parse(recast(modifiedAst)) | ||||
|             if (trap(pResult) || !resultIsOk(pResult)) | ||||
|               return Promise.reject(new Error('Unexpected compilation error')) | ||||
|             const _modifiedAst = pResult.program | ||||
|             if (!sketchDetails) | ||||
|               return Promise.reject(new Error('No sketch details')) | ||||
|             const updatedPathToNode = updatePathToNodeFromMap( | ||||
| @ -1016,9 +1051,10 @@ export const ModelingMachineProvider = ({ | ||||
|             const { variableName } = await getVarNameModal({ | ||||
|               valueName: data?.variableName || 'var', | ||||
|             }) | ||||
|             let parsed = parse(recast(kclManager.ast)) | ||||
|             if (trap(parsed)) return Promise.reject(parsed) | ||||
|             parsed = parsed as Node<Program> | ||||
|             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( | ||||
| @ -1027,7 +1063,11 @@ export const ModelingMachineProvider = ({ | ||||
|                 data?.pathToNode || [], | ||||
|                 variableName | ||||
|               ) | ||||
|             parsed = parse(recast(_modifiedAst)) | ||||
|             pResult = parse(recast(_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) | ||||
| @ -1166,13 +1206,10 @@ export const ModelingMachineProvider = ({ | ||||
|         state: modelingState, | ||||
|         context: modelingState.context, | ||||
|         send: modelingSend, | ||||
|         streamRef, | ||||
|       }} | ||||
|     > | ||||
|       {/* TODO #818: maybe pass reff down to children/app.ts or render app.tsx directly? | ||||
|       since realistically it won't ever have generic children that isn't app.tsx */} | ||||
|       <div className="h-screen overflow-hidden select-none" ref={streamRef}> | ||||
|         {children} | ||||
|       </div> | ||||
|       {children} | ||||
|     </ModelingMachineContext.Provider> | ||||
|   ) | ||||
| } | ||||
|  | ||||
| @ -1,6 +1,6 @@ | ||||
| import { processMemory } from './MemoryPane' | ||||
| import { enginelessExecutor } from '../../../lib/testHelpers' | ||||
| import { initPromise, parse, ProgramMemory } from '../../../lang/wasm' | ||||
| import { assertParse, initPromise, ProgramMemory } from '../../../lang/wasm' | ||||
|  | ||||
| beforeAll(async () => { | ||||
|   await initPromise | ||||
| @ -28,12 +28,16 @@ describe('processMemory', () => { | ||||
|     |> lineTo([0.98, 5.16], %) | ||||
|     |> lineTo([2.15, 4.32], %) | ||||
|     // |> rx(90, %)` | ||||
|     const ast = parse(code) | ||||
|     const ast = assertParse(code) | ||||
|     const execState = await enginelessExecutor(ast, ProgramMemory.empty()) | ||||
|     const output = processMemory(execState.memory) | ||||
|     expect(output.myVar).toEqual(5) | ||||
|     expect(output.otherVar).toEqual(3) | ||||
|     expect(output).toEqual({ | ||||
|       HALF_TURN: 180, | ||||
|       QUARTER_TURN: 90, | ||||
|       THREE_QUARTER_TURN: 270, | ||||
|       ZERO: 0, | ||||
|       myVar: 5, | ||||
|       myFn: '__function(a)__', | ||||
|       otherVar: 3, | ||||
|  | ||||
| @ -90,7 +90,7 @@ export const sidebarPanes: SidebarPane[] = [ | ||||
|     keybinding: 'Shift + C', | ||||
|     showBadge: { | ||||
|       value: ({ kclContext }) => { | ||||
|         return kclContext.errors.length | ||||
|         return kclContext.diagnostics.length | ||||
|       }, | ||||
|       onClick: (e) => { | ||||
|         e.preventDefault() | ||||
|  | ||||
| @ -53,7 +53,7 @@ export function ModelingSidebar({ paneOpacity }: ModelingSidebarProps) { | ||||
|       settings: settings.context, | ||||
|       platform: getPlatformString(), | ||||
|     }), | ||||
|     [kclContext.errors, settings.context] | ||||
|     [kclContext.diagnostics, settings.context] | ||||
|   ) | ||||
|  | ||||
|   const sidebarActions: SidebarAction[] = [ | ||||
|  | ||||
| @ -6,6 +6,7 @@ import { useNetworkContext } from '../hooks/useNetworkContext' | ||||
| import { NetworkHealthState } from '../hooks/useNetworkStatus' | ||||
| import { toSync } from 'lib/utils' | ||||
| import { reportRejection } from 'lib/trap' | ||||
| import { StatusBarItemType } from './statusBar/statusBarTypes' | ||||
|  | ||||
| export const NETWORK_HEALTH_TEXT: Record<NetworkHealthState, string> = { | ||||
|   [NetworkHealthState.Ok]: 'Connected', | ||||
| @ -64,14 +65,28 @@ const overallConnectionStateColor: Record<NetworkHealthState, IconColorConfig> = | ||||
|     }, | ||||
|   } | ||||
|  | ||||
| const overallConnectionStateIcon: Record< | ||||
|   NetworkHealthState, | ||||
|   ActionIconProps['icon'] | ||||
| > = { | ||||
| const overallConnectionStateIcon = { | ||||
|   [NetworkHealthState.Ok]: 'network', | ||||
|   [NetworkHealthState.Weak]: 'network', | ||||
|   [NetworkHealthState.Issue]: 'networkCrossedOut', | ||||
|   [NetworkHealthState.Disconnected]: 'networkCrossedOut', | ||||
| } as const | ||||
|  | ||||
| export const useNetworkHealthStatus = (): StatusBarItemType => { | ||||
|   const { overallState } = useNetworkContext() | ||||
|  | ||||
|   return { | ||||
|     id: 'network-health', | ||||
|     label: `Network health (${NETWORK_HEALTH_TEXT[overallState]})`, | ||||
|     hideLabel: true, | ||||
|     element: 'popover', | ||||
|     className: overallConnectionStateColor[overallState].icon, | ||||
|     toolTip: { | ||||
|       children: `Network health (${NETWORK_HEALTH_TEXT[overallState]})`, | ||||
|     }, | ||||
|     icon: overallConnectionStateIcon[overallState], | ||||
|     popoverContent: <NetworkHealthPopoverContent />, | ||||
|   } | ||||
| } | ||||
|  | ||||
| export const NetworkHealthIndicator = () => { | ||||
| @ -109,81 +124,95 @@ export const NetworkHealthIndicator = () => { | ||||
|           Network health ({NETWORK_HEALTH_TEXT[overallState]}) | ||||
|         </Tooltip> | ||||
|       </Popover.Button> | ||||
|       <Popover.Panel | ||||
|         className="absolute right-0 left-auto bottom-full mb-1 w-64 flex flex-col gap-1 align-stretch bg-chalkboard-10 dark:bg-chalkboard-90 rounded shadow-lg border border-solid border-chalkboard-20/50 dark:border-chalkboard-80/50 text-sm" | ||||
|         data-testid="network-popover" | ||||
|       > | ||||
|         <div | ||||
|           className={`flex items-center justify-between p-2 rounded-t-sm ${overallConnectionStateColor[overallState].bg} ${overallConnectionStateColor[overallState].icon}`} | ||||
|         > | ||||
|           <h2 className="text-sm font-sans font-normal">Network health</h2> | ||||
|           <p | ||||
|             data-testid="network" | ||||
|             className="font-bold text-xs uppercase px-2 py-1 rounded-sm" | ||||
|           > | ||||
|             {NETWORK_HEALTH_TEXT[overallState]} | ||||
|           </p> | ||||
|         </div> | ||||
|         <ul className="divide-y divide-chalkboard-20 dark:divide-chalkboard-80"> | ||||
|           {Object.keys(steps).map((name) => ( | ||||
|             <li | ||||
|               key={name} | ||||
|               className={'flex flex-col px-2 py-4 gap-1 last:mb-0 '} | ||||
|             > | ||||
|               <div className="flex items-center text-left gap-1"> | ||||
|                 <p className="flex-1">{name}</p> | ||||
|                 {internetConnected ? ( | ||||
|                   <ActionIcon | ||||
|                     size="lg" | ||||
|                     icon={ | ||||
|                       hasIssueToIcon[ | ||||
|                         String(issues[name as ConnectingTypeGroup]) | ||||
|                       ] | ||||
|                     } | ||||
|                     iconClassName={ | ||||
|                       hasIssueToIconColors[ | ||||
|                         String(issues[name as ConnectingTypeGroup]) | ||||
|                       ].icon | ||||
|                     } | ||||
|                     bgClassName={ | ||||
|                       'rounded-sm ' + | ||||
|                       hasIssueToIconColors[ | ||||
|                         String(issues[name as ConnectingTypeGroup]) | ||||
|                       ].bg | ||||
|                     } | ||||
|                   /> | ||||
|                 ) : ( | ||||
|                   <ActionIcon | ||||
|                     icon={hasIssueToIcon.true} | ||||
|                     bgClassName={hasIssueToIconColors.true.bg} | ||||
|                     iconClassName={hasIssueToIconColors.true.icon} | ||||
|                   /> | ||||
|                 )} | ||||
|               </div> | ||||
|               {issues[name as ConnectingTypeGroup] && ( | ||||
|                 <button | ||||
|                   onClick={toSync(async () => { | ||||
|                     await navigator.clipboard.writeText( | ||||
|                       JSON.stringify(error, null, 2) || '' | ||||
|                     ) | ||||
|                     setHasCopied(true) | ||||
|                     setTimeout(() => setHasCopied(false), 5000) | ||||
|                   }, reportRejection)} | ||||
|                   className="flex w-fit gap-2 items-center bg-transparent text-sm p-1 py-0 my-0 -mx-1 text-destroy-80 dark:text-destroy-10 hover:bg-transparent border-transparent dark:border-transparent hover:border-destroy-80 dark:hover:border-destroy-80 dark:hover:bg-destroy-80" | ||||
|                 > | ||||
|                   {hasCopied ? 'Copied' : 'Copy Error'} | ||||
|                   <ActionIcon | ||||
|                     size="lg" | ||||
|                     icon={hasCopied ? 'clipboardCheckmark' : 'clipboardPlus'} | ||||
|                     iconClassName="text-inherit dark:text-inherit" | ||||
|                     bgClassName="!bg-transparent" | ||||
|                   /> | ||||
|                 </button> | ||||
|               )} | ||||
|             </li> | ||||
|           ))} | ||||
|         </ul> | ||||
|       <Popover.Panel> | ||||
|         <NetworkHealthPopoverContent /> | ||||
|       </Popover.Panel> | ||||
|     </Popover> | ||||
|   ) | ||||
| } | ||||
|  | ||||
| const NetworkHealthPopoverContent = () => { | ||||
|   const { | ||||
|     hasIssues, | ||||
|     overallState, | ||||
|     internetConnected, | ||||
|     steps, | ||||
|     issues, | ||||
|     error, | ||||
|     setHasCopied, | ||||
|     hasCopied, | ||||
|   } = useNetworkContext() | ||||
|  | ||||
|   return ( | ||||
|     <div | ||||
|       className="absolute left-2 bottom-full mb-1 w-64 flex flex-col gap-1 align-stretch bg-chalkboard-10 dark:bg-chalkboard-90 rounded shadow-lg border border-solid border-chalkboard-20/50 dark:border-chalkboard-80/50 text-sm" | ||||
|       data-testid="network-popover" | ||||
|     > | ||||
|       <div | ||||
|         className={`flex items-center justify-between p-2 rounded-t-sm ${overallConnectionStateColor[overallState].bg} ${overallConnectionStateColor[overallState].icon}`} | ||||
|       > | ||||
|         <h2 className="text-sm font-sans font-normal">Network health</h2> | ||||
|         <p | ||||
|           data-testid="network" | ||||
|           className="font-bold text-xs uppercase px-2 py-1 rounded-sm" | ||||
|         > | ||||
|           {NETWORK_HEALTH_TEXT[overallState]} | ||||
|         </p> | ||||
|       </div> | ||||
|       <ul className="divide-y divide-chalkboard-20 dark:divide-chalkboard-80"> | ||||
|         {Object.keys(steps).map((name) => ( | ||||
|           <li key={name} className={'flex flex-col px-2 py-4 gap-1 last:mb-0 '}> | ||||
|             <div className="flex items-center text-left gap-1"> | ||||
|               <p className="flex-1">{name}</p> | ||||
|               {internetConnected ? ( | ||||
|                 <ActionIcon | ||||
|                   size="lg" | ||||
|                   icon={ | ||||
|                     hasIssueToIcon[String(issues[name as ConnectingTypeGroup])] | ||||
|                   } | ||||
|                   iconClassName={ | ||||
|                     hasIssueToIconColors[ | ||||
|                       String(issues[name as ConnectingTypeGroup]) | ||||
|                     ].icon | ||||
|                   } | ||||
|                   bgClassName={ | ||||
|                     'rounded-sm ' + | ||||
|                     hasIssueToIconColors[ | ||||
|                       String(issues[name as ConnectingTypeGroup]) | ||||
|                     ].bg | ||||
|                   } | ||||
|                 /> | ||||
|               ) : ( | ||||
|                 <ActionIcon | ||||
|                   icon={hasIssueToIcon.true} | ||||
|                   bgClassName={hasIssueToIconColors.true.bg} | ||||
|                   iconClassName={hasIssueToIconColors.true.icon} | ||||
|                 /> | ||||
|               )} | ||||
|             </div> | ||||
|             {issues[name as ConnectingTypeGroup] && ( | ||||
|               <button | ||||
|                 onClick={toSync(async () => { | ||||
|                   await navigator.clipboard.writeText( | ||||
|                     JSON.stringify(error, null, 2) || '' | ||||
|                   ) | ||||
|                   setHasCopied(true) | ||||
|                   setTimeout(() => setHasCopied(false), 5000) | ||||
|                 }, reportRejection)} | ||||
|                 className="flex w-fit gap-2 items-center bg-transparent text-sm p-1 py-0 my-0 -mx-1 text-destroy-80 dark:text-destroy-10 hover:bg-transparent border-transparent dark:border-transparent hover:border-destroy-80 dark:hover:border-destroy-80 dark:hover:bg-destroy-80" | ||||
|               > | ||||
|                 {hasCopied ? 'Copied' : 'Copy Error'} | ||||
|                 <ActionIcon | ||||
|                   size="lg" | ||||
|                   icon={hasCopied ? 'clipboardCheckmark' : 'clipboardPlus'} | ||||
|                   iconClassName="text-inherit dark:text-inherit" | ||||
|                   bgClassName="!bg-transparent" | ||||
|                 /> | ||||
|               </button> | ||||
|             )} | ||||
|           </li> | ||||
|         ))} | ||||
|       </ul> | ||||
|     </div> | ||||
|   ) | ||||
| } | ||||
|  | ||||
| @ -5,6 +5,7 @@ import { isDesktop } from 'lib/isDesktop' | ||||
| import { components } from 'lib/machine-api' | ||||
| import { MachineManagerContext } from 'components/MachineManagerProvider' | ||||
| import { CustomIcon } from './CustomIcon' | ||||
| import { StatusBarItemType } from './statusBar/statusBarTypes' | ||||
|  | ||||
| export const NetworkMachineIndicator = ({ | ||||
|   className, | ||||
| @ -27,12 +28,7 @@ export const NetworkMachineIndicator = ({ | ||||
|         } | ||||
|         data-testid="network-machine-toggle" | ||||
|       > | ||||
|         <CustomIcon name="printer3d" className="w-5 h-5" /> | ||||
|         {machineCount > 0 && ( | ||||
|           <p aria-hidden className="flex items-center justify-center text-xs"> | ||||
|             {machineCount} | ||||
|           </p> | ||||
|         )} | ||||
|         <NetworkMachinesIcon machineCount={machineCount} /> | ||||
|         <Tooltip position="top-right" wrapperClassName="ui-open:hidden"> | ||||
|           Network machines ({machineCount}) {reason && `: ${reason}`} | ||||
|         </Tooltip> | ||||
| @ -41,50 +37,92 @@ export const NetworkMachineIndicator = ({ | ||||
|         className="absolute right-0 left-auto bottom-full mb-1 w-64 flex flex-col gap-1 align-stretch bg-chalkboard-10 dark:bg-chalkboard-90 rounded shadow-lg border border-solid border-chalkboard-20/50 dark:border-chalkboard-80/50 text-sm" | ||||
|         data-testid="network-popover" | ||||
|       > | ||||
|         <div className="flex items-center justify-between p-2 rounded-t-sm bg-chalkboard-20 dark:bg-chalkboard-80"> | ||||
|           <h2 className="text-sm font-sans font-normal">Network machines</h2> | ||||
|           <p | ||||
|             data-testid="network" | ||||
|             className="font-bold text-xs uppercase px-2 py-1 rounded-sm" | ||||
|           > | ||||
|             {machineCount} | ||||
|           </p> | ||||
|         </div> | ||||
|         {machineCount > 0 && ( | ||||
|           <ul className="divide-y divide-chalkboard-20 dark:divide-chalkboard-80"> | ||||
|             {machines.map( | ||||
|               (machine: components['schemas']['MachineInfoResponse']) => { | ||||
|                 return ( | ||||
|                   <li key={machine.id} className={'px-2 py-4 gap-1 last:mb-0 '}> | ||||
|                     <p className="">{machine.id.toUpperCase()}</p> | ||||
|                     <p className="text-chalkboard-60 dark:text-chalkboard-50 text-xs"> | ||||
|                       {machine.make_model.model} | ||||
|                     </p> | ||||
|                     {machine.extra && | ||||
|                       machine.extra.type === 'bambu' && | ||||
|                       machine.extra.nozzle_diameter && ( | ||||
|                         <p className="text-chalkboard-60 dark:text-chalkboard-50 text-xs"> | ||||
|                           Nozzle Diameter: {machine.extra.nozzle_diameter} | ||||
|                         </p> | ||||
|                       )} | ||||
|                     <p className="text-chalkboard-60 dark:text-chalkboard-50 text-xs"> | ||||
|                       {`Status: ${machine.state.state | ||||
|                         .charAt(0) | ||||
|                         .toUpperCase()}${machine.state.state.slice(1)}`} | ||||
|                       {machine.state.state === 'failed' && machine.state.message | ||||
|                         ? ` (${machine.state.message})` | ||||
|                         : ''} | ||||
|                       {machine.state.state === 'running' && machine.progress | ||||
|                         ? ` (${Math.round(machine.progress)}%)` | ||||
|                         : ''} | ||||
|                     </p> | ||||
|                   </li> | ||||
|                 ) | ||||
|               } | ||||
|             )} | ||||
|           </ul> | ||||
|         )} | ||||
|         <NetworkMachinesPopoverContent machines={machines} /> | ||||
|       </Popover.Panel> | ||||
|     </Popover> | ||||
|   ) : null | ||||
| } | ||||
|  | ||||
| export const useNetworkMachineStatus = (): StatusBarItemType => { | ||||
|   const { | ||||
|     noMachinesReason, | ||||
|     machines, | ||||
|     machines: { length: machineCount }, | ||||
|   } = useContext(MachineManagerContext) | ||||
|   const reason = noMachinesReason() | ||||
|  | ||||
|   return { | ||||
|     id: 'network-machines', | ||||
|     label: `Network machines (${machineCount}) ${reason && `: ${reason}`}`, | ||||
|     hideLabel: true, | ||||
|     element: 'popover', | ||||
|     toolTip: { | ||||
|       children: `Network machines (${machineCount}) ${reason && `: ${reason}`}`, | ||||
|     }, | ||||
|     icon: 'printer3d', | ||||
|     popoverContent: <NetworkMachinesPopoverContent machines={machines} />, | ||||
|   } | ||||
| } | ||||
|  | ||||
| function NetworkMachinesIcon({ machineCount }: { machineCount: number }) { | ||||
|   return ( | ||||
|     <> | ||||
|       <CustomIcon name="printer3d" className="w-5 h-5" /> | ||||
|       {machineCount > 0 && ( | ||||
|         <p aria-hidden className="flex items-center justify-center text-xs"> | ||||
|           {machineCount} | ||||
|         </p> | ||||
|       )} | ||||
|     </> | ||||
|   ) | ||||
| } | ||||
|  | ||||
| function NetworkMachinesPopoverContent({ machines }: { machines: components['schemas']['MachineInfoResponse'][] }) { | ||||
|   return ( | ||||
|     <> | ||||
|       <div className="flex items-center justify-between p-2 rounded-t-sm bg-chalkboard-20 dark:bg-chalkboard-80"> | ||||
|         <h2 className="text-sm font-sans font-normal">Network machines</h2> | ||||
|         <p | ||||
|           data-testid="network" | ||||
|           className="font-bold text-xs uppercase px-2 py-1 rounded-sm" | ||||
|         > | ||||
|           {machines.length} | ||||
|         </p> | ||||
|       </div> | ||||
|       {machines.length > 0 && ( | ||||
|         <ul className="divide-y divide-chalkboard-20 dark:divide-chalkboard-80"> | ||||
|           {machines.map( | ||||
|             (machine: components['schemas']['MachineInfoResponse']) => { | ||||
|               return ( | ||||
|                 <li key={machine.id} className={'px-2 py-4 gap-1 last:mb-0 '}> | ||||
|                   <p className="">{machine.id.toUpperCase()}</p> | ||||
|                   <p className="text-chalkboard-60 dark:text-chalkboard-50 text-xs"> | ||||
|                     {machine.make_model.model} | ||||
|                   </p> | ||||
|                   {machine.extra && | ||||
|                     machine.extra.type === 'bambu' && | ||||
|                     machine.extra.nozzle_diameter && ( | ||||
|                       <p className="text-chalkboard-60 dark:text-chalkboard-50 text-xs"> | ||||
|                         Nozzle Diameter: {machine.extra.nozzle_diameter} | ||||
|                       </p> | ||||
|                     )} | ||||
|                   <p className="text-chalkboard-60 dark:text-chalkboard-50 text-xs"> | ||||
|                     {`Status: ${machine.state.state | ||||
|                       .charAt(0) | ||||
|                       .toUpperCase()}${machine.state.state.slice(1)}`} | ||||
|                     {machine.state.state === 'failed' && machine.state.message | ||||
|                       ? ` (${machine.state.message})` | ||||
|                       : ''} | ||||
|                     {machine.state.state === 'running' && machine.progress | ||||
|                       ? ` (${Math.round(machine.progress)}%)` | ||||
|                       : ''} | ||||
|                   </p> | ||||
|                 </li> | ||||
|               ) | ||||
|             } | ||||
|           )} | ||||
|         </ul> | ||||
|       )} | ||||
|     </> | ||||
|   ) | ||||
| } | ||||
| @ -68,8 +68,8 @@ function AppLogoLink({ | ||||
|       data-testid="app-logo" | ||||
|       onClick={() => { | ||||
|         onProjectClose(file || null, project?.path || null, false) | ||||
|         // Clear the scene and end the session. | ||||
|         engineCommandManager.endSession() | ||||
|         // Clear the scene. | ||||
|         engineCommandManager.clearScene() | ||||
|       }} | ||||
|       to={PATHS.HOME} | ||||
|       className={wrapperClassName + ' hover:before:brightness-110'} | ||||
| @ -190,8 +190,8 @@ function ProjectMenuPopover({ | ||||
|           className: !isDesktop() ? 'hidden' : '', | ||||
|           onClick: () => { | ||||
|             onProjectClose(file || null, project?.path || null, true) | ||||
|             // Clear the scene and end the session. | ||||
|             engineCommandManager.endSession() | ||||
|             // Clear the scene. | ||||
|             engineCommandManager.clearScene() | ||||
|           }, | ||||
|         }, | ||||
|       ].filter( | ||||
|  | ||||
| @ -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 } from 'lib/appVersion' | ||||
| import { PATHS } from 'lib/paths' | ||||
| import { | ||||
|   createAndOpenNewTutorialProject, | ||||
| @ -25,6 +25,7 @@ import { useLspContext } from 'components/LspProvider' | ||||
| import { toSync } from 'lib/utils' | ||||
| import { reportRejection } from 'lib/trap' | ||||
| import { openExternalBrowserIfDesktop } from 'lib/openWindow' | ||||
| import { PACKAGE_NAME } from 'routes/Settings' | ||||
|  | ||||
| interface AllSettingsFieldsProps { | ||||
|   searchParamTab: SettingsLevel | ||||
|  | ||||
							
								
								
									
										148
									
								
								src/components/StatusBar.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,148 @@ | ||||
| import { useEffect } from 'react' | ||||
| import { ActionButton } from './ActionButton' | ||||
| import { StatusBarItemType } from './statusBar/statusBarTypes' | ||||
| import Tooltip, { TooltipProps } from './Tooltip' | ||||
| import { ActionIcon } from './ActionIcon' | ||||
| import { Popover } from '@headlessui/react' | ||||
|  | ||||
| export function StatusBar({ | ||||
|   globalItems, | ||||
|   localItems, | ||||
| }: { | ||||
|   globalItems: StatusBarItemType[] | ||||
|   localItems: StatusBarItemType[] | ||||
| }) { | ||||
|   return ( | ||||
|     <footer | ||||
|       id="statusbar" | ||||
|       className="relative z-10 flex justify-between items-center bg-chalkboard-20 dark:bg-chalkboard-90 text-chalkboard-80 dark:text-chalkboard-30 border-t border-t-chalkboard-30 dark:border-t-chalkboard-80" | ||||
|     > | ||||
|       <menu id="statusbar-globals" className="flex items-stretch"> | ||||
|         {globalItems.map((item) => ( | ||||
|           <StatusBarItem key={item.id} {...item} position="left" /> | ||||
|         ))} | ||||
|       </menu> | ||||
|       <menu id="statusbar-locals" className="flex items-stretch"> | ||||
|         {localItems.map((item) => ( | ||||
|           <StatusBarItem key={item.id} {...item} position="right" /> | ||||
|         ))} | ||||
|       </menu> | ||||
|     </footer> | ||||
|   ) | ||||
| } | ||||
|  | ||||
| function StatusBarItem( | ||||
|   props: StatusBarItemType & { position: 'left' | 'middle' | 'right' } | ||||
| ) { | ||||
|   const defaultClassNames = `px-2 py-1 text-xs text-chalkboard-80 dark:text-chalkboard-30 rounded-none border-none hover:bg-chalkboard-30 dark:hover:bg-chalkboard-80 focus:bg-chalkboard-30 dark:focus:bg-chalkboard-80 hover:text-chalkboard-100 dark:hover:text-chalkboard-10 focustext-chalkboard-100 dark:focus:text-chalkboard-10  focus:outline-none focus-visible:ring-2 focus:ring-primary focus:ring-opacity-50` | ||||
|   const tooltipPosition: TooltipProps['position'] = | ||||
|     props.position === 'middle' ? 'top' : `top-${props.position}` | ||||
|  | ||||
|   switch (props.element) { | ||||
|     case 'button': | ||||
|       return ( | ||||
|         <ActionButton | ||||
|           Element="button" | ||||
|           iconStart={ | ||||
|             props.icon && { | ||||
|               icon: props.icon, | ||||
|               iconClassName: props.icon === 'loading' ? 'animate-spin' : '', | ||||
|               bgClassName: 'bg-transparent dark:bg-transparent', | ||||
|             } | ||||
|           } | ||||
|           className={defaultClassNames + ' ' + props.className} | ||||
|           data-testid={props['data-testid']} | ||||
|         > | ||||
|           {props.label && ( | ||||
|             <span className={props.hideLabel ? 'sr-only' : ''}> | ||||
|               {props.label} | ||||
|             </span> | ||||
|           )} | ||||
|           {props.toolTip && ( | ||||
|             <Tooltip {...props.toolTip} position={tooltipPosition} /> | ||||
|           )} | ||||
|         </ActionButton> | ||||
|       ) | ||||
|     case 'popover': | ||||
|       return ( | ||||
|         <Popover className="relative"> | ||||
|           <Popover.Button | ||||
|             as={ActionButton} | ||||
|             Element="button" | ||||
|             iconStart={ | ||||
|               props.icon && { | ||||
|                 icon: props.icon, | ||||
|                 iconClassName: props.icon === 'loading' ? 'animate-spin' : '', | ||||
|                 bgClassName: 'bg-transparent dark:bg-transparent', | ||||
|               } | ||||
|             } | ||||
|             className={defaultClassNames + ' ' + props.className} | ||||
|             data-testid={props['data-testid']} | ||||
|           > | ||||
|             {props.label && ( | ||||
|               <span className={props.hideLabel ? 'sr-only' : ''}> | ||||
|                 {props.label} | ||||
|               </span> | ||||
|             )} | ||||
|             {props.toolTip && ( | ||||
|               <Tooltip | ||||
|                 {...props.toolTip} | ||||
|                 wrapperClassName={`${ | ||||
|                   props.toolTip?.wrapperClassName || '' | ||||
|                 } ui-open:hidden`} | ||||
|                 position={tooltipPosition} | ||||
|               /> | ||||
|             )} | ||||
|           </Popover.Button> | ||||
|           <Popover.Panel>{props.popoverContent}</Popover.Panel> | ||||
|         </Popover> | ||||
|       ) | ||||
|     case 'text': | ||||
|       return ( | ||||
|         <div | ||||
|           role="tooltip" | ||||
|           className={defaultClassNames + ' ' + props.className} | ||||
|         > | ||||
|           {props.icon && ( | ||||
|             <ActionIcon | ||||
|               icon={props.icon} | ||||
|               iconClassName={props.icon === 'loading' ? 'animate-spin' : ''} | ||||
|               bgClassName="bg-transparent dark:bg-transparent" | ||||
|             /> | ||||
|           )} | ||||
|           {props.label && ( | ||||
|             <span className={props.hideLabel ? 'sr-only' : ''}> | ||||
|               {props.label} | ||||
|             </span> | ||||
|           )} | ||||
|           {props.toolTip && ( | ||||
|             <Tooltip {...props.toolTip} position={tooltipPosition} /> | ||||
|           )} | ||||
|         </div> | ||||
|       ) | ||||
|     default: | ||||
|       return ( | ||||
|         <ActionButton | ||||
|           Element={props.element} | ||||
|           to={props.href} | ||||
|           iconStart={ | ||||
|             props.icon && { | ||||
|               icon: props.icon, | ||||
|               bgClassName: 'bg-transparent dark:bg-transparent', | ||||
|             } | ||||
|           } | ||||
|           className={defaultClassNames + ' ' + props.className} | ||||
|           data-testid={props['data-testid']} | ||||
|         > | ||||
|           {props.label && ( | ||||
|             <span className={props.hideLabel ? 'sr-only' : ''}> | ||||
|               {props.label} | ||||
|             </span> | ||||
|           )} | ||||
|           {props.toolTip && ( | ||||
|             <Tooltip {...props.toolTip} position={tooltipPosition} /> | ||||
|           )} | ||||
|         </ActionButton> | ||||
|       ) | ||||
|   } | ||||
| } | ||||
| @ -40,7 +40,10 @@ export function removeConstrainingValuesInfo({ | ||||
|         otherSelections: [], | ||||
|         graphSelections: nodes.map( | ||||
|           (node): Selection => ({ | ||||
|             codeRef: codeRefFromRange([node.start, node.end], kclManager.ast), | ||||
|             codeRef: codeRefFromRange( | ||||
|               [node.start, node.end, true], | ||||
|               kclManager.ast | ||||
|             ), | ||||
|           }) | ||||
|         ), | ||||
|       } | ||||
|  | ||||
| @ -8,7 +8,7 @@ type LeftOrRight = 'left' | 'right' | ||||
| type Corner = `${TopOrBottom}-${LeftOrRight}` | ||||
| type TooltipPosition = TopOrBottom | LeftOrRight | Corner | ||||
|  | ||||
| interface TooltipProps extends React.PropsWithChildren { | ||||
| export interface TooltipProps extends React.PropsWithChildren { | ||||
|   position?: TooltipPosition | ||||
|   wrapperClassName?: string | ||||
|   contentClassName?: string | ||||
|  | ||||
							
								
								
									
										96
									
								
								src/components/statusBar/homeDefaultStatusBarItems.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,96 @@ | ||||
| import openWindow from 'lib/openWindow' | ||||
| import { StatusBarItemType } from './statusBarTypes' | ||||
| import { reportRejection } from 'lib/trap' | ||||
| import { CoreDumpManager } from 'lib/coredump' | ||||
| import toast from 'react-hot-toast' | ||||
| import { coreDump } from 'lang/wasm' | ||||
| import { APP_VERSION } from 'lib/appVersion' | ||||
| import { Location } from 'react-router-dom' | ||||
| import { PATHS } from 'lib/paths' | ||||
|  | ||||
| export const homeDefaultStatusBarItems = ({ | ||||
|   coreDumpManager, | ||||
|   location, | ||||
| }: { | ||||
|   coreDumpManager?: CoreDumpManager | ||||
|   location: Location | ||||
| }): StatusBarItemType[] => [ | ||||
|   { | ||||
|     id: 'version', | ||||
|     element: 'externalLink', | ||||
|     label: `v${APP_VERSION}`, | ||||
|     href: `https://github.com/KittyCAD/modeling-app/releases/tag/v${APP_VERSION}`, | ||||
|     toolTip: { | ||||
|       children: 'View the release notes on GitHub', | ||||
|     }, | ||||
|   }, | ||||
|   { | ||||
|     id: 'report-bug', | ||||
|     element: 'button', | ||||
|     icon: 'bug', | ||||
|     label: 'Report a bug', | ||||
|     onClick: (event) => reportBug(event, { coreDumpManager }), | ||||
|     toolTip: { | ||||
|       children: 'Send your current app state to the developers for debugging', | ||||
|     }, | ||||
|   }, | ||||
|   { | ||||
|     id: 'settings', | ||||
|     element: 'link', | ||||
|     icon: 'settings', | ||||
|     href: | ||||
|       '.' + | ||||
|       PATHS.SETTINGS + | ||||
|       (location.pathname.includes(PATHS.FILE) ? '?tab=project' : ''), | ||||
|     'data-testid': 'settings-link', | ||||
|     label: 'Settings', | ||||
|     toolTip: { | ||||
|       children: 'Settings', | ||||
|     }, | ||||
|   }, | ||||
| ] | ||||
|  | ||||
| function reportBug( | ||||
|   event: { | ||||
|     preventDefault: () => void | ||||
|     stopPropagation: () => void | ||||
|   }, | ||||
|   dependencies: { | ||||
|     coreDumpManager: CoreDumpManager | undefined | ||||
|   } | ||||
| ) { | ||||
|   event?.preventDefault() | ||||
|   event?.stopPropagation() | ||||
|   const { coreDumpManager } = dependencies | ||||
|  | ||||
|   if (!coreDumpManager) { | ||||
|     // open default reporting option | ||||
|     openWindow( | ||||
|       'https://github.com/KittyCAD/modeling-app/issues/new/choose' | ||||
|     ).catch(reportRejection) | ||||
|   } else { | ||||
|     toast | ||||
|       .promise( | ||||
|         coreDump(coreDumpManager, true), | ||||
|         { | ||||
|           loading: 'Preparing bug report...', | ||||
|           success: 'Bug report opened in new window', | ||||
|           error: 'Unable to export a core dump. Using default reporting.', | ||||
|         }, | ||||
|         { | ||||
|           success: { | ||||
|             // Note: this extended duration is especially important for Playwright e2e testing | ||||
|             // default duration is 2000 - https://react-hot-toast.com/docs/toast#default-durations | ||||
|             duration: 6000, | ||||
|           }, | ||||
|         } | ||||
|       ) | ||||
|       .catch((err: Error) => { | ||||
|         if (err) { | ||||
|           openWindow( | ||||
|             'https://github.com/KittyCAD/modeling-app/issues/new/choose' | ||||
|           ).catch(reportRejection) | ||||
|         } | ||||
|       }) | ||||
|   } | ||||
| } | ||||
							
								
								
									
										28
									
								
								src/components/statusBar/statusBarTypes.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,28 @@ | ||||
| import { CustomIconName } from 'components/CustomIcon' | ||||
| import { TooltipProps } from 'components/Tooltip' | ||||
|  | ||||
| export type StatusBarItemType = { | ||||
|   id: string | ||||
|   label: string | ||||
|   icon?: CustomIconName | ||||
|   hideLabel?: boolean | ||||
|   toolTip?: Omit<TooltipProps, 'position'> | ||||
|   className?: string | ||||
|   ['data-testid']?: string | ||||
| } & ( | ||||
|   | { | ||||
|       element: 'button' | ||||
|       onClick: (event: React.MouseEvent<HTMLButtonElement>) => void | ||||
|     } | ||||
|   | { | ||||
|       element: 'popover' | ||||
|       popoverContent: React.ReactNode | ||||
|     } | ||||
|   | { | ||||
|       element: 'link' | 'externalLink' | ||||
|       href: string | ||||
|     } | ||||
|   | { | ||||
|       element: 'text' | ||||
|     } | ||||
| ) | ||||
| @ -139,7 +139,9 @@ export default class EditorManager { | ||||
|   } | ||||
|  | ||||
|   setHighlightRange(range: Array<Selection['codeRef']['range']>): void { | ||||
|     this._highlightRange = range | ||||
|     this._highlightRange = range.map((s): [number, number] => { | ||||
|       return [s[0], s[1]] | ||||
|     }) | ||||
|  | ||||
|     const selectionsWithSafeEnds = range.map((s): [number, number] => { | ||||
|       const safeEnd = Math.min(s[1], this._editorView?.state.doc.length || s[1]) | ||||
|  | ||||
| @ -18,7 +18,7 @@ import { | ||||
| import { err, reportRejection } from 'lib/trap' | ||||
| import { DefaultPlaneStr, getFaceDetails } from 'clientSideScene/sceneEntities' | ||||
| import { getNodeFromPath, getNodePathFromSourceRange } from 'lang/queryAst' | ||||
| import { CallExpression } from 'lang/wasm' | ||||
| import { CallExpression, defaultSourceRange } from 'lang/wasm' | ||||
| import { EdgeCutInfo, ExtrudeFacePlane } from 'machines/modelingMachine' | ||||
|  | ||||
| export function useEngineConnectionSubscriptions() { | ||||
| @ -46,7 +46,7 @@ export function useEngineConnectionSubscriptions() { | ||||
|           (editorManager.highlightRange[0][0] !== 0 && | ||||
|             editorManager.highlightRange[0][1] !== 0) | ||||
|         ) { | ||||
|           editorManager.setHighlightRange([[0, 0]]) | ||||
|           editorManager.setHighlightRange([defaultSourceRange()]) | ||||
|         } | ||||
|       }, | ||||
|     }) | ||||
| @ -201,7 +201,7 @@ export function useEngineConnectionSubscriptions() { | ||||
|               const { z_axis, y_axis, origin } = faceInfo | ||||
|               const sketchPathToNode = getNodePathFromSourceRange( | ||||
|                 kclManager.ast, | ||||
|                 err(codeRef) ? [0, 0] : codeRef.range | ||||
|                 err(codeRef) ? defaultSourceRange() : codeRef.range | ||||
|               ) | ||||
|  | ||||
|               const getEdgeCutMeta = (): null | EdgeCutInfo => { | ||||
|  | ||||
| @ -1,15 +1,15 @@ | ||||
| import { KCLError } from './errors' | ||||
| import { createContext, useContext, useEffect, useState } from 'react' | ||||
| import { type IndexLoaderData } from 'lib/types' | ||||
| import { useLoaderData } from 'react-router-dom' | ||||
| import { codeManager, kclManager } from 'lib/singletons' | ||||
| import { Diagnostic } from '@codemirror/lint' | ||||
|  | ||||
| const KclContext = createContext({ | ||||
|   code: codeManager?.code || '', | ||||
|   programMemory: kclManager?.programMemory, | ||||
|   ast: kclManager?.ast, | ||||
|   isExecuting: kclManager?.isExecuting, | ||||
|   errors: kclManager?.kclErrors, | ||||
|   diagnostics: kclManager?.diagnostics, | ||||
|   logs: kclManager?.logs, | ||||
|   wasmInitFailed: kclManager?.wasmInitFailed, | ||||
| }) | ||||
| @ -32,7 +32,7 @@ export function KclContextProvider({ | ||||
|   const [programMemory, setProgramMemory] = useState(kclManager.programMemory) | ||||
|   const [ast, setAst] = useState(kclManager.ast) | ||||
|   const [isExecuting, setIsExecuting] = useState(false) | ||||
|   const [errors, setErrors] = useState<KCLError[]>([]) | ||||
|   const [diagnostics, setErrors] = useState<Diagnostic[]>([]) | ||||
|   const [logs, setLogs] = useState<string[]>([]) | ||||
|   const [wasmInitFailed, setWasmInitFailed] = useState(false) | ||||
|  | ||||
| @ -57,7 +57,7 @@ export function KclContextProvider({ | ||||
|         programMemory, | ||||
|         ast, | ||||
|         isExecuting, | ||||
|         errors, | ||||
|         diagnostics, | ||||
|         logs, | ||||
|         wasmInitFailed, | ||||
|       }} | ||||
|  | ||||
| @ -1,6 +1,10 @@ | ||||
| import { executeAst, lintAst } from 'lang/langHelpers' | ||||
| import { Selections } from 'lib/selections' | ||||
| import { KCLError, kclErrorsToDiagnostics } from './errors' | ||||
| import { | ||||
|   KCLError, | ||||
|   complilationErrorsToDiagnostics, | ||||
|   kclErrorsToDiagnostics, | ||||
| } from './errors' | ||||
| import { uuidv4 } from 'lib/utils' | ||||
| import { EngineCommandManager } from './std/engineConnection' | ||||
| import { err } from 'lib/trap' | ||||
| @ -51,11 +55,11 @@ export class KclManager { | ||||
|   private _programMemory: ProgramMemory = ProgramMemory.empty() | ||||
|   lastSuccessfulProgramMemory: ProgramMemory = ProgramMemory.empty() | ||||
|   private _logs: string[] = [] | ||||
|   private _lints: Diagnostic[] = [] | ||||
|   private _kclErrors: KCLError[] = [] | ||||
|   private _diagnostics: Diagnostic[] = [] | ||||
|   private _isExecuting = false | ||||
|   private _executeIsStale: ExecuteArgs | null = null | ||||
|   private _wasmInitFailed = true | ||||
|   private _hasErrors = false | ||||
|  | ||||
|   engineCommandManager: EngineCommandManager | ||||
|  | ||||
| @ -63,7 +67,7 @@ export class KclManager { | ||||
|   private _astCallBack: (arg: Node<Program>) => void = () => {} | ||||
|   private _programMemoryCallBack: (arg: ProgramMemory) => void = () => {} | ||||
|   private _logsCallBack: (arg: string[]) => void = () => {} | ||||
|   private _kclErrorsCallBack: (arg: KCLError[]) => void = () => {} | ||||
|   private _kclErrorsCallBack: (errors: Diagnostic[]) => void = () => {} | ||||
|   private _wasmInitFailedCallback: (arg: boolean) => void = () => {} | ||||
|   private _executeCallback: () => void = () => {} | ||||
|  | ||||
| @ -84,7 +88,7 @@ export class KclManager { | ||||
|     this._programMemoryCallBack(programMemory) | ||||
|   } | ||||
|  | ||||
|   set execState(execState) { | ||||
|   private set execState(execState) { | ||||
|     this._execState = execState | ||||
|     this.programMemory = execState.memory | ||||
|   } | ||||
| @ -101,38 +105,28 @@ export class KclManager { | ||||
|     this._logsCallBack(logs) | ||||
|   } | ||||
|  | ||||
|   get lints() { | ||||
|     return this._lints | ||||
|   get diagnostics() { | ||||
|     return this._diagnostics | ||||
|   } | ||||
|  | ||||
|   set lints(lints) { | ||||
|     if (lints === this._lints) return | ||||
|     this._lints = lints | ||||
|     // Run the lints through the diagnostics. | ||||
|     this.kclErrors = this._kclErrors | ||||
|   } | ||||
|  | ||||
|   get kclErrors() { | ||||
|     return this._kclErrors | ||||
|   } | ||||
|   set kclErrors(kclErrors) { | ||||
|     if (kclErrors === this._kclErrors && this.lints.length === 0) return | ||||
|     this._kclErrors = kclErrors | ||||
|   set diagnostics(ds) { | ||||
|     if (ds === this._diagnostics) return | ||||
|     this._diagnostics = ds | ||||
|     this.setDiagnosticsForCurrentErrors() | ||||
|     this._kclErrorsCallBack(kclErrors) | ||||
|   } | ||||
|  | ||||
|   addDiagnostics(ds: Diagnostic[]) { | ||||
|     if (ds.length === 0) return | ||||
|     this.diagnostics = this.diagnostics.concat(ds) | ||||
|   } | ||||
|  | ||||
|   hasErrors(): boolean { | ||||
|     return this._hasErrors | ||||
|   } | ||||
|  | ||||
|   setDiagnosticsForCurrentErrors() { | ||||
|     let diagnostics = kclErrorsToDiagnostics(this.kclErrors) | ||||
|     if (this.lints.length > 0) { | ||||
|       diagnostics = diagnostics.concat(this.lints) | ||||
|     } | ||||
|     editorManager?.setDiagnostics(diagnostics) | ||||
|   } | ||||
|  | ||||
|   addKclErrors(kclErrors: KCLError[]) { | ||||
|     if (kclErrors.length === 0) return | ||||
|     this.kclErrors = this.kclErrors.concat(kclErrors) | ||||
|     editorManager?.setDiagnostics(this.diagnostics) | ||||
|     this._kclErrorsCallBack(this.diagnostics) | ||||
|   } | ||||
|  | ||||
|   get isExecuting() { | ||||
| @ -188,7 +182,7 @@ export class KclManager { | ||||
|     setProgramMemory: (arg: ProgramMemory) => void | ||||
|     setAst: (arg: Node<Program>) => void | ||||
|     setLogs: (arg: string[]) => void | ||||
|     setKclErrors: (arg: KCLError[]) => void | ||||
|     setKclErrors: (errors: Diagnostic[]) => void | ||||
|     setIsExecuting: (arg: boolean) => void | ||||
|     setWasmInitFailed: (arg: boolean) => void | ||||
|   }) { | ||||
| @ -218,17 +212,26 @@ export class KclManager { | ||||
|   } | ||||
|  | ||||
|   safeParse(code: string): Node<Program> | null { | ||||
|     const ast = parse(code) | ||||
|     this.lints = [] | ||||
|     this.kclErrors = [] | ||||
|     if (!err(ast)) return ast | ||||
|     const kclerror: KCLError = ast as KCLError | ||||
|     const result = parse(code) | ||||
|     this.diagnostics = [] | ||||
|     this._hasErrors = false | ||||
|  | ||||
|     this.addKclErrors([kclerror]) | ||||
|     // TODO: re-eval if session should end? | ||||
|     if (kclerror.msg === 'file is empty') | ||||
|       this.engineCommandManager?.endSession() | ||||
|     return null | ||||
|     if (err(result)) { | ||||
|       const kclerror: KCLError = result as KCLError | ||||
|       this.diagnostics = kclErrorsToDiagnostics([kclerror]) | ||||
|       this._hasErrors = true | ||||
|       return null | ||||
|     } | ||||
|  | ||||
|     this.addDiagnostics(complilationErrorsToDiagnostics(result.errors)) | ||||
|     this.addDiagnostics(complilationErrorsToDiagnostics(result.warnings)) | ||||
|     if (result.errors.length > 0) { | ||||
|       this._hasErrors = true | ||||
|  | ||||
|       return null | ||||
|     } | ||||
|  | ||||
|     return result.program | ||||
|   } | ||||
|  | ||||
|   async ensureWasmInit() { | ||||
| @ -267,19 +270,16 @@ export class KclManager { | ||||
|     this._cancelTokens.set(currentExecutionId, false) | ||||
|  | ||||
|     this.isExecuting = true | ||||
|     // Make sure we clear before starting again. End session will do this. | ||||
|     this.engineCommandManager?.endSession() | ||||
|     await this.ensureWasmInit() | ||||
|     const { logs, errors, execState, isInterrupted } = await executeAst({ | ||||
|       ast, | ||||
|       idGenerator: this.execState.idGenerator, | ||||
|       engineCommandManager: this.engineCommandManager, | ||||
|     }) | ||||
|  | ||||
|     // Program was not interrupted, setup the scene | ||||
|     // Do not send send scene commands if the program was interrupted, go to clean up | ||||
|     if (!isInterrupted) { | ||||
|       this.lints = await lintAst({ ast: ast }) | ||||
|       this.addDiagnostics(await lintAst({ ast: ast })) | ||||
|  | ||||
|       sceneInfra.modelingSend({ type: 'code edit during sketch' }) | ||||
|       setSelectionFilterToDefault(execState.memory, this.engineCommandManager) | ||||
| @ -321,9 +321,7 @@ export class KclManager { | ||||
|  | ||||
|     this.logs = logs | ||||
|     // Do not add the errors since the program was interrupted and the error is not a real KCL error | ||||
|     this.addKclErrors(isInterrupted ? [] : errors) | ||||
|     // Reset the next ID index so that we reuse the previous IDs next time. | ||||
|     execState.idGenerator.nextId = 0 | ||||
|     this.addDiagnostics(isInterrupted ? [] : kclErrorsToDiagnostics(errors)) | ||||
|     this.execState = execState | ||||
|     if (!errors.length) { | ||||
|       this.lastSuccessfulProgramMemory = execState.memory | ||||
| @ -364,13 +362,13 @@ export class KclManager { | ||||
|  | ||||
|     const { logs, errors, execState } = await executeAst({ | ||||
|       ast: newAst, | ||||
|       idGenerator: this.execState.idGenerator, | ||||
|       engineCommandManager: this.engineCommandManager, | ||||
|       useFakeExecutor: true, | ||||
|       // We make sure to send an empty program memory to denote we mean mock mode. | ||||
|       programMemoryOverride: ProgramMemory.empty(), | ||||
|     }) | ||||
|  | ||||
|     this._logs = logs | ||||
|     this._kclErrors = errors | ||||
|     this.addDiagnostics(kclErrorsToDiagnostics(errors)) | ||||
|     this._execState = execState | ||||
|     this._programMemory = execState.memory | ||||
|     if (!errors.length) { | ||||
| @ -398,7 +396,7 @@ export class KclManager { | ||||
|           ...artifact, | ||||
|           codeRef: { | ||||
|             ...artifact.codeRef, | ||||
|             range: [node.start, node.end], | ||||
|             range: [node.start, node.end, true], | ||||
|           }, | ||||
|         }) | ||||
|       } | ||||
| @ -490,7 +488,7 @@ export class KclManager { | ||||
|         if (start && end) { | ||||
|           returnVal.graphSelections.push({ | ||||
|             codeRef: { | ||||
|               range: [start, end], | ||||
|               range: [start, end, true], | ||||
|               pathToNode: path, | ||||
|             }, | ||||
|           }) | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| import { parse, initPromise } from './wasm' | ||||
| import { assertParse, initPromise } from './wasm' | ||||
| import { enginelessExecutor } from '../lib/testHelpers' | ||||
|  | ||||
| beforeAll(async () => { | ||||
| @ -14,7 +14,7 @@ const mySketch001 = startSketchOn('XY') | ||||
|   |> lineTo([-1.59, -1.54], %) | ||||
|   |> lineTo([0.46, -5.82], %) | ||||
|   // |> rx(45, %)` | ||||
|     const execState = await enginelessExecutor(parse(code)) | ||||
|     const execState = await enginelessExecutor(assertParse(code)) | ||||
|     // @ts-ignore | ||||
|     const sketch001 = execState.memory.get('mySketch001') | ||||
|     expect(sketch001).toEqual({ | ||||
| @ -67,7 +67,7 @@ const mySketch001 = startSketchOn('XY') | ||||
|   |> lineTo([0.46, -5.82], %) | ||||
|   // |> rx(45, %) | ||||
|   |> extrude(2, %)` | ||||
|     const execState = await enginelessExecutor(parse(code)) | ||||
|     const execState = await enginelessExecutor(assertParse(code)) | ||||
|     // @ts-ignore | ||||
|     const sketch001 = execState.memory.get('mySketch001') | ||||
|     expect(sketch001).toEqual({ | ||||
| @ -147,7 +147,7 @@ const sk2 = startSketchOn('XY') | ||||
|   |> extrude(2, %) | ||||
|  | ||||
| ` | ||||
|     const execState = await enginelessExecutor(parse(code)) | ||||
|     const execState = await enginelessExecutor(assertParse(code)) | ||||
|     const programMemory = execState.memory | ||||
|     // @ts-ignore | ||||
|     const geos = [programMemory.get('theExtrude'), programMemory.get('sk2')] | ||||
|  | ||||
| @ -8,20 +8,14 @@ describe('test kclErrToDiagnostic', () => { | ||||
|         message: '', | ||||
|         kind: 'semantic', | ||||
|         msg: 'Semantic error', | ||||
|         sourceRanges: [ | ||||
|           [0, 1, 0], | ||||
|           [2, 3, 0], | ||||
|         ], | ||||
|         sourceRange: [0, 1, true], | ||||
|       }, | ||||
|       { | ||||
|         name: '', | ||||
|         message: '', | ||||
|         kind: 'type', | ||||
|         msg: 'Type error', | ||||
|         sourceRanges: [ | ||||
|           [4, 5, 0], | ||||
|           [6, 7, 0], | ||||
|         ], | ||||
|         sourceRange: [4, 5, true], | ||||
|       }, | ||||
|     ] | ||||
|     const diagnostics = kclErrorsToDiagnostics(errors) | ||||
| @ -32,24 +26,12 @@ describe('test kclErrToDiagnostic', () => { | ||||
|         message: 'Semantic error', | ||||
|         severity: 'error', | ||||
|       }, | ||||
|       { | ||||
|         from: 2, | ||||
|         to: 3, | ||||
|         message: 'Semantic error', | ||||
|         severity: 'error', | ||||
|       }, | ||||
|       { | ||||
|         from: 4, | ||||
|         to: 5, | ||||
|         message: 'Type error', | ||||
|         severity: 'error', | ||||
|       }, | ||||
|       { | ||||
|         from: 6, | ||||
|         to: 7, | ||||
|         message: 'Type error', | ||||
|         severity: 'error', | ||||
|       }, | ||||
|     ]) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| @ -1,88 +1,90 @@ | ||||
| import { KclError as RustKclError } from '../wasm-lib/kcl/bindings/KclError' | ||||
| import { CompilationError } from 'wasm-lib/kcl/bindings/CompilationError' | ||||
| import { Diagnostic as CodeMirrorDiagnostic } from '@codemirror/lint' | ||||
| import { posToOffset } from '@kittycad/codemirror-lsp-client' | ||||
| import { Diagnostic as LspDiagnostic } from 'vscode-languageserver-protocol' | ||||
| import { Text } from '@codemirror/state' | ||||
|  | ||||
| const TOP_LEVEL_MODULE_ID = 0 | ||||
| import { EditorView } from 'codemirror' | ||||
| import { SourceRange } from 'lang/wasm' | ||||
|  | ||||
| type ExtractKind<T> = T extends { kind: infer K } ? K : never | ||||
| export class KCLError extends Error { | ||||
|   kind: ExtractKind<RustKclError> | 'name' | ||||
|   sourceRanges: [number, number, number][] | ||||
|   sourceRange: SourceRange | ||||
|   msg: string | ||||
|  | ||||
|   constructor( | ||||
|     kind: ExtractKind<RustKclError> | 'name', | ||||
|     msg: string, | ||||
|     sourceRanges: [number, number, number][] | ||||
|     sourceRange: SourceRange | ||||
|   ) { | ||||
|     super() | ||||
|     this.kind = kind | ||||
|     this.msg = msg | ||||
|     this.sourceRanges = sourceRanges | ||||
|     this.sourceRange = sourceRange | ||||
|     Object.setPrototypeOf(this, KCLError.prototype) | ||||
|   } | ||||
| } | ||||
|  | ||||
| export class KCLLexicalError extends KCLError { | ||||
|   constructor(msg: string, sourceRanges: [number, number, number][]) { | ||||
|     super('lexical', msg, sourceRanges) | ||||
|   constructor(msg: string, sourceRange: SourceRange) { | ||||
|     super('lexical', msg, sourceRange) | ||||
|     Object.setPrototypeOf(this, KCLSyntaxError.prototype) | ||||
|   } | ||||
| } | ||||
|  | ||||
| export class KCLInternalError extends KCLError { | ||||
|   constructor(msg: string, sourceRanges: [number, number, number][]) { | ||||
|     super('internal', msg, sourceRanges) | ||||
|   constructor(msg: string, sourceRange: SourceRange) { | ||||
|     super('internal', msg, sourceRange) | ||||
|     Object.setPrototypeOf(this, KCLSyntaxError.prototype) | ||||
|   } | ||||
| } | ||||
|  | ||||
| export class KCLSyntaxError extends KCLError { | ||||
|   constructor(msg: string, sourceRanges: [number, number, number][]) { | ||||
|     super('syntax', msg, sourceRanges) | ||||
|   constructor(msg: string, sourceRange: SourceRange) { | ||||
|     super('syntax', msg, sourceRange) | ||||
|     Object.setPrototypeOf(this, KCLSyntaxError.prototype) | ||||
|   } | ||||
| } | ||||
|  | ||||
| export class KCLSemanticError extends KCLError { | ||||
|   constructor(msg: string, sourceRanges: [number, number, number][]) { | ||||
|     super('semantic', msg, sourceRanges) | ||||
|   constructor(msg: string, sourceRange: SourceRange) { | ||||
|     super('semantic', msg, sourceRange) | ||||
|     Object.setPrototypeOf(this, KCLSemanticError.prototype) | ||||
|   } | ||||
| } | ||||
|  | ||||
| export class KCLTypeError extends KCLError { | ||||
|   constructor(msg: string, sourceRanges: [number, number, number][]) { | ||||
|     super('type', msg, sourceRanges) | ||||
|   constructor(msg: string, sourceRange: SourceRange) { | ||||
|     super('type', msg, sourceRange) | ||||
|     Object.setPrototypeOf(this, KCLTypeError.prototype) | ||||
|   } | ||||
| } | ||||
|  | ||||
| export class KCLUnimplementedError extends KCLError { | ||||
|   constructor(msg: string, sourceRanges: [number, number, number][]) { | ||||
|     super('unimplemented', msg, sourceRanges) | ||||
|   constructor(msg: string, sourceRange: SourceRange) { | ||||
|     super('unimplemented', msg, sourceRange) | ||||
|     Object.setPrototypeOf(this, KCLUnimplementedError.prototype) | ||||
|   } | ||||
| } | ||||
|  | ||||
| export class KCLUnexpectedError extends KCLError { | ||||
|   constructor(msg: string, sourceRanges: [number, number, number][]) { | ||||
|     super('unexpected', msg, sourceRanges) | ||||
|   constructor(msg: string, sourceRange: SourceRange) { | ||||
|     super('unexpected', msg, sourceRange) | ||||
|     Object.setPrototypeOf(this, KCLUnexpectedError.prototype) | ||||
|   } | ||||
| } | ||||
|  | ||||
| export class KCLValueAlreadyDefined extends KCLError { | ||||
|   constructor(key: string, sourceRanges: [number, number, number][]) { | ||||
|     super('name', `Key ${key} was already defined elsewhere`, sourceRanges) | ||||
|   constructor(key: string, sourceRange: SourceRange) { | ||||
|     super('name', `Key ${key} was already defined elsewhere`, sourceRange) | ||||
|     Object.setPrototypeOf(this, KCLValueAlreadyDefined.prototype) | ||||
|   } | ||||
| } | ||||
|  | ||||
| export class KCLUndefinedValueError extends KCLError { | ||||
|   constructor(key: string, sourceRanges: [number, number, number][]) { | ||||
|     super('name', `Key ${key} has not been defined`, sourceRanges) | ||||
|   constructor(key: string, sourceRange: SourceRange) { | ||||
|     super('name', `Key ${key} has not been defined`, sourceRange) | ||||
|     Object.setPrototypeOf(this, KCLUndefinedValueError.prototype) | ||||
|   } | ||||
| } | ||||
| @ -99,27 +101,14 @@ export function lspDiagnosticsToKclErrors( | ||||
|     .flatMap( | ||||
|       ({ range, message }) => | ||||
|         new KCLError('unexpected', message, [ | ||||
|           [ | ||||
|             posToOffset(doc, range.start)!, | ||||
|             posToOffset(doc, range.end)!, | ||||
|             TOP_LEVEL_MODULE_ID, | ||||
|           ], | ||||
|           posToOffset(doc, range.start)!, | ||||
|           posToOffset(doc, range.end)!, | ||||
|           true, | ||||
|         ]) | ||||
|     ) | ||||
|     .filter(({ sourceRanges }) => { | ||||
|       const [from, to, moduleId] = sourceRanges[0] | ||||
|       return ( | ||||
|         from !== null && | ||||
|         to !== null && | ||||
|         from !== undefined && | ||||
|         to !== undefined && | ||||
|         // Filter out errors that are not from the top-level module. | ||||
|         moduleId === TOP_LEVEL_MODULE_ID | ||||
|       ) | ||||
|     }) | ||||
|     .sort((a, b) => { | ||||
|       const c = a.sourceRanges[0][0] | ||||
|       const d = b.sourceRanges[0][0] | ||||
|       const c = a.sourceRange[0] | ||||
|       const d = b.sourceRange[0] | ||||
|       switch (true) { | ||||
|         case c < d: | ||||
|           return -1 | ||||
| @ -137,17 +126,48 @@ export function lspDiagnosticsToKclErrors( | ||||
| export function kclErrorsToDiagnostics( | ||||
|   errors: KCLError[] | ||||
| ): CodeMirrorDiagnostic[] { | ||||
|   return errors?.flatMap((err) => { | ||||
|     const sourceRanges: CodeMirrorDiagnostic[] = err.sourceRanges | ||||
|       // Filter out errors that are not from the top-level module. | ||||
|       .filter(([_start, _end, moduleId]) => moduleId === TOP_LEVEL_MODULE_ID) | ||||
|       .map(([from, to]) => { | ||||
|         return { from, to, message: err.msg, severity: 'error' } | ||||
|       }) | ||||
|     // Make sure we didn't filter out all the source ranges. | ||||
|     if (sourceRanges.length === 0) { | ||||
|       sourceRanges.push({ from: 0, to: 0, message: err.msg, severity: 'error' }) | ||||
|     } | ||||
|     return sourceRanges | ||||
|   }) | ||||
|   return errors | ||||
|     ?.filter((err) => err.sourceRange[2]) | ||||
|     .map((err) => { | ||||
|       return { | ||||
|         from: err.sourceRange[0], | ||||
|         to: err.sourceRange[1], | ||||
|         message: err.msg, | ||||
|         severity: 'error', | ||||
|       } | ||||
|     }) | ||||
| } | ||||
|  | ||||
| export function complilationErrorsToDiagnostics( | ||||
|   errors: CompilationError[] | ||||
| ): CodeMirrorDiagnostic[] { | ||||
|   return errors | ||||
|     ?.filter((err) => err.sourceRange[2] === 0) | ||||
|     .map((err) => { | ||||
|       let severity: any = 'error' | ||||
|       if (err.severity === 'Warning') { | ||||
|         severity = 'warning' | ||||
|       } | ||||
|       let actions | ||||
|       const suggestion = err.suggestion | ||||
|       if (suggestion) { | ||||
|         actions = [ | ||||
|           { | ||||
|             name: suggestion.title, | ||||
|             apply: (view: EditorView, from: number, to: number) => { | ||||
|               view.dispatch({ | ||||
|                 changes: { from, to, insert: suggestion.insert }, | ||||
|               }) | ||||
|             }, | ||||
|           }, | ||||
|         ] | ||||
|       } | ||||
|       return { | ||||
|         from: err.sourceRange[0], | ||||
|         to: err.sourceRange[1], | ||||
|         message: err.message, | ||||
|         severity, | ||||
|         actions, | ||||
|       } | ||||
|     }) | ||||
| } | ||||
|  | ||||
| @ -1,7 +1,7 @@ | ||||
| import fs from 'node:fs' | ||||
|  | ||||
| import { | ||||
|   parse, | ||||
|   assertParse, | ||||
|   ProgramMemory, | ||||
|   Sketch, | ||||
|   initPromise, | ||||
| @ -472,7 +472,7 @@ describe('Testing Errors', () => { | ||||
| const theExtrude = startSketchOn('XY') | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line([-2.4, 5], %) | ||||
|   |> line([-0.76], myVarZ, %) | ||||
|   |> line(myVarZ, %) | ||||
|   |> line([5,5], %) | ||||
|   |> close(%) | ||||
|   |> extrude(4, %)` | ||||
| @ -480,7 +480,7 @@ const theExtrude = startSketchOn('XY') | ||||
|       new KCLError( | ||||
|         'undefined_value', | ||||
|         'memory item key `myVarZ` is not defined', | ||||
|         [[129, 135, 0]] | ||||
|         [129, 135, true] | ||||
|       ) | ||||
|     ) | ||||
|   }) | ||||
| @ -492,7 +492,7 @@ async function exe( | ||||
|   code: string, | ||||
|   programMemory: ProgramMemory = ProgramMemory.empty() | ||||
| ) { | ||||
|   const ast = parse(code) | ||||
|   const ast = assertParse(code) | ||||
|  | ||||
|   const execState = await enginelessExecutor(ast, programMemory) | ||||
|   return execState.memory | ||||
|  | ||||
| @ -1,5 +1,5 @@ | ||||
| import { getNodePathFromSourceRange, getNodeFromPath } from './queryAst' | ||||
| import { Identifier, parse, initPromise, Parameter } from './wasm' | ||||
| import { Identifier, assertParse, initPromise, Parameter } from './wasm' | ||||
| import { err } from 'lib/trap' | ||||
|  | ||||
| beforeAll(async () => { | ||||
| @ -17,19 +17,19 @@ const sk3 = startSketchAt([0, 0]) | ||||
| ` | ||||
|     const subStr = 'lineTo([3, 4], %, $yo)' | ||||
|     const lineToSubstringIndex = code.indexOf(subStr) | ||||
|     const sourceRange: [number, number] = [ | ||||
|     const sourceRange: [number, number, boolean] = [ | ||||
|       lineToSubstringIndex, | ||||
|       lineToSubstringIndex + subStr.length, | ||||
|       true, | ||||
|     ] | ||||
|  | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const nodePath = getNodePathFromSourceRange(ast, sourceRange) | ||||
|     const _node = getNodeFromPath<any>(ast, nodePath) | ||||
|     if (err(_node)) throw _node | ||||
|     const { node } = _node | ||||
|  | ||||
|     expect([node.start, node.end]).toEqual(sourceRange) | ||||
|     expect([node.start, node.end, true]).toEqual(sourceRange) | ||||
|     expect(node.type).toBe('CallExpression') | ||||
|   }) | ||||
|   it('gets path right for function definition params', () => { | ||||
| @ -45,13 +45,13 @@ const sk3 = startSketchAt([0, 0]) | ||||
| const b1 = cube([0,0], 10)` | ||||
|     const subStr = 'pos, scale' | ||||
|     const subStrIndex = code.indexOf(subStr) | ||||
|     const sourceRange: [number, number] = [ | ||||
|     const sourceRange: [number, number, boolean] = [ | ||||
|       subStrIndex, | ||||
|       subStrIndex + 'pos'.length, | ||||
|       true, | ||||
|     ] | ||||
|  | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const nodePath = getNodePathFromSourceRange(ast, sourceRange) | ||||
|     const _node = getNodeFromPath<Parameter>(ast, nodePath) | ||||
|     if (err(_node)) throw _node | ||||
| @ -82,13 +82,13 @@ const b1 = cube([0,0], 10)` | ||||
| const b1 = cube([0,0], 10)` | ||||
|     const subStr = 'scale, 0' | ||||
|     const subStrIndex = code.indexOf(subStr) | ||||
|     const sourceRange: [number, number] = [ | ||||
|     const sourceRange: [number, number, boolean] = [ | ||||
|       subStrIndex, | ||||
|       subStrIndex + 'scale'.length, | ||||
|       true, | ||||
|     ] | ||||
|  | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const nodePath = getNodePathFromSourceRange(ast, sourceRange) | ||||
|     const _node = getNodeFromPath<Identifier>(ast, nodePath) | ||||
|     if (err(_node)) throw _node | ||||
|  | ||||
| @ -1,6 +1,5 @@ | ||||
| import { parse, initPromise, programMemoryInit } from './wasm' | ||||
| import { assertParse, initPromise, programMemoryInit } from './wasm' | ||||
| import { enginelessExecutor } from '../lib/testHelpers' | ||||
| import { assert } from 'vitest' | ||||
| // These unit tests makes web requests to a public github repository. | ||||
|  | ||||
| interface KclSampleFile { | ||||
| @ -58,8 +57,7 @@ describe('Test KCL Samples from public Github repository', () => { | ||||
|     files.forEach((file: KclSampleFile) => { | ||||
|       it(`should parse ${file.filename} without errors`, async () => { | ||||
|         const code = await getKclSampleCodeFromGithub(file.filename) | ||||
|         const parsed = parse(code) | ||||
|         assert(!(parsed instanceof Error)) | ||||
|         assertParse(code) | ||||
|       }, 1000) | ||||
|     }) | ||||
|   }) | ||||
| @ -71,9 +69,8 @@ describe('Test KCL Samples from public Github repository', () => { | ||||
|         for (let i = 0; i < files.length; i++) { | ||||
|           const file: KclSampleFile = files[i] | ||||
|           const code = await getKclSampleCodeFromGithub(file.filename) | ||||
|           const parsed = parse(code) | ||||
|           assert(!(parsed instanceof Error)) | ||||
|           await enginelessExecutor(parsed, programMemoryInit()) | ||||
|           const ast = assertParse(code) | ||||
|           await enginelessExecutor(ast, programMemoryInit()) | ||||
|         } | ||||
|       }, | ||||
|       files.length * 1000 | ||||
|  | ||||
| @ -2,7 +2,6 @@ import { | ||||
|   Program, | ||||
|   _executor, | ||||
|   ProgramMemory, | ||||
|   programMemoryInit, | ||||
|   kclLint, | ||||
|   emptyExecState, | ||||
|   ExecState, | ||||
| @ -11,7 +10,6 @@ import { enginelessExecutor } from 'lib/testHelpers' | ||||
| import { EngineCommandManager } from 'lang/std/engineConnection' | ||||
| import { KCLError } from 'lang/errors' | ||||
| import { Diagnostic } from '@codemirror/lint' | ||||
| import { IdGenerator } from 'wasm-lib/kcl/bindings/IdGenerator' | ||||
| import { Node } from 'wasm-lib/kcl/bindings/Node' | ||||
|  | ||||
| export type ToolTip = | ||||
| @ -49,15 +47,13 @@ export const toolTips: Array<ToolTip> = [ | ||||
| export async function executeAst({ | ||||
|   ast, | ||||
|   engineCommandManager, | ||||
|   useFakeExecutor = false, | ||||
|   // If you set programMemoryOverride we assume you mean mock mode. Since that | ||||
|   // is the only way to go about it. | ||||
|   programMemoryOverride, | ||||
|   idGenerator, | ||||
| }: { | ||||
|   ast: Node<Program> | ||||
|   engineCommandManager: EngineCommandManager | ||||
|   useFakeExecutor?: boolean | ||||
|   programMemoryOverride?: ProgramMemory | ||||
|   idGenerator?: IdGenerator | ||||
|   isInterrupted?: boolean | ||||
| }): Promise<{ | ||||
|   logs: string[] | ||||
| @ -66,22 +62,14 @@ export async function executeAst({ | ||||
|   isInterrupted: boolean | ||||
| }> { | ||||
|   try { | ||||
|     if (!useFakeExecutor) { | ||||
|       engineCommandManager.endSession() | ||||
|       // eslint-disable-next-line @typescript-eslint/no-floating-promises | ||||
|       engineCommandManager.startNewSession() | ||||
|     } | ||||
|     const execState = await (useFakeExecutor | ||||
|       ? enginelessExecutor(ast, programMemoryOverride || programMemoryInit()) | ||||
|       : _executor( | ||||
|           ast, | ||||
|           programMemoryInit(), | ||||
|           idGenerator, | ||||
|           engineCommandManager, | ||||
|           false | ||||
|         )) | ||||
|     const execState = await (programMemoryOverride | ||||
|       ? enginelessExecutor(ast, programMemoryOverride) | ||||
|       : _executor(ast, engineCommandManager)) | ||||
|  | ||||
|     await engineCommandManager.waitForAllCommands( | ||||
|       programMemoryOverride !== undefined | ||||
|     ) | ||||
|  | ||||
|     await engineCommandManager.waitForAllCommands(useFakeExecutor) | ||||
|     return { | ||||
|       logs: [], | ||||
|       errors: [], | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| import { parse, recast, initPromise, Identifier } from './wasm' | ||||
| import { assertParse, recast, initPromise, Identifier } from './wasm' | ||||
| import { | ||||
|   createLiteral, | ||||
|   createIdentifier, | ||||
| @ -146,10 +146,13 @@ function giveSketchFnCallTagTestHelper( | ||||
|   // giveSketchFnCallTag inputs and outputs an ast, which is very verbose for testing | ||||
|   // this wrapper changes the input and output to code | ||||
|   // making it more of an integration test, but easier to read the test intention is the goal | ||||
|   const ast = parse(code) | ||||
|   if (err(ast)) throw ast | ||||
|   const ast = assertParse(code) | ||||
|   const start = code.indexOf(searchStr) | ||||
|   const range: [number, number] = [start, start + searchStr.length] | ||||
|   const range: [number, number, boolean] = [ | ||||
|     start, | ||||
|     start + searchStr.length, | ||||
|     true, | ||||
|   ] | ||||
|   const sketchRes = giveSketchFnCallTag(ast, range) | ||||
|   if (err(sketchRes)) throw sketchRes | ||||
|   const { modifiedAst, tag, isTagExisting } = sketchRes | ||||
| @ -221,14 +224,13 @@ part001 = startSketchOn('XY') | ||||
| |> angledLine([jkl(yo) + 2, 3.09], %) | ||||
| yo2 = hmm([identifierGuy + 5])` | ||||
|   it('should move a binary expression into a new variable', async () => { | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const execState = await enginelessExecutor(ast) | ||||
|     const startIndex = code.indexOf('100 + 100') + 1 | ||||
|     const { modifiedAst } = moveValueIntoNewVariable( | ||||
|       ast, | ||||
|       execState.memory, | ||||
|       [startIndex, startIndex], | ||||
|       [startIndex, startIndex, true], | ||||
|       'newVar' | ||||
|     ) | ||||
|     const newCode = recast(modifiedAst) | ||||
| @ -236,14 +238,13 @@ yo2 = hmm([identifierGuy + 5])` | ||||
|     expect(newCode).toContain(`angledLine([newVar, 3.09], %)`) | ||||
|   }) | ||||
|   it('should move a value into a new variable', async () => { | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const execState = await enginelessExecutor(ast) | ||||
|     const startIndex = code.indexOf('2.8') + 1 | ||||
|     const { modifiedAst } = moveValueIntoNewVariable( | ||||
|       ast, | ||||
|       execState.memory, | ||||
|       [startIndex, startIndex], | ||||
|       [startIndex, startIndex, true], | ||||
|       'newVar' | ||||
|     ) | ||||
|     const newCode = recast(modifiedAst) | ||||
| @ -251,14 +252,13 @@ yo2 = hmm([identifierGuy + 5])` | ||||
|     expect(newCode).toContain(`line([newVar, 0], %)`) | ||||
|   }) | ||||
|   it('should move a callExpression into a new variable', async () => { | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const execState = await enginelessExecutor(ast) | ||||
|     const startIndex = code.indexOf('def(') | ||||
|     const { modifiedAst } = moveValueIntoNewVariable( | ||||
|       ast, | ||||
|       execState.memory, | ||||
|       [startIndex, startIndex], | ||||
|       [startIndex, startIndex, true], | ||||
|       'newVar' | ||||
|     ) | ||||
|     const newCode = recast(modifiedAst) | ||||
| @ -266,14 +266,13 @@ yo2 = hmm([identifierGuy + 5])` | ||||
|     expect(newCode).toContain(`angledLine([newVar, 3.09], %)`) | ||||
|   }) | ||||
|   it('should move a binary expression with call expression into a new variable', async () => { | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const execState = await enginelessExecutor(ast) | ||||
|     const startIndex = code.indexOf('jkl(') + 1 | ||||
|     const { modifiedAst } = moveValueIntoNewVariable( | ||||
|       ast, | ||||
|       execState.memory, | ||||
|       [startIndex, startIndex], | ||||
|       [startIndex, startIndex, true], | ||||
|       'newVar' | ||||
|     ) | ||||
|     const newCode = recast(modifiedAst) | ||||
| @ -281,14 +280,13 @@ yo2 = hmm([identifierGuy + 5])` | ||||
|     expect(newCode).toContain(`angledLine([newVar, 3.09], %)`) | ||||
|   }) | ||||
|   it('should move a identifier into a new variable', async () => { | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const execState = await enginelessExecutor(ast) | ||||
|     const startIndex = code.indexOf('identifierGuy +') + 1 | ||||
|     const { modifiedAst } = moveValueIntoNewVariable( | ||||
|       ast, | ||||
|       execState.memory, | ||||
|       [startIndex, startIndex], | ||||
|       [startIndex, startIndex, true], | ||||
|       'newVar' | ||||
|     ) | ||||
|     const newCode = recast(modifiedAst) | ||||
| @ -305,19 +303,20 @@ describe('testing sketchOnExtrudedFace', () => { | ||||
|   |> line([8.62, -9.57], %) | ||||
|   |> close(%) | ||||
|   |> extrude(5 + 7, %)` | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|  | ||||
|     const segmentSnippet = `line([9.7, 9.19], %)` | ||||
|     const segmentRange: [number, number] = [ | ||||
|     const segmentRange: [number, number, boolean] = [ | ||||
|       code.indexOf(segmentSnippet), | ||||
|       code.indexOf(segmentSnippet) + segmentSnippet.length, | ||||
|       true, | ||||
|     ] | ||||
|     const segmentPathToNode = getNodePathFromSourceRange(ast, segmentRange) | ||||
|     const extrudeSnippet = `extrude(5 + 7, %)` | ||||
|     const extrudeRange: [number, number] = [ | ||||
|     const extrudeRange: [number, number, boolean] = [ | ||||
|       code.indexOf(extrudeSnippet), | ||||
|       code.indexOf(extrudeSnippet) + extrudeSnippet.length, | ||||
|       true, | ||||
|     ] | ||||
|     const extrudePathToNode = getNodePathFromSourceRange(ast, extrudeRange) | ||||
|  | ||||
| @ -345,18 +344,19 @@ sketch001 = startSketchOn(part001, seg01)`) | ||||
|   |> line([8.62, -9.57], %) | ||||
|   |> close(%) | ||||
|   |> extrude(5 + 7, %)` | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const segmentSnippet = `close(%)` | ||||
|     const segmentRange: [number, number] = [ | ||||
|     const segmentRange: [number, number, boolean] = [ | ||||
|       code.indexOf(segmentSnippet), | ||||
|       code.indexOf(segmentSnippet) + segmentSnippet.length, | ||||
|       true, | ||||
|     ] | ||||
|     const segmentPathToNode = getNodePathFromSourceRange(ast, segmentRange) | ||||
|     const extrudeSnippet = `extrude(5 + 7, %)` | ||||
|     const extrudeRange: [number, number] = [ | ||||
|     const extrudeRange: [number, number, boolean] = [ | ||||
|       code.indexOf(extrudeSnippet), | ||||
|       code.indexOf(extrudeSnippet) + extrudeSnippet.length, | ||||
|       true, | ||||
|     ] | ||||
|     const extrudePathToNode = getNodePathFromSourceRange(ast, extrudeRange) | ||||
|  | ||||
| @ -384,18 +384,19 @@ sketch001 = startSketchOn(part001, seg01)`) | ||||
|   |> line([8.62, -9.57], %) | ||||
|   |> close(%) | ||||
|   |> extrude(5 + 7, %)` | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const sketchSnippet = `startProfileAt([3.58, 2.06], %)` | ||||
|     const sketchRange: [number, number] = [ | ||||
|     const sketchRange: [number, number, boolean] = [ | ||||
|       code.indexOf(sketchSnippet), | ||||
|       code.indexOf(sketchSnippet) + sketchSnippet.length, | ||||
|       true, | ||||
|     ] | ||||
|     const sketchPathToNode = getNodePathFromSourceRange(ast, sketchRange) | ||||
|     const extrudeSnippet = `extrude(5 + 7, %)` | ||||
|     const extrudeRange: [number, number] = [ | ||||
|     const extrudeRange: [number, number, boolean] = [ | ||||
|       code.indexOf(extrudeSnippet), | ||||
|       code.indexOf(extrudeSnippet) + extrudeSnippet.length, | ||||
|       true, | ||||
|     ] | ||||
|     const extrudePathToNode = getNodePathFromSourceRange(ast, extrudeRange) | ||||
|  | ||||
| @ -432,18 +433,19 @@ sketch001 = startSketchOn(part001, 'END')`) | ||||
|     |> line([-17.67, 0.85], %) | ||||
|     |> close(%) | ||||
|     part001 = extrude(5 + 7, sketch001)` | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const segmentSnippet = `line([4.99, -0.46], %)` | ||||
|     const segmentRange: [number, number] = [ | ||||
|     const segmentRange: [number, number, boolean] = [ | ||||
|       code.indexOf(segmentSnippet), | ||||
|       code.indexOf(segmentSnippet) + segmentSnippet.length, | ||||
|       true, | ||||
|     ] | ||||
|     const segmentPathToNode = getNodePathFromSourceRange(ast, segmentRange) | ||||
|     const extrudeSnippet = `extrude(5 + 7, sketch001)` | ||||
|     const extrudeRange: [number, number] = [ | ||||
|     const extrudeRange: [number, number, boolean] = [ | ||||
|       code.indexOf(extrudeSnippet), | ||||
|       code.indexOf(extrudeSnippet) + extrudeSnippet.length, | ||||
|       true, | ||||
|     ] | ||||
|     const extrudePathToNode = getNodePathFromSourceRange(ast, extrudeRange) | ||||
|  | ||||
| @ -466,13 +468,13 @@ describe('Testing deleteSegmentFromPipeExpression', () => { | ||||
|   |> line([306.21, 198.82], %) | ||||
|   |> line([306.21, 198.85], %, $a) | ||||
|   |> line([306.21, 198.87], %)` | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const execState = await enginelessExecutor(ast) | ||||
|     const lineOfInterest = 'line([306.21, 198.85], %, $a)' | ||||
|     const range: [number, number] = [ | ||||
|     const range: [number, number, boolean] = [ | ||||
|       code.indexOf(lineOfInterest), | ||||
|       code.indexOf(lineOfInterest) + lineOfInterest.length, | ||||
|       true, | ||||
|     ] | ||||
|     const pathToNode = getNodePathFromSourceRange(ast, range) | ||||
|     const modifiedAst = deleteSegmentFromPipeExpression( | ||||
| @ -544,13 +546,13 @@ ${!replace1 ? `  |> ${line}\n` : ''}  |> angledLine([-65, ${ | ||||
|       ], | ||||
|     ])(`%s`, async (_, line, [replace1, replace2]) => { | ||||
|       const code = makeCode(line) | ||||
|       const ast = parse(code) | ||||
|       if (err(ast)) throw ast | ||||
|       const ast = assertParse(code) | ||||
|       const execState = await enginelessExecutor(ast) | ||||
|       const lineOfInterest = line | ||||
|       const range: [number, number] = [ | ||||
|       const range: [number, number, boolean] = [ | ||||
|         code.indexOf(lineOfInterest), | ||||
|         code.indexOf(lineOfInterest) + lineOfInterest.length, | ||||
|         true, | ||||
|       ] | ||||
|       const pathToNode = getNodePathFromSourceRange(ast, range) | ||||
|       const dependentSegments = findUsesOfTagInPipe(ast, pathToNode) | ||||
| @ -632,14 +634,14 @@ describe('Testing removeSingleConstraintInfo', () => { | ||||
|       ], | ||||
|       ['tangentialArcTo([3.14 + 0, 13.14], %)', 'arrayIndex', 1], | ||||
|     ] as const)('stdlib fn: %s', async (expectedFinish, key, value) => { | ||||
|       const ast = parse(code) | ||||
|       if (err(ast)) throw ast | ||||
|       const ast = assertParse(code) | ||||
|  | ||||
|       const execState = await enginelessExecutor(ast) | ||||
|       const lineOfInterest = expectedFinish.split('(')[0] + '(' | ||||
|       const range: [number, number] = [ | ||||
|       const range: [number, number, boolean] = [ | ||||
|         code.indexOf(lineOfInterest) + 1, | ||||
|         code.indexOf(lineOfInterest) + lineOfInterest.length, | ||||
|         true, | ||||
|       ] | ||||
|       const pathToNode = getNodePathFromSourceRange(ast, range) | ||||
|       let argPosition: SimplifiedArgDetails | ||||
| @ -686,14 +688,14 @@ describe('Testing removeSingleConstraintInfo', () => { | ||||
|       ['angledLineToX([12.14 + 0, 12], %)', 'arrayIndex', 1], | ||||
|       ['angledLineToY([30, 10.14 + 0], %)', 'arrayIndex', 0], | ||||
|     ])('stdlib fn: %s', async (expectedFinish, key, value) => { | ||||
|       const ast = parse(code) | ||||
|       if (err(ast)) throw ast | ||||
|       const ast = assertParse(code) | ||||
|  | ||||
|       const execState = await enginelessExecutor(ast) | ||||
|       const lineOfInterest = expectedFinish.split('(')[0] + '(' | ||||
|       const range: [number, number] = [ | ||||
|       const range: [number, number, boolean] = [ | ||||
|         code.indexOf(lineOfInterest) + 1, | ||||
|         code.indexOf(lineOfInterest) + lineOfInterest.length, | ||||
|         true, | ||||
|       ] | ||||
|       let argPosition: SimplifiedArgDetails | ||||
|       if (key === 'arrayIndex' && typeof value === 'number') { | ||||
| @ -883,14 +885,14 @@ sketch002 = startSketchOn({ | ||||
|     '%s', | ||||
|     async (name, { codeBefore, codeAfter, lineOfInterest, type }) => { | ||||
|       // const lineOfInterest = 'line([-2.94, 2.7], %)' | ||||
|       const ast = parse(codeBefore) | ||||
|       if (err(ast)) throw ast | ||||
|       const ast = assertParse(codeBefore) | ||||
|       const execState = await enginelessExecutor(ast) | ||||
|  | ||||
|       // deleteFromSelection | ||||
|       const range: [number, number] = [ | ||||
|       const range: [number, number, boolean] = [ | ||||
|         codeBefore.indexOf(lineOfInterest), | ||||
|         codeBefore.indexOf(lineOfInterest) + lineOfInterest.length, | ||||
|         true, | ||||
|       ] | ||||
|       const artifact = { type } as Artifact | ||||
|       const newAst = await deleteFromSelection( | ||||
|  | ||||
| @ -346,6 +346,37 @@ export function extrudeSketch( | ||||
|   } | ||||
| } | ||||
|  | ||||
| export function loftSketches( | ||||
|   node: Node<Program>, | ||||
|   declarators: VariableDeclarator[] | ||||
| ): { | ||||
|   modifiedAst: Node<Program> | ||||
|   pathToNode: PathToNode | ||||
| } { | ||||
|   const modifiedAst = structuredClone(node) | ||||
|   const name = findUniqueName(node, KCL_DEFAULT_CONSTANT_PREFIXES.LOFT) | ||||
|   const elements = declarators.map((d) => createIdentifier(d.id.name)) | ||||
|   const loft = createCallExpressionStdLib('loft', [ | ||||
|     createArrayExpression(elements), | ||||
|   ]) | ||||
|   const declaration = createVariableDeclaration(name, loft) | ||||
|   modifiedAst.body.push(declaration) | ||||
|   const pathToNode: PathToNode = [ | ||||
|     ['body', ''], | ||||
|     [modifiedAst.body.length - 1, 'index'], | ||||
|     ['declarations', 'VariableDeclaration'], | ||||
|     ['0', 'index'], | ||||
|     ['init', 'VariableDeclarator'], | ||||
|     ['arguments', 'CallExpression'], | ||||
|     [0, 'index'], | ||||
|   ] | ||||
|  | ||||
|   return { | ||||
|     modifiedAst, | ||||
|     pathToNode, | ||||
|   } | ||||
| } | ||||
|  | ||||
| export function revolveSketch( | ||||
|   node: Node<Program>, | ||||
|   pathToNode: PathToNode, | ||||
|  | ||||
| @ -1,5 +1,5 @@ | ||||
| import { | ||||
|   parse, | ||||
|   assertParse, | ||||
|   recast, | ||||
|   initPromise, | ||||
|   PathToNode, | ||||
| @ -78,9 +78,10 @@ const runGetPathToExtrudeForSegmentSelectionTest = async ( | ||||
|     code: string, | ||||
|     expectedExtrudeSnippet: string | ||||
|   ): CallExpression | PipeExpression | Error { | ||||
|     const extrudeRange: [number, number] = [ | ||||
|     const extrudeRange: [number, number, boolean] = [ | ||||
|       code.indexOf(expectedExtrudeSnippet), | ||||
|       code.indexOf(expectedExtrudeSnippet) + expectedExtrudeSnippet.length, | ||||
|       true, | ||||
|     ] | ||||
|     const expectedExtrudePath = getNodePathFromSourceRange(ast, extrudeRange) | ||||
|     const expectedExtrudeNodeResult = getNodeFromPath< | ||||
| @ -109,14 +110,13 @@ const runGetPathToExtrudeForSegmentSelectionTest = async ( | ||||
|   } | ||||
|  | ||||
|   // ast | ||||
|   const astOrError = parse(code) | ||||
|   if (err(astOrError)) return new Error('AST not found') | ||||
|   const ast = astOrError | ||||
|   const ast = assertParse(code) | ||||
|  | ||||
|   // selection | ||||
|   const segmentRange: [number, number] = [ | ||||
|   const segmentRange: [number, number, boolean] = [ | ||||
|     code.indexOf(selectedSegmentSnippet), | ||||
|     code.indexOf(selectedSegmentSnippet) + selectedSegmentSnippet.length, | ||||
|     true, | ||||
|   ] | ||||
|   const selection: Selections = { | ||||
|     graphSelections: [ | ||||
| @ -263,17 +263,14 @@ const runModifyAstCloneWithEdgeTreatmentAndTag = async ( | ||||
|   expectedCode: string | ||||
| ) => { | ||||
|   // ast | ||||
|   const astOrError = parse(code) | ||||
|   if (err(astOrError)) { | ||||
|     return new Error('AST not found') | ||||
|   } | ||||
|   const ast = astOrError | ||||
|   const ast = assertParse(code) | ||||
|  | ||||
|   // selection | ||||
|   const segmentRanges: Array<[number, number]> = selectionSnippets.map( | ||||
|   const segmentRanges: Array<[number, number, boolean]> = selectionSnippets.map( | ||||
|     (selectionSnippet) => [ | ||||
|       code.indexOf(selectionSnippet), | ||||
|       code.indexOf(selectionSnippet) + selectionSnippet.length, | ||||
|       true, | ||||
|     ] | ||||
|   ) | ||||
|  | ||||
| @ -603,12 +600,12 @@ extrude001 = extrude(-5, sketch001) | ||||
|      }, %) | ||||
| ` | ||||
|   it('should correctly identify getOppositeEdge and baseEdge edges', () => { | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) return | ||||
|     const ast = assertParse(code) | ||||
|     const lineOfInterest = `line([7.11, 3.48], %, $seg01)` | ||||
|     const range: [number, number] = [ | ||||
|     const range: [number, number, boolean] = [ | ||||
|       code.indexOf(lineOfInterest), | ||||
|       code.indexOf(lineOfInterest) + lineOfInterest.length, | ||||
|       true, | ||||
|     ] | ||||
|     const pathToNode = getNodePathFromSourceRange(ast, range) | ||||
|     if (err(pathToNode)) return | ||||
| @ -622,12 +619,12 @@ extrude001 = extrude(-5, sketch001) | ||||
|     expect(edges).toEqual(['getOppositeEdge', 'baseEdge']) | ||||
|   }) | ||||
|   it('should correctly identify getPreviousAdjacentEdge edges', () => { | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) return | ||||
|     const ast = assertParse(code) | ||||
|     const lineOfInterest = `line([-6.37, 3.88], %, $seg02)` | ||||
|     const range: [number, number] = [ | ||||
|     const range: [number, number, boolean] = [ | ||||
|       code.indexOf(lineOfInterest), | ||||
|       code.indexOf(lineOfInterest) + lineOfInterest.length, | ||||
|       true, | ||||
|     ] | ||||
|     const pathToNode = getNodePathFromSourceRange(ast, range) | ||||
|     if (err(pathToNode)) return | ||||
| @ -641,12 +638,12 @@ extrude001 = extrude(-5, sketch001) | ||||
|     expect(edges).toEqual(['getPreviousAdjacentEdge']) | ||||
|   }) | ||||
|   it('should correctly identify no edges', () => { | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) return | ||||
|     const ast = assertParse(code) | ||||
|     const lineOfInterest = `line([-3.29, -13.85], %)` | ||||
|     const range: [number, number] = [ | ||||
|     const range: [number, number, boolean] = [ | ||||
|       code.indexOf(lineOfInterest), | ||||
|       code.indexOf(lineOfInterest) + lineOfInterest.length, | ||||
|       true, | ||||
|     ] | ||||
|     const pathToNode = getNodePathFromSourceRange(ast, range) | ||||
|     if (err(pathToNode)) return | ||||
| @ -667,19 +664,15 @@ describe('Testing button states', () => { | ||||
|     segmentSnippet: string, | ||||
|     expectedState: boolean | ||||
|   ) => { | ||||
|     // ast | ||||
|     const astOrError = parse(code) | ||||
|     if (err(astOrError)) { | ||||
|       return new Error('AST not found') | ||||
|     } | ||||
|     const ast = astOrError | ||||
|     const ast = assertParse(code) | ||||
|  | ||||
|     const range: [number, number] = segmentSnippet | ||||
|     const range: [number, number, boolean] = segmentSnippet | ||||
|       ? [ | ||||
|           code.indexOf(segmentSnippet), | ||||
|           code.indexOf(segmentSnippet) + segmentSnippet.length, | ||||
|           true, | ||||
|         ] | ||||
|       : [ast.end, ast.end] // empty line in the end of the code | ||||
|       : [ast.end, ast.end, true] // empty line in the end of the code | ||||
|  | ||||
|     const selectionRanges: Selections = { | ||||
|       graphSelections: [ | ||||
|  | ||||
| @ -1,4 +1,10 @@ | ||||
| import { parse, recast, initPromise, PathToNode, Identifier } from './wasm' | ||||
| import { | ||||
|   assertParse, | ||||
|   recast, | ||||
|   initPromise, | ||||
|   PathToNode, | ||||
|   Identifier, | ||||
| } from './wasm' | ||||
| import { | ||||
|   findAllPreviousVariables, | ||||
|   isNodeSafeToReplace, | ||||
| @ -45,14 +51,13 @@ part001 = startSketchOn('XY') | ||||
| variableBelowShouldNotBeIncluded = 3 | ||||
| ` | ||||
|     const rangeStart = code.indexOf('// selection-range-7ish-before-this') - 7 | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const execState = await enginelessExecutor(ast) | ||||
|  | ||||
|     const { variables, bodyPath, insertIndex } = findAllPreviousVariables( | ||||
|       ast, | ||||
|       execState.memory, | ||||
|       [rangeStart, rangeStart] | ||||
|       [rangeStart, rangeStart, true] | ||||
|     ) | ||||
|     expect(variables).toEqual([ | ||||
|       { key: 'baseThick', value: 1 }, | ||||
| @ -80,10 +85,9 @@ describe('testing argIsNotIdentifier', () => { | ||||
| yo = 5 + 6 | ||||
| yo2 = hmm([identifierGuy + 5])` | ||||
|   it('find a safe binaryExpression', () => { | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const rangeStart = code.indexOf('100 + 100') + 2 | ||||
|     const result = isNodeSafeToReplace(ast, [rangeStart, rangeStart]) | ||||
|     const result = isNodeSafeToReplace(ast, [rangeStart, rangeStart, true]) | ||||
|     if (err(result)) throw result | ||||
|     expect(result.isSafe).toBe(true) | ||||
|     expect(result.value?.type).toBe('BinaryExpression') | ||||
| @ -94,20 +98,18 @@ yo2 = hmm([identifierGuy + 5])` | ||||
|     expect(outCode).toContain(`angledLine([replaceName, 3.09], %)`) | ||||
|   }) | ||||
|   it('find a safe Identifier', () => { | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const rangeStart = code.indexOf('abc') | ||||
|     const result = isNodeSafeToReplace(ast, [rangeStart, rangeStart]) | ||||
|     const result = isNodeSafeToReplace(ast, [rangeStart, rangeStart, true]) | ||||
|     if (err(result)) throw result | ||||
|     expect(result.isSafe).toBe(true) | ||||
|     expect(result.value?.type).toBe('Identifier') | ||||
|     expect(code.slice(result.value.start, result.value.end)).toBe('abc') | ||||
|   }) | ||||
|   it('find a safe CallExpression', () => { | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const rangeStart = code.indexOf('def') | ||||
|     const result = isNodeSafeToReplace(ast, [rangeStart, rangeStart]) | ||||
|     const result = isNodeSafeToReplace(ast, [rangeStart, rangeStart, true]) | ||||
|     if (err(result)) throw result | ||||
|     expect(result.isSafe).toBe(true) | ||||
|     expect(result.value?.type).toBe('CallExpression') | ||||
| @ -118,10 +120,9 @@ yo2 = hmm([identifierGuy + 5])` | ||||
|     expect(outCode).toContain(`angledLine([replaceName, 3.09], %)`) | ||||
|   }) | ||||
|   it('find an UNsafe CallExpression, as it has a PipeSubstitution', () => { | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const rangeStart = code.indexOf('ghi') | ||||
|     const range: [number, number] = [rangeStart, rangeStart] | ||||
|     const range: [number, number, boolean] = [rangeStart, rangeStart, true] | ||||
|     const result = isNodeSafeToReplace(ast, range) | ||||
|     if (err(result)) throw result | ||||
|     expect(result.isSafe).toBe(false) | ||||
| @ -129,10 +130,9 @@ yo2 = hmm([identifierGuy + 5])` | ||||
|     expect(code.slice(result.value.start, result.value.end)).toBe('ghi(%)') | ||||
|   }) | ||||
|   it('find an UNsafe Identifier, as it is a callee', () => { | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const rangeStart = code.indexOf('ine([2.8,') | ||||
|     const result = isNodeSafeToReplace(ast, [rangeStart, rangeStart]) | ||||
|     const result = isNodeSafeToReplace(ast, [rangeStart, rangeStart, true]) | ||||
|     if (err(result)) throw result | ||||
|     expect(result.isSafe).toBe(false) | ||||
|     expect(result.value?.type).toBe('CallExpression') | ||||
| @ -141,10 +141,9 @@ yo2 = hmm([identifierGuy + 5])` | ||||
|     ) | ||||
|   }) | ||||
|   it("find a safe BinaryExpression that's assigned to a variable", () => { | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const rangeStart = code.indexOf('5 + 6') + 1 | ||||
|     const result = isNodeSafeToReplace(ast, [rangeStart, rangeStart]) | ||||
|     const result = isNodeSafeToReplace(ast, [rangeStart, rangeStart, true]) | ||||
|     if (err(result)) throw result | ||||
|     expect(result.isSafe).toBe(true) | ||||
|     expect(result.value?.type).toBe('BinaryExpression') | ||||
| @ -155,10 +154,9 @@ yo2 = hmm([identifierGuy + 5])` | ||||
|     expect(outCode).toContain(`yo = replaceName`) | ||||
|   }) | ||||
|   it('find a safe BinaryExpression that has a CallExpression within', () => { | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     const rangeStart = code.indexOf('jkl') + 1 | ||||
|     const result = isNodeSafeToReplace(ast, [rangeStart, rangeStart]) | ||||
|     const result = isNodeSafeToReplace(ast, [rangeStart, rangeStart, true]) | ||||
|     if (err(result)) throw result | ||||
|     expect(result.isSafe).toBe(true) | ||||
|     expect(result.value?.type).toBe('BinaryExpression') | ||||
| @ -172,11 +170,10 @@ yo2 = hmm([identifierGuy + 5])` | ||||
|     expect(outCode).toContain(`angledLine([replaceName, 3.09], %)`) | ||||
|   }) | ||||
|   it('find a safe BinaryExpression within a CallExpression', () => { | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|  | ||||
|     const rangeStart = code.indexOf('identifierGuy') + 1 | ||||
|     const result = isNodeSafeToReplace(ast, [rangeStart, rangeStart]) | ||||
|     const result = isNodeSafeToReplace(ast, [rangeStart, rangeStart, true]) | ||||
|     if (err(result)) throw result | ||||
|  | ||||
|     expect(result.isSafe).toBe(true) | ||||
| @ -223,10 +220,13 @@ describe('testing getNodePathFromSourceRange', () => { | ||||
|   it('finds the second line when cursor is put at the end', () => { | ||||
|     const searchLn = `line([0.94, 2.61], %)` | ||||
|     const sourceIndex = code.indexOf(searchLn) + searchLn.length | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|  | ||||
|     const result = getNodePathFromSourceRange(ast, [sourceIndex, sourceIndex]) | ||||
|     const result = getNodePathFromSourceRange(ast, [ | ||||
|       sourceIndex, | ||||
|       sourceIndex, | ||||
|       true, | ||||
|     ]) | ||||
|     expect(result).toEqual([ | ||||
|       ['body', ''], | ||||
|       [0, 'index'], | ||||
| @ -240,10 +240,13 @@ describe('testing getNodePathFromSourceRange', () => { | ||||
|   it('finds the last line when cursor is put at the end', () => { | ||||
|     const searchLn = `line([-0.21, -1.4], %)` | ||||
|     const sourceIndex = code.indexOf(searchLn) + searchLn.length | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|  | ||||
|     const result = getNodePathFromSourceRange(ast, [sourceIndex, sourceIndex]) | ||||
|     const result = getNodePathFromSourceRange(ast, [ | ||||
|       sourceIndex, | ||||
|       sourceIndex, | ||||
|       true, | ||||
|     ]) | ||||
|     const expected = [ | ||||
|       ['body', ''], | ||||
|       [0, 'index'], | ||||
| @ -259,12 +262,14 @@ describe('testing getNodePathFromSourceRange', () => { | ||||
|     const startResult = getNodePathFromSourceRange(ast, [ | ||||
|       startSourceIndex, | ||||
|       startSourceIndex, | ||||
|       true, | ||||
|     ]) | ||||
|     expect(startResult).toEqual([...expected, ['callee', 'CallExpression']]) | ||||
|     // expect similar result when whole line is selected | ||||
|     const selectWholeThing = getNodePathFromSourceRange(ast, [ | ||||
|       startSourceIndex, | ||||
|       sourceIndex, | ||||
|       true, | ||||
|     ]) | ||||
|     expect(selectWholeThing).toEqual(expected) | ||||
|   }) | ||||
| @ -278,10 +283,13 @@ describe('testing getNodePathFromSourceRange', () => { | ||||
|     }` | ||||
|     const searchLn = `x > y` | ||||
|     const sourceIndex = code.indexOf(searchLn) | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|  | ||||
|     const result = getNodePathFromSourceRange(ast, [sourceIndex, sourceIndex]) | ||||
|     const result = getNodePathFromSourceRange(ast, [ | ||||
|       sourceIndex, | ||||
|       sourceIndex, | ||||
|       true, | ||||
|     ]) | ||||
|     expect(result).toEqual([ | ||||
|       ['body', ''], | ||||
|       [1, 'index'], | ||||
| @ -306,10 +314,13 @@ describe('testing getNodePathFromSourceRange', () => { | ||||
|     }` | ||||
|     const searchLn = `x + 1` | ||||
|     const sourceIndex = code.indexOf(searchLn) | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|  | ||||
|     const result = getNodePathFromSourceRange(ast, [sourceIndex, sourceIndex]) | ||||
|     const result = getNodePathFromSourceRange(ast, [ | ||||
|       sourceIndex, | ||||
|       sourceIndex, | ||||
|       true, | ||||
|     ]) | ||||
|     expect(result).toEqual([ | ||||
|       ['body', ''], | ||||
|       [1, 'index'], | ||||
| @ -332,10 +343,13 @@ describe('testing getNodePathFromSourceRange', () => { | ||||
|     const code = `import foo, bar as baz from 'thing.kcl'` | ||||
|     const searchLn = `bar` | ||||
|     const sourceIndex = code.indexOf(searchLn) | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|  | ||||
|     const result = getNodePathFromSourceRange(ast, [sourceIndex, sourceIndex]) | ||||
|     const result = getNodePathFromSourceRange(ast, [ | ||||
|       sourceIndex, | ||||
|       sourceIndex, | ||||
|       true, | ||||
|     ]) | ||||
|     expect(result).toEqual([ | ||||
|       ['body', ''], | ||||
|       [0, 'index'], | ||||
| @ -360,14 +374,13 @@ part001 = startSketchAt([-1.41, 3.46]) | ||||
|   |> angledLine([-175, segLen(seg01)], %) | ||||
|   |> close(%) | ||||
| ` | ||||
|     const ast = parse(exampleCode) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(exampleCode) | ||||
|  | ||||
|     const result = doesPipeHaveCallExp({ | ||||
|       calleeName: 'close', | ||||
|       ast, | ||||
|       selection: { | ||||
|         codeRef: codeRefFromRange([100, 101], ast), | ||||
|         codeRef: codeRefFromRange([100, 101, true], ast), | ||||
|       }, | ||||
|     }) | ||||
|     expect(result).toEqual(true) | ||||
| @ -382,14 +395,13 @@ part001 = startSketchAt([-1.41, 3.46]) | ||||
|   |> close(%) | ||||
|   |> extrude(1, %) | ||||
| ` | ||||
|     const ast = parse(exampleCode) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(exampleCode) | ||||
|  | ||||
|     const result = doesPipeHaveCallExp({ | ||||
|       calleeName: 'extrude', | ||||
|       ast, | ||||
|       selection: { | ||||
|         codeRef: codeRefFromRange([100, 101], ast), | ||||
|         codeRef: codeRefFromRange([100, 101, true], ast), | ||||
|       }, | ||||
|     }) | ||||
|     expect(result).toEqual(true) | ||||
| @ -402,28 +414,26 @@ part001 = startSketchAt([-1.41, 3.46]) | ||||
|   |> line([-3.22, -7.36], %) | ||||
|   |> angledLine([-175, segLen(seg01)], %) | ||||
| ` | ||||
|     const ast = parse(exampleCode) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(exampleCode) | ||||
|  | ||||
|     const result = doesPipeHaveCallExp({ | ||||
|       calleeName: 'close', | ||||
|       ast, | ||||
|       selection: { | ||||
|         codeRef: codeRefFromRange([100, 101], ast), | ||||
|         codeRef: codeRefFromRange([100, 101, true], ast), | ||||
|       }, | ||||
|     }) | ||||
|     expect(result).toEqual(false) | ||||
|   }) | ||||
|   it('returns false if not a pipe', () => { | ||||
|     const exampleCode = `length001 = 2` | ||||
|     const ast = parse(exampleCode) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(exampleCode) | ||||
|  | ||||
|     const result = doesPipeHaveCallExp({ | ||||
|       calleeName: 'close', | ||||
|       ast, | ||||
|       selection: { | ||||
|         codeRef: codeRefFromRange([9, 10], ast), | ||||
|         codeRef: codeRefFromRange([9, 10, true], ast), | ||||
|       }, | ||||
|     }) | ||||
|     expect(result).toEqual(false) | ||||
| @ -438,14 +448,13 @@ part001 = startSketchAt([-1.41, 3.46]) | ||||
|   |> angledLine([-35, length001], %) | ||||
|   |> line([-3.22, -7.36], %) | ||||
|   |> angledLine([-175, segLen(seg01)], %)` | ||||
|     const ast = parse(exampleCode) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(exampleCode) | ||||
|  | ||||
|     const execState = await enginelessExecutor(ast) | ||||
|     const result = hasExtrudeSketch({ | ||||
|       ast, | ||||
|       selection: { | ||||
|         codeRef: codeRefFromRange([100, 101], ast), | ||||
|         codeRef: codeRefFromRange([100, 101, true], ast), | ||||
|       }, | ||||
|       programMemory: execState.memory, | ||||
|     }) | ||||
| @ -459,14 +468,13 @@ part001 = startSketchAt([-1.41, 3.46]) | ||||
|   |> line([-3.22, -7.36], %) | ||||
|   |> angledLine([-175, segLen(seg01)], %) | ||||
|   |> extrude(1, %)` | ||||
|     const ast = parse(exampleCode) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(exampleCode) | ||||
|  | ||||
|     const execState = await enginelessExecutor(ast) | ||||
|     const result = hasExtrudeSketch({ | ||||
|       ast, | ||||
|       selection: { | ||||
|         codeRef: codeRefFromRange([100, 101], ast), | ||||
|         codeRef: codeRefFromRange([100, 101, true], ast), | ||||
|       }, | ||||
|       programMemory: execState.memory, | ||||
|     }) | ||||
| @ -474,14 +482,13 @@ part001 = startSketchAt([-1.41, 3.46]) | ||||
|   }) | ||||
|   it('finds nothing', async () => { | ||||
|     const exampleCode = `length001 = 2` | ||||
|     const ast = parse(exampleCode) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(exampleCode) | ||||
|  | ||||
|     const execState = await enginelessExecutor(ast) | ||||
|     const result = hasExtrudeSketch({ | ||||
|       ast, | ||||
|       selection: { | ||||
|         codeRef: codeRefFromRange([10, 11], ast), | ||||
|         codeRef: codeRefFromRange([10, 11, true], ast), | ||||
|       }, | ||||
|       programMemory: execState.memory, | ||||
|     }) | ||||
| @ -498,8 +505,7 @@ describe('Testing findUsesOfTagInPipe', () => { | ||||
| |> line([306.21, 198.87], %) | ||||
| |> angledLine([65, segLen(seg01)], %)` | ||||
|   it('finds the current segment', async () => { | ||||
|     const ast = parse(exampleCode) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(exampleCode) | ||||
|  | ||||
|     const lineOfInterest = `198.85], %, $seg01` | ||||
|     const characterIndex = | ||||
| @ -507,6 +513,7 @@ describe('Testing findUsesOfTagInPipe', () => { | ||||
|     const pathToNode = getNodePathFromSourceRange(ast, [ | ||||
|       characterIndex, | ||||
|       characterIndex, | ||||
|       true, | ||||
|     ]) | ||||
|     const result = findUsesOfTagInPipe(ast, pathToNode) | ||||
|     expect(result).toHaveLength(2) | ||||
| @ -515,8 +522,7 @@ describe('Testing findUsesOfTagInPipe', () => { | ||||
|     }) | ||||
|   }) | ||||
|   it('find no tag if line has no tag', () => { | ||||
|     const ast = parse(exampleCode) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(exampleCode) | ||||
|  | ||||
|     const lineOfInterest = `line([306.21, 198.82], %)` | ||||
|     const characterIndex = | ||||
| @ -524,6 +530,7 @@ describe('Testing findUsesOfTagInPipe', () => { | ||||
|     const pathToNode = getNodePathFromSourceRange(ast, [ | ||||
|       characterIndex, | ||||
|       characterIndex, | ||||
|       true, | ||||
|     ]) | ||||
|     const result = findUsesOfTagInPipe(ast, pathToNode) | ||||
|     expect(result).toHaveLength(0) | ||||
| @ -564,42 +571,39 @@ sketch003 = startSketchOn(extrude001, 'END') | ||||
|   |> extrude(3.14, %) | ||||
| ` | ||||
|   it('identifies sketch001 pipe as extruded (extrusion after pipe)', async () => { | ||||
|     const ast = parse(exampleCode) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(exampleCode) | ||||
|     const lineOfInterest = `line([4.99, -0.46], %, $seg01)` | ||||
|     const characterIndex = | ||||
|       exampleCode.indexOf(lineOfInterest) + lineOfInterest.length | ||||
|     const extruded = hasSketchPipeBeenExtruded( | ||||
|       { | ||||
|         codeRef: codeRefFromRange([characterIndex, characterIndex], ast), | ||||
|         codeRef: codeRefFromRange([characterIndex, characterIndex, true], ast), | ||||
|       }, | ||||
|       ast | ||||
|     ) | ||||
|     expect(extruded).toBeTruthy() | ||||
|   }) | ||||
|   it('identifies sketch002 pipe as not extruded', async () => { | ||||
|     const ast = parse(exampleCode) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(exampleCode) | ||||
|     const lineOfInterest = `line([2.45, -0.2], %)` | ||||
|     const characterIndex = | ||||
|       exampleCode.indexOf(lineOfInterest) + lineOfInterest.length | ||||
|     const extruded = hasSketchPipeBeenExtruded( | ||||
|       { | ||||
|         codeRef: codeRefFromRange([characterIndex, characterIndex], ast), | ||||
|         codeRef: codeRefFromRange([characterIndex, characterIndex, true], ast), | ||||
|       }, | ||||
|       ast | ||||
|     ) | ||||
|     expect(extruded).toBeFalsy() | ||||
|   }) | ||||
|   it('identifies sketch003 pipe as extruded (extrusion within pipe)', async () => { | ||||
|     const ast = parse(exampleCode) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(exampleCode) | ||||
|     const lineOfInterest = `|> line([3.12, 1.74], %)` | ||||
|     const characterIndex = | ||||
|       exampleCode.indexOf(lineOfInterest) + lineOfInterest.length | ||||
|     const extruded = hasSketchPipeBeenExtruded( | ||||
|       { | ||||
|         codeRef: codeRefFromRange([characterIndex, characterIndex], ast), | ||||
|         codeRef: codeRefFromRange([characterIndex, characterIndex, true], ast), | ||||
|       }, | ||||
|       ast | ||||
|     ) | ||||
| @ -623,11 +627,21 @@ sketch002 = startSketchOn(extrude001, $seg01) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
| ` | ||||
|     const ast = parse(exampleCode) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(exampleCode) | ||||
|     const extrudable = doesSceneHaveSweepableSketch(ast) | ||||
|     expect(extrudable).toBeTruthy() | ||||
|   }) | ||||
|   it('finds sketch001 and sketch002 pipes to be lofted', async () => { | ||||
|     const exampleCode = `sketch001 = startSketchOn('XZ') | ||||
|   |> circle({ center = [0, 0], radius = 1 }, %) | ||||
| plane001 = offsetPlane('XZ', 2) | ||||
| sketch002 = startSketchOn(plane001) | ||||
|   |> circle({ center = [0, 0], radius = 3 }, %) | ||||
| ` | ||||
|     const ast = assertParse(exampleCode) | ||||
|     const extrudable = doesSceneHaveSweepableSketch(ast, 2) | ||||
|     expect(extrudable).toBeTruthy() | ||||
|   }) | ||||
|   it('find sketch002 NOT pipe to be extruded', async () => { | ||||
|     const exampleCode = `sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([3.29, 7.86], %) | ||||
| @ -637,8 +651,7 @@ sketch002 = startSketchOn(extrude001, $seg01) | ||||
|   |> close(%) | ||||
| extrude001 = extrude(10, sketch001) | ||||
| ` | ||||
|     const ast = parse(exampleCode) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(exampleCode) | ||||
|     const extrudable = doesSceneHaveSweepableSketch(ast) | ||||
|     expect(extrudable).toBeFalsy() | ||||
|   }) | ||||
| @ -666,8 +679,7 @@ myNestedVar = [ | ||||
| } | ||||
| ] | ||||
|   ` | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) throw ast | ||||
|     const ast = assertParse(code) | ||||
|     let pathToNode: PathToNode = [] | ||||
|     traverse(ast, { | ||||
|       enter: (node, path) => { | ||||
| @ -689,6 +701,7 @@ myNestedVar = [ | ||||
|     const pathToNode2 = getNodePathFromSourceRange(ast, [ | ||||
|       literalIndex + 2, | ||||
|       literalIndex + 2, | ||||
|       true, | ||||
|     ]) | ||||
|     expect(pathToNode).toEqual(pathToNode2) | ||||
|   }) | ||||
|  | ||||
| @ -16,6 +16,7 @@ import { | ||||
|   sketchFromKclValue, | ||||
|   sketchFromKclValueOptional, | ||||
|   SourceRange, | ||||
|   sourceRangeFromRust, | ||||
|   SyntaxType, | ||||
|   VariableDeclaration, | ||||
|   VariableDeclarator, | ||||
| @ -173,6 +174,30 @@ function moreNodePathFromSourceRange( | ||||
|     } | ||||
|     return path | ||||
|   } | ||||
|  | ||||
|   if (_node.type === 'CallExpressionKw' && isInRange) { | ||||
|     const { callee, arguments: args } = _node | ||||
|     if ( | ||||
|       callee.type === 'Identifier' && | ||||
|       callee.start <= start && | ||||
|       callee.end >= end | ||||
|     ) { | ||||
|       path.push(['callee', 'CallExpressionKw']) | ||||
|       return path | ||||
|     } | ||||
|     if (args.length > 0) { | ||||
|       for (let argIndex = 0; argIndex < args.length; argIndex++) { | ||||
|         const arg = args[argIndex].arg | ||||
|         if (arg.start <= start && arg.end >= end) { | ||||
|           path.push(['arguments', 'CallExpressionKw']) | ||||
|           path.push([argIndex, 'index']) | ||||
|           return moreNodePathFromSourceRange(arg, sourceRange, path) | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|     return path | ||||
|   } | ||||
|  | ||||
|   if (_node.type === 'BinaryExpression' && isInRange) { | ||||
|     const { left, right } = _node | ||||
|     if (left.start <= start && left.end >= end) { | ||||
| @ -645,7 +670,7 @@ export function isNodeSafeToReplacePath( | ||||
|  | ||||
| export function isNodeSafeToReplace( | ||||
|   ast: Node<Program>, | ||||
|   sourceRange: [number, number] | ||||
|   sourceRange: SourceRange | ||||
| ): | ||||
|   | { | ||||
|       isSafe: boolean | ||||
| @ -797,7 +822,7 @@ export function isLinesParallelAndConstrained( | ||||
|     return { | ||||
|       isParallelAndConstrained, | ||||
|       selection: { | ||||
|         codeRef: codeRefFromRange(prevSourceRange, ast), | ||||
|         codeRef: codeRefFromRange(sourceRangeFromRust(prevSourceRange), ast), | ||||
|         artifact: artifactGraph.get(prevSegment.__geoMeta.id), | ||||
|       }, | ||||
|     } | ||||
| @ -933,7 +958,8 @@ export function findUsesOfTagInPipe( | ||||
|         return | ||||
|       const tagArgValue = | ||||
|         tagArg.type === 'TagDeclarator' ? String(tagArg.value) : tagArg.name | ||||
|       if (tagArgValue === tag) dependentRanges.push([node.start, node.end]) | ||||
|       if (tagArgValue === tag) | ||||
|         dependentRanges.push([node.start, node.end, true]) | ||||
|     }, | ||||
|   }) | ||||
|   return dependentRanges | ||||
| @ -975,7 +1001,9 @@ export function hasSketchPipeBeenExtruded(selection: Selection, ast: Program) { | ||||
|         if ( | ||||
|           node.type === 'CallExpression' && | ||||
|           node.callee.type === 'Identifier' && | ||||
|           (node.callee.name === 'extrude' || node.callee.name === 'revolve') && | ||||
|           (node.callee.name === 'extrude' || | ||||
|             node.callee.name === 'revolve' || | ||||
|             node.callee.name === 'loft') && | ||||
|           node.arguments?.[1]?.type === 'Identifier' && | ||||
|           node.arguments[1].name === varDec.id.name | ||||
|         ) { | ||||
| @ -988,7 +1016,7 @@ export function hasSketchPipeBeenExtruded(selection: Selection, ast: Program) { | ||||
| } | ||||
|  | ||||
| /** File must contain at least one sketch that has not been extruded already */ | ||||
| export function doesSceneHaveSweepableSketch(ast: Node<Program>) { | ||||
| export function doesSceneHaveSweepableSketch(ast: Node<Program>, count = 1) { | ||||
|   const theMap: any = {} | ||||
|   traverse(ast as any, { | ||||
|     enter(node) { | ||||
| @ -1037,7 +1065,7 @@ export function doesSceneHaveSweepableSketch(ast: Node<Program>) { | ||||
|       } | ||||
|     }, | ||||
|   }) | ||||
|   return Object.keys(theMap).length > 0 | ||||
|   return Object.keys(theMap).length >= count | ||||
| } | ||||
|  | ||||
| export function getObjExprProperty( | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| import { parse, Program, recast, initPromise } from './wasm' | ||||
| import { assertParse, Program, recast, initPromise } from './wasm' | ||||
| import fs from 'node:fs' | ||||
| import { err } from 'lib/trap' | ||||
|  | ||||
| @ -394,8 +394,6 @@ describe('it recasts binary expression using brackets where needed', () => { | ||||
| // helpers | ||||
|  | ||||
| function code2ast(code: string): { ast: Program } { | ||||
|   const ast = parse(code) | ||||
|   // eslint-ignore-next-line | ||||
|   if (err(ast)) throw ast | ||||
|   const ast = assertParse(code) | ||||
|   return { ast } | ||||
| } | ||||
|  | ||||
| @ -11,8 +11,8 @@ Map { | ||||
|         ], | ||||
|       ], | ||||
|       "range": [ | ||||
|         37, | ||||
|         64, | ||||
|         12, | ||||
|         31, | ||||
|         0, | ||||
|       ], | ||||
|     }, | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| import { makeDefaultPlanes, parse, initPromise, Program } from 'lang/wasm' | ||||
| import { makeDefaultPlanes, assertParse, initPromise, Program } from 'lang/wasm' | ||||
| import { Models } from '@kittycad/lib' | ||||
| import { | ||||
|   OrderedCommand, | ||||
| @ -148,11 +148,7 @@ beforeAll(async () => { | ||||
|         ][] | ||||
|         const cacheToWriteToFileTemp: Partial<CacheShape> = {} | ||||
|         for (const [codeKey, code] of cacheEntries) { | ||||
|           const ast = parse(code) | ||||
|           if (err(ast)) { | ||||
|             console.error(ast) | ||||
|             return Promise.reject(ast) | ||||
|           } | ||||
|           const ast = assertParse(code) | ||||
|           await kclManager.executeAst({ ast }) | ||||
|  | ||||
|           cacheToWriteToFileTemp[codeKey] = { | ||||
| @ -403,11 +399,7 @@ describe('capture graph of sketchOnFaceOnFace...', () => { | ||||
| }) | ||||
|  | ||||
| function getCommands(codeKey: CodeKey): CacheShape[CodeKey] & { ast: Program } { | ||||
|   const ast = parse(codeKey) | ||||
|   if (err(ast)) { | ||||
|     console.error(ast) | ||||
|     throw ast | ||||
|   } | ||||
|   const ast = assertParse(codeKey) | ||||
|   const file = fs.readFileSync(fullPath, 'utf-8') | ||||
|   const parsed: CacheShape = JSON.parse(file) | ||||
|   // these either already exist from the last run, or were created in | ||||
|  | ||||
| Before Width: | Height: | Size: 378 KiB After Width: | Height: | Size: 357 KiB | 
| Before Width: | Height: | Size: 92 KiB After Width: | Height: | Size: 88 KiB | 
| Before Width: | Height: | Size: 613 KiB After Width: | Height: | Size: 577 KiB | 
| @ -1,4 +1,4 @@ | ||||
| import { SourceRange } from 'lang/wasm' | ||||
| import { defaultSourceRange, SourceRange } from 'lang/wasm' | ||||
| import { VITE_KC_API_WS_MODELING_URL, VITE_KC_DEV_TOKEN } from 'env' | ||||
| import { Models } from '@kittycad/lib' | ||||
| import { exportSave } from 'lib/exportSave' | ||||
| @ -1879,7 +1879,7 @@ export class EngineCommandManager extends EventTarget { | ||||
|     } | ||||
|     return JSON.stringify(this.defaultPlanes) | ||||
|   } | ||||
|   endSession() { | ||||
|   clearScene(): void { | ||||
|     const deleteCmd: EngineCommand = { | ||||
|       type: 'modeling_cmd_req', | ||||
|       cmd_id: uuidv4(), | ||||
| @ -2014,7 +2014,7 @@ export class EngineCommandManager extends EventTarget { | ||||
|       { | ||||
|         command, | ||||
|         idToRangeMap: {}, | ||||
|         range: [0, 0], | ||||
|         range: defaultSourceRange(), | ||||
|       }, | ||||
|       true // isSceneCommand | ||||
|     ) | ||||
|  | ||||
| @ -8,7 +8,7 @@ import { | ||||
|   getConstraintInfo, | ||||
| } from './sketch' | ||||
| import { | ||||
|   parse, | ||||
|   assertParse, | ||||
|   recast, | ||||
|   initPromise, | ||||
|   SourceRange, | ||||
| @ -115,8 +115,7 @@ describe('testing changeSketchArguments', () => { | ||||
| ` | ||||
|     const code = genCode(lineToChange) | ||||
|     const expectedCode = genCode(lineAfterChange) | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) return ast | ||||
|     const ast = assertParse(code) | ||||
|  | ||||
|     const execState = await enginelessExecutor(ast) | ||||
|     const sourceStart = code.indexOf(lineToChange) | ||||
| @ -125,7 +124,7 @@ describe('testing changeSketchArguments', () => { | ||||
|       execState.memory, | ||||
|       { | ||||
|         type: 'sourceRange', | ||||
|         sourceRange: [sourceStart, sourceStart + lineToChange.length], | ||||
|         sourceRange: [sourceStart, sourceStart + lineToChange.length, true], | ||||
|       }, | ||||
|       { | ||||
|         type: 'straight-segment', | ||||
| @ -148,8 +147,7 @@ mySketch001 = startSketchOn('XY') | ||||
|   // |> rx(45, %) | ||||
|   |> lineTo([-1.59, -1.54], %) | ||||
|   |> lineTo([0.46, -5.82], %)` | ||||
|     const ast = parse(code) | ||||
|     if (err(ast)) return ast | ||||
|     const ast = assertParse(code) | ||||
|  | ||||
|     const execState = await enginelessExecutor(ast) | ||||
|     const sourceStart = code.indexOf(lineToChange) | ||||
| @ -220,12 +218,13 @@ describe('testing addTagForSketchOnFace', () => { | ||||
|   |> lineTo([0.46, -5.82], %) | ||||
| ` | ||||
|     const code = genCode(originalLine) | ||||
|     const ast = parse(code) | ||||
|     const ast = assertParse(code) | ||||
|     await enginelessExecutor(ast) | ||||
|     const sourceStart = code.indexOf(originalLine) | ||||
|     const sourceRange: [number, number] = [ | ||||
|     const sourceRange: [number, number, boolean] = [ | ||||
|       sourceStart, | ||||
|       sourceStart + originalLine.length, | ||||
|       true, | ||||
|     ] | ||||
|     if (err(ast)) return ast | ||||
|     const pathToNode = getNodePathFromSourceRange(ast, sourceRange) | ||||
| @ -291,13 +290,14 @@ extrude001 = extrude(100, sketch001) | ||||
| ${insertCode} | ||||
| ` | ||||
|       const code = genCode(originalChamfer) | ||||
|       const ast = parse(code) | ||||
|       const ast = assertParse(code) | ||||
|       await enginelessExecutor(ast) | ||||
|       const sourceStart = code.indexOf(originalChamfer) | ||||
|       const extraChars = originalChamfer.indexOf('chamfer') | ||||
|       const sourceRange: [number, number] = [ | ||||
|       const sourceRange: [number, number, boolean] = [ | ||||
|         sourceStart + extraChars, | ||||
|         sourceStart + originalChamfer.length - extraChars, | ||||
|         true, | ||||
|       ] | ||||
|  | ||||
|       if (err(ast)) throw ast | ||||
| @ -335,7 +335,7 @@ describe('testing getConstraintInfo', () => { | ||||
|   |> lineTo([6.14, 3.14], %) | ||||
|   |> xLineTo(8, %) | ||||
|   |> yLineTo(5, %) | ||||
|   |> yLine(3.14, %, 'a') | ||||
|   |> yLine(3.14, %, $a) | ||||
|   |> xLine(3.14, %) | ||||
|   |> angledLineOfXLength({ | ||||
|     angle = 3.14, | ||||
| @ -355,11 +355,11 @@ describe('testing getConstraintInfo', () => { | ||||
|   }, %) | ||||
|   |> angledLineThatIntersects({ | ||||
|     angle = 3.14, | ||||
|     intersectTag = 'a', | ||||
|     intersectTag = a, | ||||
|     offset = 0 | ||||
|   }, %) | ||||
|   |> tangentialArcTo([3.14, 13.14], %)` | ||||
|     const ast = parse(code) | ||||
|     const ast = assertParse(code) | ||||
|     test.each([ | ||||
|       [ | ||||
|         'line', | ||||
| @ -368,7 +368,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'xRelative', | ||||
|             isConstrained: false, | ||||
|             value: '3', | ||||
|             sourceRange: [78, 79], | ||||
|             sourceRange: [78, 79, true], | ||||
|             argPosition: { type: 'arrayItem', index: 0 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'line', | ||||
| @ -377,7 +377,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'yRelative', | ||||
|             isConstrained: false, | ||||
|             value: '4', | ||||
|             sourceRange: [81, 82], | ||||
|             sourceRange: [81, 82, true], | ||||
|             argPosition: { type: 'arrayItem', index: 1 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'line', | ||||
| @ -391,7 +391,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'angle', | ||||
|             isConstrained: false, | ||||
|             value: '3.14', | ||||
|             sourceRange: [117, 121], | ||||
|             sourceRange: [118, 122, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'angle' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLine', | ||||
| @ -400,7 +400,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'length', | ||||
|             isConstrained: false, | ||||
|             value: '3.14', | ||||
|             sourceRange: [135, 139], | ||||
|             sourceRange: [137, 141, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'length' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLine', | ||||
| @ -414,7 +414,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'xAbsolute', | ||||
|             isConstrained: false, | ||||
|             value: '6.14', | ||||
|             sourceRange: [162, 166], | ||||
|             sourceRange: [164, 168, true], | ||||
|             argPosition: { type: 'arrayItem', index: 0 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'lineTo', | ||||
| @ -423,7 +423,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'yAbsolute', | ||||
|             isConstrained: false, | ||||
|             value: '3.14', | ||||
|             sourceRange: [168, 172], | ||||
|             sourceRange: [170, 174, true], | ||||
|             argPosition: { type: 'arrayItem', index: 1 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'lineTo', | ||||
| @ -437,7 +437,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'horizontal', | ||||
|             isConstrained: true, | ||||
|             value: 'xLineTo', | ||||
|             sourceRange: [183, 190], | ||||
|             sourceRange: [185, 192, true], | ||||
|             argPosition: undefined, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'xLineTo', | ||||
| @ -446,7 +446,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'xAbsolute', | ||||
|             isConstrained: false, | ||||
|             value: '8', | ||||
|             sourceRange: [191, 192], | ||||
|             sourceRange: [193, 194, true], | ||||
|             argPosition: { type: 'singleValue' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'xLineTo', | ||||
| @ -460,7 +460,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'vertical', | ||||
|             isConstrained: true, | ||||
|             value: 'yLineTo', | ||||
|             sourceRange: [202, 209], | ||||
|             sourceRange: [204, 211, true], | ||||
|             argPosition: undefined, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'yLineTo', | ||||
| @ -469,7 +469,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'yAbsolute', | ||||
|             isConstrained: false, | ||||
|             value: '5', | ||||
|             sourceRange: [210, 211], | ||||
|             sourceRange: [212, 213, true], | ||||
|             argPosition: { type: 'singleValue' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'yLineTo', | ||||
| @ -483,7 +483,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'vertical', | ||||
|             isConstrained: true, | ||||
|             value: 'yLine', | ||||
|             sourceRange: [221, 226], | ||||
|             sourceRange: [223, 228, true], | ||||
|             argPosition: undefined, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'yLine', | ||||
| @ -492,7 +492,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'yRelative', | ||||
|             isConstrained: false, | ||||
|             value: '3.14', | ||||
|             sourceRange: [227, 231], | ||||
|             sourceRange: [229, 233, true], | ||||
|             argPosition: { type: 'singleValue' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'yLine', | ||||
| @ -506,7 +506,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'horizontal', | ||||
|             isConstrained: true, | ||||
|             value: 'xLine', | ||||
|             sourceRange: [246, 251], | ||||
|             sourceRange: [247, 252, true], | ||||
|             argPosition: undefined, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'xLine', | ||||
| @ -515,7 +515,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'xRelative', | ||||
|             isConstrained: false, | ||||
|             value: '3.14', | ||||
|             sourceRange: [252, 256], | ||||
|             sourceRange: [253, 257, true], | ||||
|             argPosition: { type: 'singleValue' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'xLine', | ||||
| @ -529,7 +529,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'angle', | ||||
|             isConstrained: false, | ||||
|             value: '3.14', | ||||
|             sourceRange: [299, 303], | ||||
|             sourceRange: [301, 305, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'angle' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineOfXLength', | ||||
| @ -538,7 +538,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'xRelative', | ||||
|             isConstrained: false, | ||||
|             value: '3.14', | ||||
|             sourceRange: [317, 321], | ||||
|             sourceRange: [320, 324, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'length' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineOfXLength', | ||||
| @ -552,7 +552,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'angle', | ||||
|             isConstrained: false, | ||||
|             value: '30', | ||||
|             sourceRange: [369, 371], | ||||
|             sourceRange: [373, 375, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'angle' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineOfYLength', | ||||
| @ -561,7 +561,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'yRelative', | ||||
|             isConstrained: false, | ||||
|             value: '3', | ||||
|             sourceRange: [385, 386], | ||||
|             sourceRange: [390, 391, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'length' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineOfYLength', | ||||
| @ -575,7 +575,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'angle', | ||||
|             isConstrained: false, | ||||
|             value: '12.14', | ||||
|             sourceRange: [428, 433], | ||||
|             sourceRange: [434, 439, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'angle' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineToX', | ||||
| @ -584,7 +584,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'xAbsolute', | ||||
|             isConstrained: false, | ||||
|             value: '12', | ||||
|             sourceRange: [443, 445], | ||||
|             sourceRange: [450, 452, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'to' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineToX', | ||||
| @ -598,7 +598,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'angle', | ||||
|             isConstrained: false, | ||||
|             value: '30', | ||||
|             sourceRange: [487, 489], | ||||
|             sourceRange: [495, 497, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'angle' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineToY', | ||||
| @ -607,7 +607,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'yAbsolute', | ||||
|             isConstrained: false, | ||||
|             value: '10.14', | ||||
|             sourceRange: [499, 504], | ||||
|             sourceRange: [508, 513, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'to' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineToY', | ||||
| @ -621,7 +621,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'angle', | ||||
|             isConstrained: false, | ||||
|             value: '3.14', | ||||
|             sourceRange: [557, 561], | ||||
|             sourceRange: [567, 571, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'angle' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineThatIntersects', | ||||
| @ -630,7 +630,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'intersectionOffset', | ||||
|             isConstrained: false, | ||||
|             value: '0', | ||||
|             sourceRange: [598, 599], | ||||
|             sourceRange: [608, 609, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'offset' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineThatIntersects', | ||||
| @ -638,8 +638,8 @@ describe('testing getConstraintInfo', () => { | ||||
|           { | ||||
|             type: 'intersectionTag', | ||||
|             isConstrained: false, | ||||
|             value: "'a'", | ||||
|             sourceRange: [581, 584], | ||||
|             value: 'a', | ||||
|             sourceRange: [592, 593, true], | ||||
|             argPosition: { | ||||
|               key: 'intersectTag', | ||||
|               type: 'objectProperty', | ||||
| @ -656,7 +656,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'tangentialWithPrevious', | ||||
|             isConstrained: true, | ||||
|             value: 'tangentialArcTo', | ||||
|             sourceRange: [613, 628], | ||||
|             sourceRange: [623, 638, true], | ||||
|             argPosition: undefined, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'tangentialArcTo', | ||||
| @ -665,7 +665,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'xAbsolute', | ||||
|             isConstrained: false, | ||||
|             value: '3.14', | ||||
|             sourceRange: [630, 634], | ||||
|             sourceRange: [640, 644, true], | ||||
|             argPosition: { type: 'arrayItem', index: 0 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'tangentialArcTo', | ||||
| @ -674,7 +674,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'yAbsolute', | ||||
|             isConstrained: false, | ||||
|             value: '13.14', | ||||
|             sourceRange: [636, 641], | ||||
|             sourceRange: [646, 651, true], | ||||
|             argPosition: { type: 'arrayItem', index: 1 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'tangentialArcTo', | ||||
| @ -685,6 +685,7 @@ describe('testing getConstraintInfo', () => { | ||||
|       const sourceRange: SourceRange = [ | ||||
|         code.indexOf(functionName), | ||||
|         code.indexOf(functionName) + functionName.length, | ||||
|         true, | ||||
|       ] | ||||
|       if (err(ast)) return ast | ||||
|       const pathToNode = getNodePathFromSourceRange(ast, sourceRange) | ||||
| @ -706,7 +707,7 @@ describe('testing getConstraintInfo', () => { | ||||
|     |> lineTo([6.14, 3.14], %) | ||||
|     |> xLineTo(8, %) | ||||
|     |> yLineTo(5, %) | ||||
|     |> yLine(3.14, %, 'a') | ||||
|     |> yLine(3.14, %, $a) | ||||
|     |> xLine(3.14, %) | ||||
|     |> angledLineOfXLength([3.14, 3.14], %) | ||||
|     |> angledLineOfYLength([30, 3], %) | ||||
| @ -714,11 +715,11 @@ describe('testing getConstraintInfo', () => { | ||||
|     |> angledLineToY([30, 10], %) | ||||
|     |> angledLineThatIntersects({ | ||||
|          angle = 3.14, | ||||
|          intersectTag = 'a', | ||||
|          intersectTag = a, | ||||
|          offset = 0 | ||||
|        }, %) | ||||
|     |> tangentialArcTo([3.14, 13.14], %)` | ||||
|     const ast = parse(code) | ||||
|     const ast = assertParse(code) | ||||
|     test.each([ | ||||
|       [ | ||||
|         `angledLine(`, | ||||
| @ -727,7 +728,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'angle', | ||||
|             isConstrained: false, | ||||
|             value: '3.14', | ||||
|             sourceRange: [112, 116], | ||||
|             sourceRange: [112, 116, true], | ||||
|             argPosition: { type: 'arrayItem', index: 0 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLine', | ||||
| @ -736,7 +737,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'length', | ||||
|             isConstrained: false, | ||||
|             value: '3.14', | ||||
|             sourceRange: [118, 122], | ||||
|             sourceRange: [118, 122, true], | ||||
|             argPosition: { type: 'arrayItem', index: 1 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLine', | ||||
| @ -750,7 +751,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'angle', | ||||
|             isConstrained: false, | ||||
|             value: '3.14', | ||||
|             sourceRange: [278, 282], | ||||
|             sourceRange: [277, 281, true], | ||||
|             argPosition: { type: 'arrayItem', index: 0 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineOfXLength', | ||||
| @ -759,7 +760,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'xRelative', | ||||
|             isConstrained: false, | ||||
|             value: '3.14', | ||||
|             sourceRange: [284, 288], | ||||
|             sourceRange: [283, 287, true], | ||||
|             argPosition: { type: 'arrayItem', index: 1 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineOfXLength', | ||||
| @ -773,7 +774,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'angle', | ||||
|             isConstrained: false, | ||||
|             value: '30', | ||||
|             sourceRange: [322, 324], | ||||
|             sourceRange: [321, 323, true], | ||||
|             argPosition: { type: 'arrayItem', index: 0 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineOfYLength', | ||||
| @ -782,7 +783,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'yRelative', | ||||
|             isConstrained: false, | ||||
|             value: '3', | ||||
|             sourceRange: [326, 327], | ||||
|             sourceRange: [325, 326, true], | ||||
|             argPosition: { type: 'arrayItem', index: 1 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineOfYLength', | ||||
| @ -796,7 +797,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'angle', | ||||
|             isConstrained: false, | ||||
|             value: '12', | ||||
|             sourceRange: [355, 357], | ||||
|             sourceRange: [354, 356, true], | ||||
|             argPosition: { type: 'arrayItem', index: 0 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineToX', | ||||
| @ -805,7 +806,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'xAbsolute', | ||||
|             isConstrained: false, | ||||
|             value: '12', | ||||
|             sourceRange: [359, 361], | ||||
|             sourceRange: [358, 360, true], | ||||
|             argPosition: { type: 'arrayItem', index: 1 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineToX', | ||||
| @ -819,7 +820,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'angle', | ||||
|             isConstrained: false, | ||||
|             value: '30', | ||||
|             sourceRange: [389, 391], | ||||
|             sourceRange: [388, 390, true], | ||||
|             argPosition: { type: 'arrayItem', index: 0 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineToY', | ||||
| @ -828,7 +829,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'yAbsolute', | ||||
|             isConstrained: false, | ||||
|             value: '10', | ||||
|             sourceRange: [393, 395], | ||||
|             sourceRange: [392, 394, true], | ||||
|             argPosition: { type: 'arrayItem', index: 1 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineToY', | ||||
| @ -839,6 +840,7 @@ describe('testing getConstraintInfo', () => { | ||||
|       const sourceRange: SourceRange = [ | ||||
|         code.indexOf(functionName), | ||||
|         code.indexOf(functionName) + functionName.length, | ||||
|         true, | ||||
|       ] | ||||
|       if (err(ast)) return ast | ||||
|       const pathToNode = getNodePathFromSourceRange(ast, sourceRange) | ||||
| @ -860,7 +862,7 @@ describe('testing getConstraintInfo', () => { | ||||
|     |> lineTo([6.14 + 0, 3.14 + 0], %) | ||||
|     |> xLineTo(8 + 0, %) | ||||
|     |> yLineTo(5 + 0, %) | ||||
|     |> yLine(3.14 + 0, %, 'a') | ||||
|     |> yLine(3.14 + 0, %, $a) | ||||
|     |> xLine(3.14 + 0, %) | ||||
|     |> angledLineOfXLength({ angle = 3.14 + 0, length = 3.14 + 0 }, %) | ||||
|     |> angledLineOfYLength({ angle = 30 + 0, length = 3 + 0 }, %) | ||||
| @ -868,11 +870,11 @@ describe('testing getConstraintInfo', () => { | ||||
|     |> angledLineToY({ angle = 30 + 0, to = 10.14 + 0 }, %) | ||||
|     |> angledLineThatIntersects({ | ||||
|          angle = 3.14 + 0, | ||||
|          intersectTag = 'a', | ||||
|          intersectTag = a, | ||||
|          offset = 0 + 0 | ||||
|        }, %) | ||||
|     |> tangentialArcTo([3.14 + 0, 13.14 + 0], %)` | ||||
|     const ast = parse(code) | ||||
|     const ast = assertParse(code) | ||||
|     test.each([ | ||||
|       [ | ||||
|         'line', | ||||
| @ -881,7 +883,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'xRelative', | ||||
|             isConstrained: true, | ||||
|             value: '3 + 0', | ||||
|             sourceRange: [83, 88], | ||||
|             sourceRange: [83, 88, true], | ||||
|             argPosition: { type: 'arrayItem', index: 0 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'line', | ||||
| @ -890,7 +892,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'yRelative', | ||||
|             isConstrained: true, | ||||
|             value: '4 + 0', | ||||
|             sourceRange: [90, 95], | ||||
|             sourceRange: [90, 95, true], | ||||
|             argPosition: { type: 'arrayItem', index: 1 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'line', | ||||
| @ -904,7 +906,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'angle', | ||||
|             isConstrained: true, | ||||
|             value: '3.14 + 0', | ||||
|             sourceRange: [128, 136], | ||||
|             sourceRange: [129, 137, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'angle' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLine', | ||||
| @ -913,7 +915,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'length', | ||||
|             isConstrained: true, | ||||
|             value: '3.14 + 0', | ||||
|             sourceRange: [146, 154], | ||||
|             sourceRange: [148, 156, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'length' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLine', | ||||
| @ -927,7 +929,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'xAbsolute', | ||||
|             isConstrained: true, | ||||
|             value: '6.14 + 0', | ||||
|             sourceRange: [176, 184], | ||||
|             sourceRange: [178, 186, true], | ||||
|             argPosition: { type: 'arrayItem', index: 0 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'lineTo', | ||||
| @ -936,7 +938,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'yAbsolute', | ||||
|             isConstrained: true, | ||||
|             value: '3.14 + 0', | ||||
|             sourceRange: [186, 194], | ||||
|             sourceRange: [188, 196, true], | ||||
|             argPosition: { type: 'arrayItem', index: 1 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'lineTo', | ||||
| @ -950,7 +952,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'horizontal', | ||||
|             isConstrained: true, | ||||
|             value: 'xLineTo', | ||||
|             sourceRange: [207, 214], | ||||
|             sourceRange: [209, 216, true], | ||||
|             argPosition: undefined, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'xLineTo', | ||||
| @ -959,7 +961,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'xAbsolute', | ||||
|             isConstrained: true, | ||||
|             value: '8 + 0', | ||||
|             sourceRange: [215, 220], | ||||
|             sourceRange: [217, 222, true], | ||||
|             argPosition: { type: 'singleValue' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'xLineTo', | ||||
| @ -973,7 +975,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'vertical', | ||||
|             isConstrained: true, | ||||
|             value: 'yLineTo', | ||||
|             sourceRange: [232, 239], | ||||
|             sourceRange: [234, 241, true], | ||||
|             argPosition: undefined, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'yLineTo', | ||||
| @ -982,7 +984,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'yAbsolute', | ||||
|             isConstrained: true, | ||||
|             value: '5 + 0', | ||||
|             sourceRange: [240, 245], | ||||
|             sourceRange: [242, 247, true], | ||||
|             argPosition: { type: 'singleValue' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'yLineTo', | ||||
| @ -996,7 +998,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'vertical', | ||||
|             isConstrained: true, | ||||
|             value: 'yLine', | ||||
|             sourceRange: [257, 262], | ||||
|             sourceRange: [259, 264, true], | ||||
|             argPosition: undefined, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'yLine', | ||||
| @ -1005,7 +1007,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'yRelative', | ||||
|             isConstrained: true, | ||||
|             value: '3.14 + 0', | ||||
|             sourceRange: [263, 271], | ||||
|             sourceRange: [265, 273, true], | ||||
|             argPosition: { type: 'singleValue' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'yLine', | ||||
| @ -1019,7 +1021,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'horizontal', | ||||
|             isConstrained: true, | ||||
|             value: 'xLine', | ||||
|             sourceRange: [288, 293], | ||||
|             sourceRange: [289, 294, true], | ||||
|             argPosition: undefined, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'xLine', | ||||
| @ -1028,7 +1030,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'xRelative', | ||||
|             isConstrained: true, | ||||
|             value: '3.14 + 0', | ||||
|             sourceRange: [294, 302], | ||||
|             sourceRange: [295, 303, true], | ||||
|             argPosition: { type: 'singleValue' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'xLine', | ||||
| @ -1042,7 +1044,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'angle', | ||||
|             isConstrained: true, | ||||
|             value: '3.14 + 0', | ||||
|             sourceRange: [343, 351], | ||||
|             sourceRange: [345, 353, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'angle' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineOfXLength', | ||||
| @ -1051,7 +1053,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'xRelative', | ||||
|             isConstrained: true, | ||||
|             value: '3.14 + 0', | ||||
|             sourceRange: [361, 369], | ||||
|             sourceRange: [364, 372, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'length' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineOfXLength', | ||||
| @ -1065,7 +1067,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'angle', | ||||
|             isConstrained: true, | ||||
|             value: '30 + 0', | ||||
|             sourceRange: [412, 418], | ||||
|             sourceRange: [416, 422, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'angle' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineOfYLength', | ||||
| @ -1074,7 +1076,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'yRelative', | ||||
|             isConstrained: true, | ||||
|             value: '3 + 0', | ||||
|             sourceRange: [428, 433], | ||||
|             sourceRange: [433, 438, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'length' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineOfYLength', | ||||
| @ -1088,7 +1090,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'angle', | ||||
|             isConstrained: true, | ||||
|             value: '12.14 + 0', | ||||
|             sourceRange: [470, 479], | ||||
|             sourceRange: [476, 485, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'angle' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineToX', | ||||
| @ -1097,7 +1099,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'xAbsolute', | ||||
|             isConstrained: true, | ||||
|             value: '12 + 0', | ||||
|             sourceRange: [485, 491], | ||||
|             sourceRange: [492, 498, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'to' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineToX', | ||||
| @ -1111,7 +1113,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'angle', | ||||
|             isConstrained: true, | ||||
|             value: '30 + 0', | ||||
|             sourceRange: [528, 534], | ||||
|             sourceRange: [536, 542, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'angle' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineToY', | ||||
| @ -1120,7 +1122,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'yAbsolute', | ||||
|             isConstrained: true, | ||||
|             value: '10.14 + 0', | ||||
|             sourceRange: [540, 549], | ||||
|             sourceRange: [549, 558, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'to' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineToY', | ||||
| @ -1134,7 +1136,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'angle', | ||||
|             isConstrained: true, | ||||
|             value: '3.14 + 0', | ||||
|             sourceRange: [606, 614], | ||||
|             sourceRange: [616, 624, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'angle' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineThatIntersects', | ||||
| @ -1143,7 +1145,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'intersectionOffset', | ||||
|             isConstrained: true, | ||||
|             value: '0 + 0', | ||||
|             sourceRange: [661, 666], | ||||
|             sourceRange: [671, 676, true], | ||||
|             argPosition: { type: 'objectProperty', key: 'offset' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineThatIntersects', | ||||
| @ -1151,8 +1153,8 @@ describe('testing getConstraintInfo', () => { | ||||
|           { | ||||
|             type: 'intersectionTag', | ||||
|             isConstrained: false, | ||||
|             value: "'a'", | ||||
|             sourceRange: [639, 642], | ||||
|             value: 'a', | ||||
|             sourceRange: [650, 651, true], | ||||
|             argPosition: { key: 'intersectTag', type: 'objectProperty' }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'angledLineThatIntersects', | ||||
| @ -1166,7 +1168,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'tangentialWithPrevious', | ||||
|             isConstrained: true, | ||||
|             value: 'tangentialArcTo', | ||||
|             sourceRange: [687, 702], | ||||
|             sourceRange: [697, 712, true], | ||||
|             argPosition: undefined, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'tangentialArcTo', | ||||
| @ -1175,7 +1177,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'xAbsolute', | ||||
|             isConstrained: true, | ||||
|             value: '3.14 + 0', | ||||
|             sourceRange: [704, 712], | ||||
|             sourceRange: [714, 722, true], | ||||
|             argPosition: { type: 'arrayItem', index: 0 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'tangentialArcTo', | ||||
| @ -1184,7 +1186,7 @@ describe('testing getConstraintInfo', () => { | ||||
|             type: 'yAbsolute', | ||||
|             isConstrained: true, | ||||
|             value: '13.14 + 0', | ||||
|             sourceRange: [714, 723], | ||||
|             sourceRange: [724, 733, true], | ||||
|             argPosition: { type: 'arrayItem', index: 1 }, | ||||
|             pathToNode: expect.any(Array), | ||||
|             stdLibFnName: 'tangentialArcTo', | ||||
| @ -1195,6 +1197,7 @@ describe('testing getConstraintInfo', () => { | ||||
|       const sourceRange: SourceRange = [ | ||||
|         code.indexOf(functionName), | ||||
|         code.indexOf(functionName) + functionName.length, | ||||
|         true, | ||||
|       ] | ||||
|       if (err(ast)) return ast | ||||
|       const pathToNode = getNodePathFromSourceRange(ast, sourceRange) | ||||
|  | ||||
| @ -222,7 +222,7 @@ const commonConstraintInfoHelper = ( | ||||
|         code.slice(input1.start, input1.end), | ||||
|         stdLibFnName, | ||||
|         isArr ? abbreviatedInputs[0].arrayInput : abbreviatedInputs[0].objInput, | ||||
|         [input1.start, input1.end], | ||||
|         [input1.start, input1.end, true], | ||||
|         pathToFirstArg | ||||
|       ) | ||||
|     ) | ||||
| @ -234,7 +234,7 @@ const commonConstraintInfoHelper = ( | ||||
|         code.slice(input2.start, input2.end), | ||||
|         stdLibFnName, | ||||
|         isArr ? abbreviatedInputs[1].arrayInput : abbreviatedInputs[1].objInput, | ||||
|         [input2.start, input2.end], | ||||
|         [input2.start, input2.end, true], | ||||
|         pathToSecondArg | ||||
|       ) | ||||
|     ) | ||||
| @ -266,7 +266,7 @@ const horzVertConstraintInfoHelper = ( | ||||
|       callee.name, | ||||
|       stdLibFnName, | ||||
|       undefined, | ||||
|       [callee.start, callee.end], | ||||
|       [callee.start, callee.end, true], | ||||
|       pathToCallee | ||||
|     ), | ||||
|     constrainInfo( | ||||
| @ -275,7 +275,7 @@ const horzVertConstraintInfoHelper = ( | ||||
|       code.slice(firstArg.start, firstArg.end), | ||||
|       stdLibFnName, | ||||
|       abbreviatedInput, | ||||
|       [firstArg.start, firstArg.end], | ||||
|       [firstArg.start, firstArg.end, true], | ||||
|       pathToFirstArg | ||||
|     ), | ||||
|   ] | ||||
| @ -905,7 +905,7 @@ export const tangentialArcTo: SketchLineHelper = { | ||||
|         callee.name, | ||||
|         'tangentialArcTo', | ||||
|         undefined, | ||||
|         [callee.start, callee.end], | ||||
|         [callee.start, callee.end, true], | ||||
|         pathToCallee | ||||
|       ), | ||||
|       constrainInfo( | ||||
| @ -914,7 +914,7 @@ export const tangentialArcTo: SketchLineHelper = { | ||||
|         code.slice(firstArg.elements[0].start, firstArg.elements[0].end), | ||||
|         'tangentialArcTo', | ||||
|         0, | ||||
|         [firstArg.elements[0].start, firstArg.elements[0].end], | ||||
|         [firstArg.elements[0].start, firstArg.elements[0].end, true], | ||||
|         pathToFirstArg | ||||
|       ), | ||||
|       constrainInfo( | ||||
| @ -923,7 +923,7 @@ export const tangentialArcTo: SketchLineHelper = { | ||||
|         code.slice(firstArg.elements[1].start, firstArg.elements[1].end), | ||||
|         'tangentialArcTo', | ||||
|         1, | ||||
|         [firstArg.elements[1].start, firstArg.elements[1].end], | ||||
|         [firstArg.elements[1].start, firstArg.elements[1].end, true], | ||||
|         pathToSecondArg | ||||
|       ), | ||||
|     ] | ||||
| @ -1052,7 +1052,7 @@ export const circle: SketchLineHelper = { | ||||
|         code.slice(radiusDetails.expr.start, radiusDetails.expr.end), | ||||
|         'circle', | ||||
|         'radius', | ||||
|         [radiusDetails.expr.start, radiusDetails.expr.end], | ||||
|         [radiusDetails.expr.start, radiusDetails.expr.end, true], | ||||
|         pathToRadiusLiteral | ||||
|       ), | ||||
|       { | ||||
| @ -1064,6 +1064,7 @@ export const circle: SketchLineHelper = { | ||||
|         sourceRange: [ | ||||
|           centerDetails.expr.elements[0].start, | ||||
|           centerDetails.expr.elements[0].end, | ||||
|           true, | ||||
|         ], | ||||
|         pathToNode: pathToXArg, | ||||
|         value: code.slice( | ||||
| @ -1085,6 +1086,7 @@ export const circle: SketchLineHelper = { | ||||
|         sourceRange: [ | ||||
|           centerDetails.expr.elements[1].start, | ||||
|           centerDetails.expr.elements[1].end, | ||||
|           true, | ||||
|         ], | ||||
|         pathToNode: pathToYArg, | ||||
|         value: code.slice( | ||||
| @ -1761,7 +1763,7 @@ export const angledLineThatIntersects: SketchLineHelper = { | ||||
|           code.slice(angle.start, angle.end), | ||||
|           'angledLineThatIntersects', | ||||
|           'angle', | ||||
|           [angle.start, angle.end], | ||||
|           [angle.start, angle.end, true], | ||||
|           pathToAngleProp | ||||
|         ) | ||||
|       ) | ||||
| @ -1780,7 +1782,7 @@ export const angledLineThatIntersects: SketchLineHelper = { | ||||
|           code.slice(offset.start, offset.end), | ||||
|           'angledLineThatIntersects', | ||||
|           'offset', | ||||
|           [offset.start, offset.end], | ||||
|           [offset.start, offset.end, true], | ||||
|           pathToOffsetProp | ||||
|         ) | ||||
|       ) | ||||
| @ -1799,7 +1801,7 @@ export const angledLineThatIntersects: SketchLineHelper = { | ||||
|         code.slice(tag.start, tag.end), | ||||
|         'angledLineThatIntersects', | ||||
|         'intersectTag', | ||||
|         [tag.start, tag.end], | ||||
|         [tag.start, tag.end, true], | ||||
|         pathToTagProp | ||||
|       ) | ||||
|       returnVal.push(info) | ||||
|  | ||||
| @ -1,5 +1,5 @@ | ||||
| import { | ||||
|   parse, | ||||
|   assertParse, | ||||
|   Sketch, | ||||
|   recast, | ||||
|   initPromise, | ||||
| @ -31,12 +31,11 @@ async function testingSwapSketchFnCall({ | ||||
|   constraintType: ConstraintType | ||||
| }): Promise<{ | ||||
|   newCode: string | ||||
|   originalRange: [number, number] | ||||
|   originalRange: [number, number, boolean] | ||||
| }> { | ||||
|   const startIndex = inputCode.indexOf(callToSwap) | ||||
|   const range: SourceRange = [startIndex, startIndex + callToSwap.length] | ||||
|   const ast = parse(inputCode) | ||||
|   if (err(ast)) return Promise.reject(ast) | ||||
|   const range: SourceRange = [startIndex, startIndex + callToSwap.length, true] | ||||
|   const ast = assertParse(inputCode) | ||||
|  | ||||
|   const execState = await enginelessExecutor(ast) | ||||
|   const selections = { | ||||
| @ -370,13 +369,13 @@ part001 = startSketchOn('XY') | ||||
|   |> line([2.14, 1.35], %) // normal-segment | ||||
|   |> xLine(3.54, %)` | ||||
|   it('normal case works', async () => { | ||||
|     const execState = await enginelessExecutor(parse(code)) | ||||
|     const execState = await enginelessExecutor(assertParse(code)) | ||||
|     const index = code.indexOf('// normal-segment') - 7 | ||||
|     const sg = sketchFromKclValue( | ||||
|       execState.memory.get('part001'), | ||||
|       'part001' | ||||
|     ) as Sketch | ||||
|     const _segment = getSketchSegmentFromSourceRange(sg, [index, index]) | ||||
|     const _segment = getSketchSegmentFromSourceRange(sg, [index, index, true]) | ||||
|     if (err(_segment)) throw _segment | ||||
|     const { __geoMeta, ...segment } = _segment.segment | ||||
|     expect(segment).toEqual({ | ||||
| @ -387,11 +386,11 @@ part001 = startSketchOn('XY') | ||||
|     }) | ||||
|   }) | ||||
|   it('verify it works when the segment is in the `start` property', async () => { | ||||
|     const execState = await enginelessExecutor(parse(code)) | ||||
|     const execState = await enginelessExecutor(assertParse(code)) | ||||
|     const index = code.indexOf('// segment-in-start') - 7 | ||||
|     const _segment = getSketchSegmentFromSourceRange( | ||||
|       sketchFromKclValue(execState.memory.get('part001'), 'part001') as Sketch, | ||||
|       [index, index] | ||||
|       [index, index, true] | ||||
|     ) | ||||
|     if (err(_segment)) throw _segment | ||||
|     const { __geoMeta, ...segment } = _segment.segment | ||||
|  | ||||
| @ -31,7 +31,7 @@ export function getSketchSegmentFromPathToNode( | ||||
|   const node = nodeMeta.node | ||||
|   if (!node || typeof node.start !== 'number' || !node.end) | ||||
|     return new Error('no node found') | ||||
|   const sourceRange: SourceRange = [node.start, node.end] | ||||
|   const sourceRange: SourceRange = [node.start, node.end, true] | ||||
|   return getSketchSegmentFromSourceRange(sketch, sourceRange) | ||||
| } | ||||
| export function getSketchSegmentFromSourceRange( | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| import { parse, Expr, recast, initPromise, Program } from '../wasm' | ||||
| import { assertParse, Expr, recast, initPromise, Program } from '../wasm' | ||||
| import { | ||||
|   getConstraintType, | ||||
|   getTransformInfos, | ||||
| @ -66,8 +66,7 @@ describe('testing getConstraintType', () => { | ||||
| function getConstraintTypeFromSourceHelper( | ||||
|   code: string | ||||
| ): ReturnType<typeof getConstraintType> | Error { | ||||
|   const ast = parse(code) | ||||
|   if (err(ast)) return ast | ||||
|   const ast = assertParse(code) | ||||
|  | ||||
|   const args = (ast.body[0] as any).expression.arguments[0].elements as [ | ||||
|     Expr, | ||||
| @ -79,8 +78,7 @@ function getConstraintTypeFromSourceHelper( | ||||
| function getConstraintTypeFromSourceHelper2( | ||||
|   code: string | ||||
| ): ReturnType<typeof getConstraintType> | Error { | ||||
|   const ast = parse(code) | ||||
|   if (err(ast)) return ast | ||||
|   const ast = assertParse(code) | ||||
|  | ||||
|   const arg = (ast.body[0] as any).expression.arguments[0] as Expr | ||||
|   const fnName = (ast.body[0] as any).expression.callee.name as ToolTip | ||||
| @ -127,7 +125,7 @@ describe('testing transformAstForSketchLines for equal length constraint', () => | ||||
|         ) | ||||
|       } | ||||
|       const start = codeBeforeLine + line.indexOf('|> ' + 5) | ||||
|       const range: [number, number] = [start, start] | ||||
|       const range: [number, number, boolean] = [start, start, true] | ||||
|       return { | ||||
|         codeRef: codeRefFromRange(range, ast), | ||||
|       } | ||||
| @ -137,8 +135,7 @@ describe('testing transformAstForSketchLines for equal length constraint', () => | ||||
|       inputCode: string, | ||||
|       selectionRanges: Selections['graphSelections'] | ||||
|     ) { | ||||
|       const ast = parse(inputCode) | ||||
|       if (err(ast)) return Promise.reject(ast) | ||||
|       const ast = assertParse(inputCode) | ||||
|       const execState = await enginelessExecutor(ast) | ||||
|       const transformInfos = getTransformInfos( | ||||
|         makeSelections(selectionRanges.slice(1)), | ||||
| @ -161,8 +158,7 @@ describe('testing transformAstForSketchLines for equal length constraint', () => | ||||
|     } | ||||
|  | ||||
|     it(`Should reorder when user selects first-to-last`, async () => { | ||||
|       const ast = parse(inputScript) | ||||
|       if (err(ast)) return Promise.reject(ast) | ||||
|       const ast = assertParse(inputScript) | ||||
|       const selectionRanges: Selections['graphSelections'] = [ | ||||
|         selectLine(inputScript, 3, ast), | ||||
|         selectLine(inputScript, 4, ast), | ||||
| @ -173,8 +169,7 @@ describe('testing transformAstForSketchLines for equal length constraint', () => | ||||
|     }) | ||||
|  | ||||
|     it(`Should reorder when user selects last-to-first`, async () => { | ||||
|       const ast = parse(inputScript) | ||||
|       if (err(ast)) return Promise.reject(ast) | ||||
|       const ast = assertParse(inputScript) | ||||
|       const selectionRanges: Selections['graphSelections'] = [ | ||||
|         selectLine(inputScript, 4, ast), | ||||
|         selectLine(inputScript, 3, ast), | ||||
| @ -293,8 +288,7 @@ part001 = startSketchOn('XY') | ||||
|   |> yLine(segLen(seg01), %) // ln-yLineTo-free should convert to yLine | ||||
| ` | ||||
|   it('should transform the ast', async () => { | ||||
|     const ast = parse(inputScript) | ||||
|     if (err(ast)) return Promise.reject(ast) | ||||
|     const ast = assertParse(inputScript) | ||||
|  | ||||
|     const selectionRanges: Selections['graphSelections'] = inputScript | ||||
|       .split('\n') | ||||
| @ -303,7 +297,7 @@ part001 = startSketchOn('XY') | ||||
|         const comment = ln.split('//')[1] | ||||
|         const start = inputScript.indexOf('//' + comment) - 7 | ||||
|         return { | ||||
|           codeRef: codeRefFromRange([start, start], ast), | ||||
|           codeRef: codeRefFromRange([start, start, true], ast), | ||||
|         } | ||||
|       }) | ||||
|  | ||||
| @ -383,8 +377,7 @@ part001 = startSketchOn('XY') | ||||
|   |> xLineTo(myVar3, %) // select for horizontal constraint 10 | ||||
|   |> angledLineToY([301, myVar], %) // select for vertical constraint 10 | ||||
| ` | ||||
|     const ast = parse(inputScript) | ||||
|     if (err(ast)) return Promise.reject(ast) | ||||
|     const ast = assertParse(inputScript) | ||||
|  | ||||
|     const selectionRanges: Selections['graphSelections'] = inputScript | ||||
|       .split('\n') | ||||
| @ -393,7 +386,7 @@ part001 = startSketchOn('XY') | ||||
|         const comment = ln.split('//')[1] | ||||
|         const start = inputScript.indexOf('//' + comment) - 7 | ||||
|         return { | ||||
|           codeRef: codeRefFromRange([start, start], ast), | ||||
|           codeRef: codeRefFromRange([start, start, true], ast), | ||||
|         } | ||||
|       }) | ||||
|  | ||||
| @ -444,8 +437,7 @@ part001 = startSketchOn('XY') | ||||
|   |> angledLineToX([333, myVar3], %) // select for horizontal constraint 10 | ||||
|   |> yLineTo(myVar, %) // select for vertical constraint 10 | ||||
| ` | ||||
|     const ast = parse(inputScript) | ||||
|     if (err(ast)) return Promise.reject(ast) | ||||
|     const ast = assertParse(inputScript) | ||||
|  | ||||
|     const selectionRanges: Selections['graphSelections'] = inputScript | ||||
|       .split('\n') | ||||
| @ -454,7 +446,7 @@ part001 = startSketchOn('XY') | ||||
|         const comment = ln.split('//')[1] | ||||
|         const start = inputScript.indexOf('//' + comment) - 7 | ||||
|         return { | ||||
|           codeRef: codeRefFromRange([start, start], ast), | ||||
|           codeRef: codeRefFromRange([start, start, true], ast), | ||||
|         } | ||||
|       }) | ||||
|  | ||||
| @ -538,8 +530,7 @@ async function helperThing( | ||||
|   linesOfInterest: string[], | ||||
|   constraint: ConstraintType | ||||
| ): Promise<string> { | ||||
|   const ast = parse(inputScript) | ||||
|   if (err(ast)) return Promise.reject(ast) | ||||
|   const ast = assertParse(inputScript) | ||||
|  | ||||
|   const selectionRanges: Selections['graphSelections'] = inputScript | ||||
|     .split('\n') | ||||
| @ -550,7 +541,7 @@ async function helperThing( | ||||
|       const comment = ln.split('//')[1] | ||||
|       const start = inputScript.indexOf('//' + comment) - 7 | ||||
|       return { | ||||
|         codeRef: codeRefFromRange([start, start], ast), | ||||
|         codeRef: codeRefFromRange([start, start, true], ast), | ||||
|       } | ||||
|     }) | ||||
|  | ||||
| @ -606,7 +597,7 @@ part001 = startSketchOn('XY') | ||||
|   |> line([-1.49, 1.06], %) // free | ||||
|   |> xLine(-3.43 + 0, %) // full | ||||
|   |> angledLineOfXLength([243 + 0, 1.2 + 0], %) // full` | ||||
|     const ast = parse(code) | ||||
|     const ast = assertParse(code) | ||||
|     const constraintLevels: ConstraintLevel[] = ['full', 'partial', 'free'] | ||||
|     constraintLevels.forEach((constraintLevel) => { | ||||
|       const recursivelySearchCommentsAndCheckConstraintLevel = ( | ||||
| @ -619,7 +610,7 @@ part001 = startSketchOn('XY') | ||||
|         } | ||||
|         const offsetIndex = index - 7 | ||||
|         const expectedConstraintLevel = getConstraintLevelFromSourceRange( | ||||
|           [offsetIndex, offsetIndex], | ||||
|           [offsetIndex, offsetIndex, true], | ||||
|           ast | ||||
|         ) | ||||
|         if (err(expectedConstraintLevel)) { | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| import { parse, initPromise } from '../wasm' | ||||
| import { assertParse, initPromise } from '../wasm' | ||||
| import { enginelessExecutor } from '../../lib/testHelpers' | ||||
|  | ||||
| beforeAll(async () => { | ||||
| @ -17,9 +17,9 @@ describe('testing angledLineThatIntersects', () => { | ||||
|   offset: ${offset}, | ||||
| }, %, $yo2) | ||||
| intersect = segEndX(yo2)` | ||||
|     const execState = await enginelessExecutor(parse(code('-1'))) | ||||
|     const execState = await enginelessExecutor(assertParse(code('-1'))) | ||||
|     expect(execState.memory.get('intersect')?.value).toBe(1 + Math.sqrt(2)) | ||||
|     const noOffset = await enginelessExecutor(parse(code('0'))) | ||||
|     const noOffset = await enginelessExecutor(assertParse(code('0'))) | ||||
|     expect(noOffset.memory.get('intersect')?.value).toBeCloseTo(1) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| @ -1,13 +1,18 @@ | ||||
| import { err } from 'lib/trap' | ||||
| import { parse } from './wasm' | ||||
| import { 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' | ||||
|  | ||||
| it('can execute parsed AST', async () => { | ||||
|   const code = `x = 1 | ||||
| // A comment.` | ||||
|   const ast = parse(code) | ||||
|   expect(err(ast)).toEqual(false) | ||||
|   const execState = await enginelessExecutor(ast) | ||||
|   expect(err(ast)).toEqual(false) | ||||
|   const result = parse(code) | ||||
|   expect(err(result)).toEqual(false) | ||||
|   const pResult = result as ParseResult | ||||
|   expect(pResult.errors.length).toEqual(0) | ||||
|   expect(pResult.program).not.toEqual(null) | ||||
|   const execState = await enginelessExecutor(pResult.program as Node<Program>) | ||||
|   expect(err(execState)).toEqual(false) | ||||
|   expect(execState.memory.get('x')?.value).toEqual(1) | ||||
| }) | ||||
|  | ||||
							
								
								
									
										138
									
								
								src/lang/wasm.ts
									
									
									
									
									
								
							
							
						
						| @ -35,12 +35,13 @@ import { Configuration } from 'wasm-lib/kcl/bindings/Configuration' | ||||
| import { DeepPartial } from 'lib/types' | ||||
| import { ProjectConfiguration } from 'wasm-lib/kcl/bindings/ProjectConfiguration' | ||||
| import { Sketch } from '../wasm-lib/kcl/bindings/Sketch' | ||||
| import { IdGenerator } from 'wasm-lib/kcl/bindings/IdGenerator' | ||||
| import { ExecState as RawExecState } from '../wasm-lib/kcl/bindings/ExecState' | ||||
| import { ProgramMemory as RawProgramMemory } from '../wasm-lib/kcl/bindings/ProgramMemory' | ||||
| import { EnvironmentRef } from '../wasm-lib/kcl/bindings/EnvironmentRef' | ||||
| import { Environment } from '../wasm-lib/kcl/bindings/Environment' | ||||
| import { Node } from 'wasm-lib/kcl/bindings/Node' | ||||
| import { CompilationError } from 'wasm-lib/kcl/bindings/CompilationError' | ||||
| import { SourceRange as RustSourceRange } from 'wasm-lib/kcl/bindings/SourceRange' | ||||
|  | ||||
| export type { Program } from '../wasm-lib/kcl/bindings/Program' | ||||
| export type { Expr } from '../wasm-lib/kcl/bindings/Expr' | ||||
| @ -84,13 +85,22 @@ export type SyntaxType = | ||||
|   | 'NonCodeNode' | ||||
|   | 'UnaryExpression' | ||||
|  | ||||
| export type { SourceRange } from '../wasm-lib/kcl/bindings/SourceRange' | ||||
| export type { Path } from '../wasm-lib/kcl/bindings/Path' | ||||
| export type { Sketch } from '../wasm-lib/kcl/bindings/Sketch' | ||||
| export type { Solid } from '../wasm-lib/kcl/bindings/Solid' | ||||
| export type { KclValue } from '../wasm-lib/kcl/bindings/KclValue' | ||||
| export type { ExtrudeSurface } from '../wasm-lib/kcl/bindings/ExtrudeSurface' | ||||
|  | ||||
| export type SourceRange = [number, number, boolean] | ||||
|  | ||||
| export function sourceRangeFromRust(s: RustSourceRange): SourceRange { | ||||
|   return [s[0], s[1], s[2] === 0] | ||||
| } | ||||
|  | ||||
| export function defaultSourceRange(): SourceRange { | ||||
|   return [0, 0, true] | ||||
| } | ||||
|  | ||||
| export const wasmUrl = () => { | ||||
|   // For when we're in electron (file based) or web server (network based) | ||||
|   // For some reason relative paths don't work as expected. Otherwise we would | ||||
| @ -120,26 +130,81 @@ const initialise = async () => { | ||||
|  | ||||
| export const initPromise = initialise() | ||||
|  | ||||
| export const rangeTypeFix = (ranges: number[][]): [number, number, number][] => | ||||
|   ranges.map(([start, end, moduleId]) => [start, end, moduleId]) | ||||
| const splitErrors = ( | ||||
|   input: CompilationError[] | ||||
| ): { errors: CompilationError[]; warnings: CompilationError[] } => { | ||||
|   let errors = [] | ||||
|   let warnings = [] | ||||
|   for (const i of input) { | ||||
|     if (i.severity === 'Warning') { | ||||
|       warnings.push(i) | ||||
|     } else { | ||||
|       errors.push(i) | ||||
|     } | ||||
|   } | ||||
|  | ||||
| export const parse = (code: string | Error): Node<Program> | Error => { | ||||
|   return { errors, warnings } | ||||
| } | ||||
|  | ||||
| export class ParseResult { | ||||
|   program: Node<Program> | null | ||||
|   errors: CompilationError[] | ||||
|   warnings: CompilationError[] | ||||
|  | ||||
|   constructor( | ||||
|     program: Node<Program> | null, | ||||
|     errors: CompilationError[], | ||||
|     warnings: CompilationError[] | ||||
|   ) { | ||||
|     this.program = program | ||||
|     this.errors = errors | ||||
|     this.warnings = warnings | ||||
|   } | ||||
| } | ||||
|  | ||||
| class SuccessParseResult extends ParseResult { | ||||
|   program: Node<Program> | ||||
|  | ||||
|   constructor( | ||||
|     program: Node<Program>, | ||||
|     errors: CompilationError[], | ||||
|     warnings: CompilationError[] | ||||
|   ) { | ||||
|     super(program, errors, warnings) | ||||
|     this.program = program | ||||
|   } | ||||
| } | ||||
|  | ||||
| export function resultIsOk(result: ParseResult): result is SuccessParseResult { | ||||
|   return !!result.program && result.errors.length === 0 | ||||
| } | ||||
|  | ||||
| export const parse = (code: string | Error): ParseResult | Error => { | ||||
|   if (err(code)) return code | ||||
|  | ||||
|   try { | ||||
|     const program: Node<Program> = parse_wasm(code) | ||||
|     return program | ||||
|     const parsed: [Node<Program>, CompilationError[]] = parse_wasm(code) | ||||
|     let errs = splitErrors(parsed[1]) | ||||
|     return new ParseResult(parsed[0], errs.errors, errs.warnings) | ||||
|   } catch (e: any) { | ||||
|     // throw e | ||||
|     const parsed: RustKclError = JSON.parse(e.toString()) | ||||
|     return new KCLError( | ||||
|       parsed.kind, | ||||
|       parsed.msg, | ||||
|       rangeTypeFix(parsed.sourceRanges) | ||||
|       sourceRangeFromRust(parsed.sourceRanges[0]) | ||||
|     ) | ||||
|   } | ||||
| } | ||||
|  | ||||
| // Parse and throw an exception if there are any errors (probably not suitable for use outside of testing). | ||||
| export const assertParse = (code: string): Node<Program> => { | ||||
|   const result = parse(code) | ||||
|   // eslint-disable-next-line suggest-no-throw/suggest-no-throw | ||||
|   if (err(result) || !resultIsOk(result)) throw result | ||||
|   return result.program | ||||
| } | ||||
|  | ||||
| export type PathToNode = [string | number, string][] | ||||
|  | ||||
| export const isPathToNodeNumber = ( | ||||
| @ -150,7 +215,6 @@ export const isPathToNodeNumber = ( | ||||
|  | ||||
| export interface ExecState { | ||||
|   memory: ProgramMemory | ||||
|   idGenerator: IdGenerator | ||||
| } | ||||
|  | ||||
| /** | ||||
| @ -160,21 +224,12 @@ export interface ExecState { | ||||
| export function emptyExecState(): ExecState { | ||||
|   return { | ||||
|     memory: ProgramMemory.empty(), | ||||
|     idGenerator: defaultIdGenerator(), | ||||
|   } | ||||
| } | ||||
|  | ||||
| function execStateFromRaw(raw: RawExecState): ExecState { | ||||
|   return { | ||||
|     memory: ProgramMemory.fromRaw(raw.memory), | ||||
|     idGenerator: raw.idGenerator, | ||||
|   } | ||||
| } | ||||
|  | ||||
| export function defaultIdGenerator(): IdGenerator { | ||||
|   return { | ||||
|     nextId: 0, | ||||
|     ids: [], | ||||
|   } | ||||
| } | ||||
|  | ||||
| @ -188,6 +243,19 @@ function emptyEnvironment(): Environment { | ||||
|   return { bindings: {}, parent: null } | ||||
| } | ||||
|  | ||||
| function emptyRootEnvironment(): Environment { | ||||
|   return { | ||||
|     // This is dumb this is copied from rust. | ||||
|     bindings: { | ||||
|       ZERO: { type: 'Number', value: 0.0, __meta: [] }, | ||||
|       QUARTER_TURN: { type: 'Number', value: 90.0, __meta: [] }, | ||||
|       HALF_TURN: { type: 'Number', value: 180.0, __meta: [] }, | ||||
|       THREE_QUARTER_TURN: { type: 'Number', value: 270.0, __meta: [] }, | ||||
|     }, | ||||
|     parent: null, | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This duplicates logic in Rust.  The hope is to keep ProgramMemory internals | ||||
|  * isolated from the rest of the TypeScript code so that we can move it to Rust | ||||
| @ -210,7 +278,7 @@ export class ProgramMemory { | ||||
|   } | ||||
|  | ||||
|   constructor( | ||||
|     environments: Environment[] = [emptyEnvironment()], | ||||
|     environments: Environment[] = [emptyRootEnvironment()], | ||||
|     currentEnv: EnvironmentRef = ROOT_ENVIRONMENT_REF, | ||||
|     returnVal: KclValue | null = null | ||||
|   ) { | ||||
| @ -397,36 +465,31 @@ export function sketchFromKclValue( | ||||
|  | ||||
| export const executor = async ( | ||||
|   node: Node<Program>, | ||||
|   programMemory: ProgramMemory | Error = ProgramMemory.empty(), | ||||
|   idGenerator: IdGenerator = defaultIdGenerator(), | ||||
|   engineCommandManager: EngineCommandManager, | ||||
|   isMock: boolean = false | ||||
|   programMemoryOverride: ProgramMemory | Error | null = null | ||||
| ): Promise<ExecState> => { | ||||
|   if (err(programMemory)) return Promise.reject(programMemory) | ||||
|   if (programMemoryOverride !== null && err(programMemoryOverride)) | ||||
|     return Promise.reject(programMemoryOverride) | ||||
|  | ||||
|   // eslint-disable-next-line @typescript-eslint/no-floating-promises | ||||
|   engineCommandManager.startNewSession() | ||||
|   const _programMemory = await _executor( | ||||
|     node, | ||||
|     programMemory, | ||||
|     idGenerator, | ||||
|     engineCommandManager, | ||||
|     isMock | ||||
|     programMemoryOverride | ||||
|   ) | ||||
|   await engineCommandManager.waitForAllCommands() | ||||
|  | ||||
|   engineCommandManager.endSession() | ||||
|   return _programMemory | ||||
| } | ||||
|  | ||||
| export const _executor = async ( | ||||
|   node: Node<Program>, | ||||
|   programMemory: ProgramMemory | Error = ProgramMemory.empty(), | ||||
|   idGenerator: IdGenerator = defaultIdGenerator(), | ||||
|   engineCommandManager: EngineCommandManager, | ||||
|   isMock: boolean | ||||
|   programMemoryOverride: ProgramMemory | Error | null = null | ||||
| ): Promise<ExecState> => { | ||||
|   if (err(programMemory)) return Promise.reject(programMemory) | ||||
|   if (programMemoryOverride !== null && err(programMemoryOverride)) | ||||
|     return Promise.reject(programMemoryOverride) | ||||
|  | ||||
|   try { | ||||
|     let baseUnit = 'mm' | ||||
| @ -439,13 +502,10 @@ export const _executor = async ( | ||||
|     } | ||||
|     const execState: RawExecState = await execute_wasm( | ||||
|       JSON.stringify(node), | ||||
|       JSON.stringify(programMemory.toRaw()), | ||||
|       JSON.stringify(idGenerator), | ||||
|       JSON.stringify(programMemoryOverride?.toRaw() || null), | ||||
|       baseUnit, | ||||
|       engineCommandManager, | ||||
|       fileSystemManager, | ||||
|       undefined, | ||||
|       isMock | ||||
|       fileSystemManager | ||||
|     ) | ||||
|     return execStateFromRaw(execState) | ||||
|   } catch (e: any) { | ||||
| @ -454,7 +514,7 @@ export const _executor = async ( | ||||
|     const kclError = new KCLError( | ||||
|       parsed.kind, | ||||
|       parsed.msg, | ||||
|       rangeTypeFix(parsed.sourceRanges) | ||||
|       sourceRangeFromRust(parsed.sourceRanges[0]) | ||||
|     ) | ||||
|  | ||||
|     return Promise.reject(kclError) | ||||
| @ -527,7 +587,7 @@ export const modifyAstForSketch = async ( | ||||
|     const kclError = new KCLError( | ||||
|       parsed.kind, | ||||
|       parsed.msg, | ||||
|       rangeTypeFix(parsed.sourceRanges) | ||||
|       sourceRangeFromRust(parsed.sourceRanges[0]) | ||||
|     ) | ||||
|  | ||||
|     console.log(kclError) | ||||
| @ -595,7 +655,7 @@ export function programMemoryInit(): ProgramMemory | Error { | ||||
|     return new KCLError( | ||||
|       parsed.kind, | ||||
|       parsed.msg, | ||||
|       rangeTypeFix(parsed.sourceRanges) | ||||
|       sourceRangeFromRust(parsed.sourceRanges[0]) | ||||
|     ) | ||||
|   } | ||||
| } | ||||
|  | ||||
							
								
								
									
										12
									
								
								src/lib/appVersion.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,12 @@ | ||||
| import { NODE_ENV } from 'env' | ||||
| import { isDesktop } from './isDesktop' | ||||
| import { isTestEnv } from './isTestEnv' | ||||
|  | ||||
| /** Version number of the app */ | ||||
| export const APP_VERSION = | ||||
|   isTestEnv && NODE_ENV === 'development' | ||||
|     ? '11.22.33' | ||||
|     : isDesktop() | ||||
|     ? // @ts-ignore | ||||
|       window.electron.packageJson.version | ||||
|     : 'main' | ||||
| @ -31,6 +31,9 @@ export type ModelingCommandSchema = { | ||||
|     // result: (typeof EXTRUSION_RESULTS)[number] | ||||
|     distance: KclCommandValue | ||||
|   } | ||||
|   Loft: { | ||||
|     selection: Selections | ||||
|   } | ||||
|   Revolve: { | ||||
|     selection: Selections | ||||
|     angle: KclCommandValue | ||||
| @ -260,6 +263,20 @@ export const modelingMachineCommandConfig: StateMachineCommandSetConfig< | ||||
|       }, | ||||
|     }, | ||||
|   }, | ||||
|   Loft: { | ||||
|     description: 'Create a 3D body by blending between two or more sketches', | ||||
|     icon: 'loft', | ||||
|     needsReview: true, | ||||
|     args: { | ||||
|       selection: { | ||||
|         inputType: 'selection', | ||||
|         selectionTypes: ['solid2D'], | ||||
|         multiple: true, | ||||
|         required: true, | ||||
|         skip: false, | ||||
|       }, | ||||
|     }, | ||||
|   }, | ||||
|   // 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.', | ||||
|  | ||||
| @ -52,6 +52,7 @@ export const ONBOARDING_PROJECT_NAME = 'Tutorial Project $nn' | ||||
| export const KCL_DEFAULT_CONSTANT_PREFIXES = { | ||||
|   SKETCH: 'sketch', | ||||
|   EXTRUDE: 'extrude', | ||||
|   LOFT: 'loft', | ||||
|   SEGMENT: 'seg', | ||||
|   REVOLVE: 'revolve', | ||||
|   PLANE: 'plane', | ||||
|  | ||||
| @ -2,7 +2,7 @@ import { CommandLog, EngineCommandManager } from 'lang/std/engineConnection' | ||||
| import { WebrtcStats } from 'wasm-lib/kcl/bindings/WebrtcStats' | ||||
| import { OsInfo } from 'wasm-lib/kcl/bindings/OsInfo' | ||||
| import { isDesktop } from 'lib/isDesktop' | ||||
| import { APP_VERSION } from 'routes/Settings' | ||||
| import { APP_VERSION } from 'lib/appVersion' | ||||
| import { UAParser } from 'ua-parser-js' | ||||
| import screenshot from 'lib/screenshot' | ||||
| import { VITE_KC_API_BASE_URL } from 'env' | ||||
|  | ||||
| @ -116,8 +116,8 @@ export async function createAndOpenNewTutorialProject({ | ||||
|   ) => void | ||||
|   navigate: (path: string) => void | ||||
| }) { | ||||
|   // Clear the scene and end the session. | ||||
|   engineCommandManager.endSession() | ||||
|   // Clear the scene. | ||||
|   engineCommandManager.clearScene() | ||||
|  | ||||
|   // Create a new project with the onboarding project name | ||||
|   const configuration = await readAppSettingsFile() | ||||
|  | ||||
| @ -30,27 +30,27 @@ const bracketLeg1Sketch = startSketchOn('XY') | ||||
|   |> line([-shelfMountL + filletRadius, 0], %) | ||||
|   |> close(%) | ||||
|   |> hole(circle({ | ||||
|        center: [1, 1], | ||||
|        radius: mountingHoleDiameter / 2 | ||||
|        center = [1, 1], | ||||
|        radius = mountingHoleDiameter / 2 | ||||
|      }, %), %) | ||||
|   |> hole(circle({ | ||||
|        center: [shelfMountL - 1.5, width - 1], | ||||
|        radius: mountingHoleDiameter / 2 | ||||
|        center = [shelfMountL - 1.5, width - 1], | ||||
|        radius = mountingHoleDiameter / 2 | ||||
|      }, %), %) | ||||
|   |> hole(circle({ | ||||
|        center: [1, width - 1], | ||||
|        radius: mountingHoleDiameter / 2 | ||||
|        center = [1, width - 1], | ||||
|        radius = mountingHoleDiameter / 2 | ||||
|      }, %), %) | ||||
|   |> hole(circle({ | ||||
|        center: [shelfMountL - 1.5, 1], | ||||
|        radius: mountingHoleDiameter / 2 | ||||
|        center = [shelfMountL - 1.5, 1], | ||||
|        radius = mountingHoleDiameter / 2 | ||||
|      }, %), %) | ||||
|  | ||||
| // Extrude the leg 2 bracket sketch | ||||
| const bracketLeg1Extrude = extrude(thickness, bracketLeg1Sketch) | ||||
|   |> fillet({ | ||||
|        radius: extFilletRadius, | ||||
|        tags: [ | ||||
|        radius = extFilletRadius, | ||||
|        tags = [ | ||||
|          getNextAdjacentEdge(fillet1), | ||||
|          getNextAdjacentEdge(fillet2) | ||||
|        ] | ||||
| @ -61,15 +61,15 @@ const filletSketch = startSketchOn('XZ') | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line([0, thickness], %) | ||||
|   |> arc({ | ||||
|        angleEnd: 180, | ||||
|        angleStart: 90, | ||||
|        radius: filletRadius + thickness | ||||
|        angleEnd = 180, | ||||
|        angleStart = 90, | ||||
|        radius = filletRadius + thickness | ||||
|      }, %) | ||||
|   |> line([thickness, 0], %) | ||||
|   |> arc({ | ||||
|        angleEnd: 90, | ||||
|        angleStart: 180, | ||||
|        radius: filletRadius | ||||
|        angleEnd = 90, | ||||
|        angleStart = 180, | ||||
|        radius = filletRadius | ||||
|      }, %) | ||||
|  | ||||
| // Sketch the bend | ||||
| @ -77,11 +77,11 @@ const filletExtrude = extrude(-width, filletSketch) | ||||
|  | ||||
| // Create a custom plane for the leg that sits on the wall | ||||
| const customPlane = { | ||||
|   plane: { | ||||
|     origin: { x: -filletRadius, y: 0, z: 0 }, | ||||
|     xAxis: { x: 0, y: 1, z: 0 }, | ||||
|     yAxis: { x: 0, y: 0, z: 1 }, | ||||
|     zAxis: { x: 1, y: 0, z: 0 } | ||||
|   plane = { | ||||
|     origin = { x = -filletRadius, y = 0, z = 0 }, | ||||
|     xAxis = { x = 0, y = 1, z = 0 }, | ||||
|     yAxis = { x = 0, y = 0, z = 1 }, | ||||
|     zAxis = { x = 1, y = 0, z = 0 } | ||||
|   } | ||||
| } | ||||
|  | ||||
| @ -93,19 +93,19 @@ const bracketLeg2Sketch = startSketchOn(customPlane) | ||||
|   |> line([-width, 0], %, $fillet4) | ||||
|   |> close(%) | ||||
|   |> hole(circle({ | ||||
|        center: [1, -1.5], | ||||
|        radius: mountingHoleDiameter / 2 | ||||
|        center = [1, -1.5], | ||||
|        radius = mountingHoleDiameter / 2 | ||||
|      }, %), %) | ||||
|   |> hole(circle({ | ||||
|        center: [5, -1.5], | ||||
|        radius: mountingHoleDiameter / 2 | ||||
|        center = [5, -1.5], | ||||
|        radius = mountingHoleDiameter / 2 | ||||
|      }, %), %) | ||||
|  | ||||
| // Extrude the second leg | ||||
| const bracketLeg2Extrude = extrude(-thickness, bracketLeg2Sketch) | ||||
|   |> fillet({ | ||||
|        radius: extFilletRadius, | ||||
|        tags: [ | ||||
|        radius = extFilletRadius, | ||||
|        tags = [ | ||||
|          getNextAdjacentEdge(fillet3), | ||||
|          getNextAdjacentEdge(fillet4) | ||||
|        ] | ||||
|  | ||||
							
								
								
									
										4
									
								
								src/lib/isTestEnv.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,4 @@ | ||||
| import { IS_PLAYWRIGHT_KEY } from '../../e2e/playwright/storageStates' | ||||
|  | ||||
| export const isTestEnv = | ||||
|   globalThis.window?.localStorage.getItem(IS_PLAYWRIGHT_KEY) === 'true' | ||||
| @ -5,7 +5,12 @@ import { | ||||
|   kclManager, | ||||
|   sceneEntitiesManager, | ||||
| } from 'lib/singletons' | ||||
| import { CallExpression, SourceRange, Expr } from 'lang/wasm' | ||||
| import { | ||||
|   CallExpression, | ||||
|   SourceRange, | ||||
|   Expr, | ||||
|   defaultSourceRange, | ||||
| } from 'lang/wasm' | ||||
| import { ModelingMachineEvent } from 'machines/modelingMachine' | ||||
| import { isNonNullable, uuidv4 } from 'lib/utils' | ||||
| import { EditorSelection, SelectionRange } from '@codemirror/state' | ||||
| @ -266,7 +271,7 @@ export function getEventForSegmentSelection( | ||||
|         selectionType: 'singleCodeCursor', | ||||
|         selection: { | ||||
|           codeRef: { | ||||
|             range: [node.node.start, node.node.end], | ||||
|             range: [node.node.start, node.node.end, true], | ||||
|             pathToNode: group.userData.pathToNode, | ||||
|           }, | ||||
|         }, | ||||
| @ -309,10 +314,11 @@ export function handleSelectionBatch({ | ||||
|       selectionToEngine.push({ | ||||
|         type: 'default', | ||||
|         id: artifact?.id, | ||||
|         range: getCodeRefsByArtifactId( | ||||
|           artifact.id, | ||||
|           engineCommandManager.artifactGraph | ||||
|         )?.[0].range || [0, 0], | ||||
|         range: | ||||
|           getCodeRefsByArtifactId( | ||||
|             artifact.id, | ||||
|             engineCommandManager.artifactGraph | ||||
|           )?.[0].range || defaultSourceRange(), | ||||
|       }) | ||||
|   }) | ||||
|   const engineEvents: Models['WebSocketRequest_type'][] = | ||||
| @ -376,10 +382,10 @@ export function processCodeMirrorRanges({ | ||||
|   if (!isChange) return null | ||||
|   const codeBasedSelections: Selections['graphSelections'] = | ||||
|     codeMirrorRanges.map(({ from, to }) => { | ||||
|       const pathToNode = getNodePathFromSourceRange(ast, [from, to]) | ||||
|       const pathToNode = getNodePathFromSourceRange(ast, [from, to, true]) | ||||
|       return { | ||||
|         codeRef: { | ||||
|           range: [from, to], | ||||
|           range: [from, to, true], | ||||
|           pathToNode, | ||||
|         }, | ||||
|       } | ||||
| @ -442,7 +448,7 @@ function updateSceneObjectColors(codeBasedSelections: Selection[]) { | ||||
|     if (err(nodeMeta)) return | ||||
|     const node = nodeMeta.node | ||||
|     const groupHasCursor = codeBasedSelections.some((selection) => { | ||||
|       return isOverlap(selection?.codeRef?.range, [node.start, node.end]) | ||||
|       return isOverlap(selection?.codeRef?.range, [node.start, node.end, true]) | ||||
|     }) | ||||
|  | ||||
|     const color = groupHasCursor | ||||
| @ -529,6 +535,10 @@ function nodeHasExtrude(node: CommonASTNode) { | ||||
|     doesPipeHaveCallExp({ | ||||
|       calleeName: 'revolve', | ||||
|       ...node, | ||||
|     }) || | ||||
|     doesPipeHaveCallExp({ | ||||
|       calleeName: 'loft', | ||||
|       ...node, | ||||
|     }) | ||||
|   ) | ||||
| } | ||||
| @ -559,6 +569,22 @@ export function canSweepSelection(selection: Selections) { | ||||
|   ) | ||||
| } | ||||
|  | ||||
| export function canLoftSelection(selection: Selections) { | ||||
|   const commonNodes = selection.graphSelections.map((_, i) => | ||||
|     buildCommonNodeFromSelection(selection, i) | ||||
|   ) | ||||
|   return ( | ||||
|     !!isCursorInSketchCommandRange( | ||||
|       engineCommandManager.artifactGraph, | ||||
|       selection | ||||
|     ) && | ||||
|     commonNodes.length > 1 && | ||||
|     commonNodes.every((n) => !hasSketchPipeBeenExtruded(n.selection, n.ast)) && | ||||
|     commonNodes.every((n) => nodeHasClose(n) || nodeHasCircle(n)) && | ||||
|     commonNodes.every((n) => !nodeHasExtrude(n)) | ||||
|   ) | ||||
| } | ||||
|  | ||||
| // This accounts for non-geometry selections under "other" | ||||
| export type ResolvedSelectionType = Artifact['type'] | 'other' | ||||
| export type SelectionCountsByType = Map<ResolvedSelectionType, number> | ||||
| @ -905,7 +931,7 @@ export function updateSelections( | ||||
|       return { | ||||
|         artifact: artifact, | ||||
|         codeRef: { | ||||
|           range: [node.start, node.end], | ||||
|           range: [node.start, node.end, true], | ||||
|           pathToNode: pathToNode, | ||||
|         }, | ||||
|       } | ||||
| @ -919,7 +945,7 @@ export function updateSelections( | ||||
|     if (err(node)) return node | ||||
|     pathToNodeBasedSelections.push({ | ||||
|       codeRef: { | ||||
|         range: [node.node.start, node.node.end], | ||||
|         range: [node.node.start, node.node.end, true], | ||||
|         pathToNode: pathToNode, | ||||
|       }, | ||||
|     }) | ||||
|  | ||||
| @ -4,7 +4,6 @@ import { | ||||
|   _executor, | ||||
|   SourceRange, | ||||
|   ExecState, | ||||
|   defaultIdGenerator, | ||||
| } from '../lang/wasm' | ||||
| import { | ||||
|   EngineCommandManager, | ||||
| @ -16,7 +15,6 @@ import { v4 as uuidv4 } from 'uuid' | ||||
| import { DefaultPlanes } from 'wasm-lib/kcl/bindings/DefaultPlanes' | ||||
| import { err, reportRejection } from 'lib/trap' | ||||
| import { toSync } from './utils' | ||||
| import { IdGenerator } from 'wasm-lib/kcl/bindings/IdGenerator' | ||||
| import { Node } from 'wasm-lib/kcl/bindings/Node' | ||||
|  | ||||
| type WebSocketResponse = Models['WebSocketResponse_type'] | ||||
| @ -85,12 +83,10 @@ class MockEngineCommandManager { | ||||
| } | ||||
|  | ||||
| export async function enginelessExecutor( | ||||
|   ast: Node<Program> | Error, | ||||
|   pm: ProgramMemory | Error = ProgramMemory.empty(), | ||||
|   idGenerator: IdGenerator = defaultIdGenerator() | ||||
|   ast: Node<Program>, | ||||
|   pmo: ProgramMemory | Error = ProgramMemory.empty() | ||||
| ): Promise<ExecState> { | ||||
|   if (err(ast)) return Promise.reject(ast) | ||||
|   if (err(pm)) return Promise.reject(pm) | ||||
|   if (pmo !== null && err(pmo)) return Promise.reject(pmo) | ||||
|  | ||||
|   const mockEngineCommandManager = new MockEngineCommandManager({ | ||||
|     setIsStreamReady: () => {}, | ||||
| @ -98,21 +94,14 @@ export async function enginelessExecutor( | ||||
|   }) as any as EngineCommandManager | ||||
|   // eslint-disable-next-line @typescript-eslint/no-floating-promises | ||||
|   mockEngineCommandManager.startNewSession() | ||||
|   const execState = await _executor( | ||||
|     ast, | ||||
|     pm, | ||||
|     idGenerator, | ||||
|     mockEngineCommandManager, | ||||
|     true | ||||
|   ) | ||||
|   const execState = await _executor(ast, mockEngineCommandManager, pmo) | ||||
|   await mockEngineCommandManager.waitForAllCommands() | ||||
|   return execState | ||||
| } | ||||
|  | ||||
| export async function executor( | ||||
|   ast: Node<Program>, | ||||
|   pm: ProgramMemory = ProgramMemory.empty(), | ||||
|   idGenerator: IdGenerator = defaultIdGenerator() | ||||
|   pmo: ProgramMemory = ProgramMemory.empty() | ||||
| ): Promise<ExecState> { | ||||
|   const engineCommandManager = new EngineCommandManager() | ||||
|   engineCommandManager.start({ | ||||
| @ -134,13 +123,7 @@ export async function executor( | ||||
|       toSync(async () => { | ||||
|         // eslint-disable-next-line @typescript-eslint/no-floating-promises | ||||
|         engineCommandManager.startNewSession() | ||||
|         const execState = await _executor( | ||||
|           ast, | ||||
|           pm, | ||||
|           idGenerator, | ||||
|           engineCommandManager, | ||||
|           false | ||||
|         ) | ||||
|         const execState = await _executor(ast, engineCommandManager, pmo) | ||||
|         await engineCommandManager.waitForAllCommands() | ||||
|         resolve(execState) | ||||
|       }, reportRejection) | ||||
|  | ||||
| @ -139,9 +139,14 @@ export const toolbarConfig: Record<ToolbarModeName, ToolbarMode> = { | ||||
|       }, | ||||
|       { | ||||
|         id: 'loft', | ||||
|         onClick: () => console.error('Loft not yet implemented'), | ||||
|         onClick: ({ commandBarSend }) => | ||||
|           commandBarSend({ | ||||
|             type: 'Find and select command', | ||||
|             data: { name: 'Loft', groupId: 'modeling' }, | ||||
|           }), | ||||
|         disabled: (state) => !state.can({ type: 'Loft' }), | ||||
|         icon: 'loft', | ||||
|         status: 'kcl-only', | ||||
|         status: 'available', | ||||
|         title: 'Loft', | ||||
|         hotkey: 'L', | ||||
|         description: | ||||
|  | ||||
| @ -3,7 +3,7 @@ import { kclManager, engineCommandManager } from 'lib/singletons' | ||||
| import { useKclContext } from 'lang/KclProvider' | ||||
| import { findUniqueName } from 'lang/modifyAst' | ||||
| import { PrevVariable, findAllPreviousVariables } from 'lang/queryAst' | ||||
| import { ProgramMemory, Expr, parse } from 'lang/wasm' | ||||
| import { ProgramMemory, Expr, parse, resultIsOk } from 'lang/wasm' | ||||
| import { useEffect, useRef, useState } from 'react' | ||||
| import { executeAst } from 'lang/langHelpers' | ||||
| import { err, trap } from 'lib/trap' | ||||
| @ -87,9 +87,9 @@ export function useCalculateKclExpression({ | ||||
|   useEffect(() => { | ||||
|     const execAstAndSetResult = async () => { | ||||
|       const _code = `const __result__ = ${value}` | ||||
|       const ast = parse(_code) | ||||
|       if (err(ast)) return | ||||
|       if (trap(ast, { suppress: true })) return | ||||
|       const pResult = parse(_code) | ||||
|       if (err(pResult) || !resultIsOk(pResult)) return | ||||
|       const ast = pResult.program | ||||
|  | ||||
|       const _programMem: ProgramMemory = ProgramMemory.empty() | ||||
|       for (const { key, value } of availableVarInfo.variables) { | ||||
| @ -103,9 +103,8 @@ export function useCalculateKclExpression({ | ||||
|       const { execState } = await executeAst({ | ||||
|         ast, | ||||
|         engineCommandManager, | ||||
|         useFakeExecutor: true, | ||||
|         // We make sure to send an empty program memory to denote we mean mock mode. | ||||
|         programMemoryOverride: kclManager.programMemory.clone(), | ||||
|         idGenerator: kclManager.execState.idGenerator, | ||||
|       }) | ||||
|       const resultDeclaration = ast.body.find( | ||||
|         (a) => | ||||
|  | ||||
| @ -9,13 +9,13 @@ import { | ||||
| import { SourceRange } from '../lang/wasm' | ||||
|  | ||||
| describe('testing isOverlapping', () => { | ||||
|   testBothOrders([0, 3], [3, 10]) | ||||
|   testBothOrders([0, 5], [3, 4]) | ||||
|   testBothOrders([0, 5], [5, 10]) | ||||
|   testBothOrders([0, 5], [6, 10], false) | ||||
|   testBothOrders([0, 5], [-1, 1]) | ||||
|   testBothOrders([0, 5], [-1, 0]) | ||||
|   testBothOrders([0, 5], [-2, -1], false) | ||||
|   testBothOrders([0, 3, true], [3, 10, true]) | ||||
|   testBothOrders([0, 5, true], [3, 4, true]) | ||||
|   testBothOrders([0, 5, true], [5, 10, true]) | ||||
|   testBothOrders([0, 5, true], [6, 10, true], false) | ||||
|   testBothOrders([0, 5, true], [-1, 1, true]) | ||||
|   testBothOrders([0, 5, true], [-1, 0, true]) | ||||
|   testBothOrders([0, 5, true], [-2, -1, true], false) | ||||
| }) | ||||
|  | ||||
| function testBothOrders(a: SourceRange, b: SourceRange, result = true) { | ||||
|  | ||||
							
								
								
									
										22
									
								
								src/lib/xStateValueToString.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,22 @@ | ||||
| import { AnyStateMachine, StateFrom } from 'xstate' | ||||
|  | ||||
| /** | ||||
|  * Convert an XState state value to a pretty string, | ||||
|  * with nested states separated by slashes | ||||
|  */ | ||||
| export function xStateValueToString( | ||||
|   stateValue: StateFrom<AnyStateMachine>['value'] | ||||
| ) { | ||||
|   const sep = ' / ' | ||||
|   let output = '' | ||||
|   let remainingValues = stateValue | ||||
|   let isFirstStep = true | ||||
|   while (remainingValues instanceof Object) { | ||||
|     const key: keyof typeof remainingValues = Object.keys(remainingValues)[0] | ||||
|     output += (isFirstStep ? '' : sep) + key | ||||
|     remainingValues = remainingValues[key] | ||||
|     isFirstStep = false | ||||
|   } | ||||
|   if (typeof remainingValues === 'string' && remainingValues.trim().length) | ||||
|     return output + sep + remainingValues.trim() | ||||
| } | ||||
| @ -1,9 +1,11 @@ | ||||
| import { | ||||
|   PathToNode, | ||||
|   ProgramMemory, | ||||
|   VariableDeclaration, | ||||
|   VariableDeclarator, | ||||
|   parse, | ||||
|   recast, | ||||
|   resultIsOk, | ||||
| } from 'lang/wasm' | ||||
| import { | ||||
|   Axis, | ||||
| @ -44,6 +46,7 @@ import { | ||||
|   addOffsetPlane, | ||||
|   deleteFromSelection, | ||||
|   extrudeSketch, | ||||
|   loftSketches, | ||||
|   revolveSketch, | ||||
| } from 'lang/modifyAst' | ||||
| import { | ||||
| @ -256,6 +259,7 @@ export type ModelingMachineEvent = | ||||
|   | { type: 'Export'; data: ModelingCommandSchema['Export'] } | ||||
|   | { type: 'Make'; data: ModelingCommandSchema['Make'] } | ||||
|   | { type: 'Extrude'; data?: ModelingCommandSchema['Extrude'] } | ||||
|   | { type: 'Loft'; data?: ModelingCommandSchema['Loft'] } | ||||
|   | { type: 'Revolve'; data?: ModelingCommandSchema['Revolve'] } | ||||
|   | { type: 'Fillet'; data?: ModelingCommandSchema['Fillet'] } | ||||
|   | { type: 'Offset plane'; data: ModelingCommandSchema['Offset plane'] } | ||||
| @ -387,6 +391,7 @@ export const modelingMachine = setup({ | ||||
|   guards: { | ||||
|     'Selection is on face': () => false, | ||||
|     'has valid sweep selection': () => false, | ||||
|     'has valid loft selection': () => false, | ||||
|     'has valid edge treatment selection': () => false, | ||||
|     'Has exportable geometry': () => false, | ||||
|     'has valid selection for deletion': () => false, | ||||
| @ -539,8 +544,11 @@ export const modelingMachine = setup({ | ||||
|       if (event.type !== 'Convert to variable') return false | ||||
|       if (!event.data) return false | ||||
|       const ast = parse(recast(kclManager.ast)) | ||||
|       if (err(ast)) return false | ||||
|       const isSafeRetVal = isNodeSafeToReplacePath(ast, event.data.pathToNode) | ||||
|       if (err(ast) || !ast.program || ast.errors.length > 0) return false | ||||
|       const isSafeRetVal = isNodeSafeToReplacePath( | ||||
|         ast.program, | ||||
|         event.data.pathToNode | ||||
|       ) | ||||
|       if (err(isSafeRetVal)) return false | ||||
|       return isSafeRetVal.isSafe | ||||
|     }, | ||||
| @ -723,9 +731,9 @@ export const modelingMachine = setup({ | ||||
|  | ||||
|         const testExecute = await executeAst({ | ||||
|           ast: modifiedAst, | ||||
|           idGenerator: kclManager.execState.idGenerator, | ||||
|           useFakeExecutor: true, | ||||
|           engineCommandManager, | ||||
|           // We make sure to send an empty program memory to denote we mean mock mode. | ||||
|           programMemoryOverride: ProgramMemory.empty(), | ||||
|         }) | ||||
|         if (testExecute.errors.length) { | ||||
|           toast.error('Unable to delete part') | ||||
| @ -1329,9 +1337,12 @@ export const modelingMachine = setup({ | ||||
|           return | ||||
|         } | ||||
|  | ||||
|         const recastAst = parse(recast(modifiedAst)) | ||||
|         if (err(recastAst) || !resultIsOk(recastAst)) return | ||||
|  | ||||
|         const updatedAst = await sceneEntitiesManager.updateAstAndRejigSketch( | ||||
|           sketchDetails?.sketchPathToNode || [], | ||||
|           parse(recast(modifiedAst)), | ||||
|           recastAst.program, | ||||
|           sketchDetails.zAxis, | ||||
|           sketchDetails.yAxis, | ||||
|           sketchDetails.origin | ||||
| @ -1529,6 +1540,50 @@ export const modelingMachine = setup({ | ||||
|           updateAstResult.newAst | ||||
|         ) | ||||
|  | ||||
|         if (updateAstResult?.selections) { | ||||
|           editorManager.selectRange(updateAstResult?.selections) | ||||
|         } | ||||
|       } | ||||
|     ), | ||||
|     loftAstMod: fromPromise( | ||||
|       async ({ | ||||
|         input, | ||||
|       }: { | ||||
|         input: ModelingCommandSchema['Loft'] | undefined | ||||
|       }) => { | ||||
|         if (!input) return new Error('No input provided') | ||||
|         // Extract inputs | ||||
|         const ast = kclManager.ast | ||||
|         const { selection } = input | ||||
|         const declarators = selection.graphSelections.flatMap((s) => { | ||||
|           const path = getNodePathFromSourceRange(ast, s?.codeRef.range) | ||||
|           const nodeFromPath = getNodeFromPath<VariableDeclarator>( | ||||
|             ast, | ||||
|             path, | ||||
|             'VariableDeclarator' | ||||
|           ) | ||||
|           return err(nodeFromPath) ? [] : nodeFromPath.node | ||||
|         }) | ||||
|  | ||||
|         // TODO: add better validation on selection | ||||
|         if (!(declarators && declarators.length > 1)) { | ||||
|           trap('Not enough sketches selected') | ||||
|         } | ||||
|  | ||||
|         // Perform the loft | ||||
|         const loftSketchesRes = loftSketches(ast, declarators) | ||||
|         const updateAstResult = await kclManager.updateAst( | ||||
|           loftSketchesRes.modifiedAst, | ||||
|           true, | ||||
|           { | ||||
|             focusPath: [loftSketchesRes.pathToNode], | ||||
|           } | ||||
|         ) | ||||
|  | ||||
|         await codeManager.updateEditorWithAstAndWriteToFile( | ||||
|           updateAstResult.newAst | ||||
|         ) | ||||
|  | ||||
|         if (updateAstResult?.selections) { | ||||
|           editorManager.selectRange(updateAstResult?.selections) | ||||
|         } | ||||
| @ -1570,6 +1625,11 @@ export const modelingMachine = setup({ | ||||
|           reenter: false, | ||||
|         }, | ||||
|  | ||||
|         Loft: { | ||||
|           target: 'Applying loft', | ||||
|           reenter: true, | ||||
|         }, | ||||
|  | ||||
|         Fillet: { | ||||
|           target: 'idle', | ||||
|           guard: 'has valid edge treatment selection', | ||||
| @ -2318,6 +2378,19 @@ export const modelingMachine = setup({ | ||||
|         onError: ['idle'], | ||||
|       }, | ||||
|     }, | ||||
|  | ||||
|     'Applying loft': { | ||||
|       invoke: { | ||||
|         src: 'loftAstMod', | ||||
|         id: 'loftAstMod', | ||||
|         input: ({ event }) => { | ||||
|           if (event.type !== 'Loft') return undefined | ||||
|           return event.data | ||||
|         }, | ||||
|         onDone: ['idle'], | ||||
|         onError: ['idle'], | ||||
|       }, | ||||
|     }, | ||||
|   }, | ||||
|  | ||||
|   initial: 'idle', | ||||
|  | ||||
| @ -13,18 +13,6 @@ import { AllSettingsFields } from 'components/Settings/AllSettingsFields' | ||||
| import { AllKeybindingsFields } from 'components/Settings/AllKeybindingsFields' | ||||
| import { KeybindingsSectionsList } from 'components/Settings/KeybindingsSectionsList' | ||||
| import { isDesktop } from 'lib/isDesktop' | ||||
| import { IS_PLAYWRIGHT_KEY } from '../../e2e/playwright/storageStates' | ||||
| import { NODE_ENV } from 'env' | ||||
|  | ||||
| const isTestEnv = window?.localStorage.getItem(IS_PLAYWRIGHT_KEY) === 'true' | ||||
|  | ||||
| export const APP_VERSION = | ||||
|   isTestEnv && NODE_ENV === 'development' | ||||
|     ? '11.22.33' | ||||
|     : isDesktop() | ||||
|     ? // @ts-ignore | ||||
|       window.electron.packageJson.version | ||||
|     : 'main' | ||||
|  | ||||
| export const PACKAGE_NAME = isDesktop() | ||||
|   ? window.electron.packageJson.name | ||||
|  | ||||
| @ -5,11 +5,11 @@ import { Themes, getSystemTheme } from '../lib/theme' | ||||
| import { PATHS } from 'lib/paths' | ||||
| import { useSettingsAuthContext } from 'hooks/useSettingsAuthContext' | ||||
| import { APP_NAME } from 'lib/constants' | ||||
| import { APP_VERSION } from 'lib/appVersion' | ||||
| import { CSSProperties, useCallback, useState } from 'react' | ||||
| import { Logo } from 'components/Logo' | ||||
| import { CustomIcon } from 'components/CustomIcon' | ||||
| import { Link } from 'react-router-dom' | ||||
| import { APP_VERSION } from './Settings' | ||||
| import { openExternalBrowserIfDesktop } from 'lib/openWindow' | ||||
| import { toSync } from 'lib/utils' | ||||
| import { reportRejection } from 'lib/trap' | ||||
|  | ||||
							
								
								
									
										82
									
								
								src/wasm-lib/Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						| @ -316,7 +316,7 @@ dependencies = [ | ||||
|  "bitvec", | ||||
|  "chrono", | ||||
|  "hex", | ||||
|  "indexmap 2.6.0", | ||||
|  "indexmap 2.7.0", | ||||
|  "js-sys", | ||||
|  "once_cell", | ||||
|  "rand 0.8.5", | ||||
| @ -712,29 +712,6 @@ version = "2.6.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "e8566979429cf69b49a5c740c60791108e86440e8be149bbea4fe54d2c32d6e2" | ||||
|  | ||||
| [[package]] | ||||
| name = "databake" | ||||
| version = "0.1.8" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "6a04fbfbecca8f0679c8c06fef907594adcc3e2052e11163a6d30535a1a5604d" | ||||
| dependencies = [ | ||||
|  "databake-derive", | ||||
|  "proc-macro2", | ||||
|  "quote", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "databake-derive" | ||||
| version = "0.1.8" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "4078275de501a61ceb9e759d37bdd3d7210e654dbc167ac1a3678ef4435ed57b" | ||||
| dependencies = [ | ||||
|  "proc-macro2", | ||||
|  "quote", | ||||
|  "syn 2.0.87", | ||||
|  "synstructure", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "deranged" | ||||
| version = "0.3.11" | ||||
| @ -746,7 +723,7 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "derive-docs" | ||||
| version = "0.1.31" | ||||
| version = "0.1.32" | ||||
| dependencies = [ | ||||
|  "Inflector", | ||||
|  "anyhow", | ||||
| @ -1136,7 +1113,7 @@ dependencies = [ | ||||
|  "futures-core", | ||||
|  "futures-sink", | ||||
|  "http 1.1.0", | ||||
|  "indexmap 2.6.0", | ||||
|  "indexmap 2.7.0", | ||||
|  "slab", | ||||
|  "tokio", | ||||
|  "tokio-util", | ||||
| @ -1182,9 +1159,9 @@ checksum = "e5274423e17b7c9fc20b6e7e208532f9b19825d82dfd615708b70edd83df41f1" | ||||
|  | ||||
| [[package]] | ||||
| name = "hashbrown" | ||||
| version = "0.15.0" | ||||
| version = "0.15.2" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "1e087f84d4f86bf4b218b927129862374b72199ae7d8657835f1e89000eea4fb" | ||||
| checksum = "bf151400ff0baff5465007dd2f3e717f3fe502074ca563069ce3a6629d07b289" | ||||
|  | ||||
| [[package]] | ||||
| name = "heck" | ||||
| @ -1584,12 +1561,12 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "indexmap" | ||||
| version = "2.6.0" | ||||
| version = "2.7.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "707907fe3c25f5424cce2cb7e1cbcafee6bdbe735ca90ef77c29e84591e5b9da" | ||||
| checksum = "62f822373a4fe84d4bb149bf54e584a7f4abec90e072ed49cda0edea5b95471f" | ||||
| dependencies = [ | ||||
|  "equivalent", | ||||
|  "hashbrown 0.15.0", | ||||
|  "hashbrown 0.15.2", | ||||
|  "serde", | ||||
| ] | ||||
|  | ||||
| @ -1706,7 +1683,7 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "kcl-lib" | ||||
| version = "0.2.27" | ||||
| version = "0.2.28" | ||||
| dependencies = [ | ||||
|  "anyhow", | ||||
|  "approx 0.5.1", | ||||
| @ -1719,7 +1696,6 @@ dependencies = [ | ||||
|  "convert_case", | ||||
|  "criterion", | ||||
|  "dashmap 6.1.0", | ||||
|  "databake", | ||||
|  "derive-docs", | ||||
|  "dhat", | ||||
|  "expectorate", | ||||
| @ -1732,7 +1708,7 @@ dependencies = [ | ||||
|  "http 1.1.0", | ||||
|  "iai", | ||||
|  "image", | ||||
|  "indexmap 2.6.0", | ||||
|  "indexmap 2.7.0", | ||||
|  "insta", | ||||
|  "itertools 0.13.0", | ||||
|  "js-sys", | ||||
| @ -1773,7 +1749,7 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "kcl-test-server" | ||||
| version = "0.1.17" | ||||
| version = "0.1.18" | ||||
| dependencies = [ | ||||
|  "anyhow", | ||||
|  "hyper 0.14.30", | ||||
| @ -1790,7 +1766,7 @@ version = "0.1.0" | ||||
| dependencies = [ | ||||
|  "anyhow", | ||||
|  "async-trait", | ||||
|  "indexmap 2.6.0", | ||||
|  "indexmap 2.7.0", | ||||
|  "kcl-lib", | ||||
|  "kittycad", | ||||
|  "kittycad-modeling-cmds", | ||||
| @ -1800,9 +1776,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "kittycad" | ||||
| version = "0.3.25" | ||||
| version = "0.3.28" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "f6359cc0a1bbccbcf78775eea17a033cf2aa89d3fe6a9784f8ce94e5f882c185" | ||||
| checksum = "933cb5f77624386c87d296e3fd493daf50156d1cbfa03b9f333a6d4da2896369" | ||||
| dependencies = [ | ||||
|  "anyhow", | ||||
|  "async-trait", | ||||
| @ -1831,7 +1807,7 @@ dependencies = [ | ||||
|  "serde_bytes", | ||||
|  "serde_json", | ||||
|  "serde_urlencoded", | ||||
|  "thiserror 1.0.68", | ||||
|  "thiserror 2.0.0", | ||||
|  "tokio", | ||||
|  "tracing", | ||||
|  "url", | ||||
| @ -2921,9 +2897,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "reqwest-conditional-middleware" | ||||
| version = "0.3.0" | ||||
| version = "0.4.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "b1663d9d4fbb6e3900f91455d6d7833301c91ae3c7fc6e116fd7acd40e478a93" | ||||
| checksum = "f67ad7fdf5c0a015763fcd164bee294b13fb7b6f89f1b55961d40f00c3e32d6b" | ||||
| dependencies = [ | ||||
|  "async-trait", | ||||
|  "http 1.1.0", | ||||
| @ -2933,9 +2909,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "reqwest-middleware" | ||||
| version = "0.3.3" | ||||
| version = "0.4.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "562ceb5a604d3f7c885a792d42c199fd8af239d0a51b2fa6a78aafa092452b04" | ||||
| checksum = "d1ccd3b55e711f91a9885a2fa6fbbb2e39db1776420b062efc058c6410f7e5e3" | ||||
| dependencies = [ | ||||
|  "anyhow", | ||||
|  "async-trait", | ||||
| @ -2948,9 +2924,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "reqwest-retry" | ||||
| version = "0.6.1" | ||||
| version = "0.7.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "a83df1aaec00176d0fabb65dea13f832d2a446ca99107afc17c5d2d4981221d0" | ||||
| checksum = "29c73e4195a6bfbcb174b790d9b3407ab90646976c55de58a6515da25d851178" | ||||
| dependencies = [ | ||||
|  "anyhow", | ||||
|  "async-trait", | ||||
| @ -2962,6 +2938,7 @@ dependencies = [ | ||||
|  "reqwest", | ||||
|  "reqwest-middleware", | ||||
|  "retry-policies", | ||||
|  "thiserror 1.0.68", | ||||
|  "tokio", | ||||
|  "tracing", | ||||
|  "wasm-timer", | ||||
| @ -2969,9 +2946,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "reqwest-tracing" | ||||
| version = "0.5.3" | ||||
| version = "0.5.4" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "bfdd9bfa64c72233d8dd99ab7883efcdefe9e16d46488ecb9228b71a2e2ceb45" | ||||
| checksum = "ff82cf5730a1311fb9413b0bc2b8e743e0157cd73f010ab4ec374a923873b6a2" | ||||
| dependencies = [ | ||||
|  "anyhow", | ||||
|  "async-trait", | ||||
| @ -3157,7 +3134,7 @@ dependencies = [ | ||||
|  "chrono", | ||||
|  "dyn-clone", | ||||
|  "indexmap 1.9.3", | ||||
|  "indexmap 2.6.0", | ||||
|  "indexmap 2.7.0", | ||||
|  "schemars_derive", | ||||
|  "serde", | ||||
|  "serde_json", | ||||
| @ -3258,7 +3235,7 @@ version = "1.0.133" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "c7fceb2473b9166b2294ef05efcb65a3db80803f0b03ef86a5fc88a2b85ee377" | ||||
| dependencies = [ | ||||
|  "indexmap 2.6.0", | ||||
|  "indexmap 2.7.0", | ||||
|  "itoa", | ||||
|  "memchr", | ||||
|  "ryu", | ||||
| @ -3846,7 +3823,7 @@ version = "0.22.22" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "4ae48d6208a266e853d946088ed816055e556cc6028c5e8e2b84d9fa5dd7c7f5" | ||||
| dependencies = [ | ||||
|  "indexmap 2.6.0", | ||||
|  "indexmap 2.7.0", | ||||
|  "serde", | ||||
|  "serde_spanned", | ||||
|  "toml_datetime", | ||||
| @ -4012,7 +3989,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "3a2f31991cee3dce1ca4f929a8a04fdd11fd8801aac0f2030b0fa8a0a3fef6b9" | ||||
| dependencies = [ | ||||
|  "chrono", | ||||
|  "indexmap 2.6.0", | ||||
|  "indexmap 2.7.0", | ||||
|  "lazy_static", | ||||
|  "serde_json", | ||||
|  "thiserror 1.0.68", | ||||
| @ -4331,6 +4308,7 @@ dependencies = [ | ||||
|  "kcl-lib", | ||||
|  "kittycad", | ||||
|  "kittycad-modeling-cmds", | ||||
|  "lazy_static", | ||||
|  "pretty_assertions", | ||||
|  "reqwest", | ||||
|  "serde_json", | ||||
| @ -4769,7 +4747,7 @@ dependencies = [ | ||||
|  "crc32fast", | ||||
|  "crossbeam-utils", | ||||
|  "displaydoc", | ||||
|  "indexmap 2.6.0", | ||||
|  "indexmap 2.7.0", | ||||
|  "memchr", | ||||
|  "thiserror 1.0.68", | ||||
| ] | ||||
|  | ||||
| @ -15,6 +15,7 @@ data-encoding = "2.6.0" | ||||
| gloo-utils = "0.2.0" | ||||
| kcl-lib = { path = "kcl" } | ||||
| kittycad.workspace = true | ||||
| lazy_static = "1.5.0" | ||||
| serde_json = "1.0.128" | ||||
| tokio = { version = "1.41.1", features = ["sync"] } | ||||
| toml = "0.8.19" | ||||
| @ -74,8 +75,11 @@ members = [ | ||||
|  | ||||
| [workspace.dependencies] | ||||
| http = "1" | ||||
| kittycad = { version = "0.3.25", default-features = false, features = ["js", "requests"] } | ||||
| kittycad-modeling-cmds = { version = "0.2.76", features = ["websocket"] } | ||||
| kittycad = { version = "0.3.28", default-features = false, features = ["js", "requests"] } | ||||
| kittycad-modeling-cmds = { version = "0.2.77", features = ["websocket"] } | ||||
|  | ||||
| [workspace.lints.clippy] | ||||
| iter_over_hash_type = "warn" | ||||
|  | ||||
| [[test]] | ||||
| name = "executor" | ||||
|  | ||||
| @ -1,7 +1,7 @@ | ||||
| [package] | ||||
| name = "derive-docs" | ||||
| description = "A tool for generating documentation from Rust derive macros" | ||||
| version = "0.1.31" | ||||
| version = "0.1.32" | ||||
| edition = "2021" | ||||
| license = "MIT" | ||||
| repository = "https://github.com/KittyCAD/modeling-app" | ||||
| @ -27,3 +27,6 @@ anyhow = "1.0.93" | ||||
| expectorate = "1.1.0" | ||||
| pretty_assertions = "1.4.1" | ||||
| rustfmt-wrapper = "0.2.1" | ||||
|  | ||||
| [lints] | ||||
| workspace = true | ||||
|  | ||||
| @ -23,17 +23,30 @@ use unbox::unbox; | ||||
| struct StdlibMetadata { | ||||
|     /// The name of the function in the API. | ||||
|     name: String, | ||||
|  | ||||
|     /// Tags for the function. | ||||
|     #[serde(default)] | ||||
|     tags: Vec<String>, | ||||
|  | ||||
|     /// Whether the function is unpublished. | ||||
|     /// Then docs will not be generated. | ||||
|     #[serde(default)] | ||||
|     unpublished: bool, | ||||
|  | ||||
|     /// Whether the function is deprecated. | ||||
|     /// Then specific docs detailing that this is deprecated will be generated. | ||||
|     #[serde(default)] | ||||
|     deprecated: bool, | ||||
|  | ||||
|     /// If true, expects keyword arguments. | ||||
|     /// If false, expects positional arguments. | ||||
|     #[serde(default)] | ||||
|     keywords: bool, | ||||
|  | ||||
|     /// If true, the first argument is unlabeled. | ||||
|     /// If false, all arguments require labels. | ||||
|     #[serde(default)] | ||||
|     unlabeled_first: bool, | ||||
| } | ||||
|  | ||||
| #[proc_macro_attribute] | ||||
| @ -169,9 +182,9 @@ fn do_stdlib_inner( | ||||
|         quote! { | ||||
|             let code_blocks = vec![#(#cb),*]; | ||||
|             code_blocks.iter().map(|cb| { | ||||
|                 let program = crate::Program::parse(cb).unwrap(); | ||||
|                 let program = crate::Program::parse_no_errs(cb).unwrap(); | ||||
|  | ||||
|                 let mut options: crate::ast::types::FormatOptions = Default::default(); | ||||
|                 let mut options: crate::parsing::ast::types::FormatOptions = Default::default(); | ||||
|                 options.insert_final_newline = false; | ||||
|                 program.ast.recast(&options, 0) | ||||
|             }).collect::<Vec<String>>() | ||||
| @ -225,6 +238,12 @@ fn do_stdlib_inner( | ||||
|         quote! { false } | ||||
|     }; | ||||
|  | ||||
|     let uses_keyword_arguments = if metadata.keywords { | ||||
|         quote! { true } | ||||
|     } else { | ||||
|         quote! { false } | ||||
|     }; | ||||
|  | ||||
|     let docs_crate = get_crate(None); | ||||
|  | ||||
|     // When the user attaches this proc macro to a function with the wrong type | ||||
| @ -233,7 +252,7 @@ fn do_stdlib_inner( | ||||
|     // of the various parameters. We do this by calling dummy functions that | ||||
|     // require a type that satisfies SharedExtractor or ExclusiveExtractor. | ||||
|     let mut arg_types = Vec::new(); | ||||
|     for arg in ast.sig.inputs.iter() { | ||||
|     for (i, arg) in ast.sig.inputs.iter().enumerate() { | ||||
|         // Get the name of the argument. | ||||
|         let arg_name = match arg { | ||||
|             syn::FnArg::Receiver(pat) => { | ||||
| @ -263,7 +282,7 @@ fn do_stdlib_inner( | ||||
|  | ||||
|         let ty_string = rust_type_to_openapi_type(&ty_string); | ||||
|         let required = !ty_ident.to_string().starts_with("Option <"); | ||||
|  | ||||
|         let label_required = !(i == 0 && metadata.unlabeled_first); | ||||
|         if ty_string != "ExecState" && ty_string != "Args" { | ||||
|             let schema = quote! { | ||||
|                generator.root_schema_for::<#ty_ident>() | ||||
| @ -274,6 +293,7 @@ fn do_stdlib_inner( | ||||
|                     type_: #ty_string.to_string(), | ||||
|                     schema: #schema, | ||||
|                     required: #required, | ||||
|                     label_required: #label_required, | ||||
|                 } | ||||
|             }); | ||||
|         } | ||||
| @ -334,6 +354,7 @@ fn do_stdlib_inner( | ||||
|                 type_: #ret_ty_string.to_string(), | ||||
|                 schema, | ||||
|                 required: true, | ||||
|                 label_required: true, | ||||
|             }) | ||||
|         } | ||||
|     } else { | ||||
| @ -400,6 +421,10 @@ fn do_stdlib_inner( | ||||
|                 vec![#(#tags),*] | ||||
|             } | ||||
|  | ||||
|             fn keyword_arguments(&self) -> bool { | ||||
|                 #uses_keyword_arguments | ||||
|             } | ||||
|  | ||||
|             fn args(&self, inline_subschemas: bool) -> Vec<#docs_crate::StdLibFnArg> { | ||||
|                 let mut settings = schemars::gen::SchemaSettings::openapi3(); | ||||
|                 // We set this to false so we can recurse them later. | ||||
| @ -744,7 +769,7 @@ fn generate_code_block_test(fn_name: &str, code_block: &str, index: usize) -> pr | ||||
|     quote! { | ||||
|         #[tokio::test(flavor = "multi_thread")] | ||||
|         async fn #test_name_mock() { | ||||
|             let program = crate::Program::parse(#code_block).unwrap(); | ||||
|             let program = crate::Program::parse_no_errs(#code_block).unwrap(); | ||||
|             let ctx = crate::executor::ExecutorContext { | ||||
|                 engine: std::sync::Arc::new(Box::new(crate::engine::conn_mock::EngineConnection::new().await.unwrap())), | ||||
|                 fs: std::sync::Arc::new(crate::fs::FileManager::new()), | ||||
| @ -753,7 +778,7 @@ fn generate_code_block_test(fn_name: &str, code_block: &str, index: usize) -> pr | ||||
|                 context_type: crate::executor::ContextType::Mock, | ||||
|             }; | ||||
|  | ||||
|             ctx.run(&program, &mut crate::ExecState::default()).await.unwrap(); | ||||
|             ctx.run(program.into(), &mut crate::ExecState::default()).await.unwrap(); | ||||
|         } | ||||
|  | ||||
|         #[tokio::test(flavor = "multi_thread", worker_threads = 5)] | ||||
|  | ||||
| @ -2,7 +2,7 @@ | ||||
| mod test_examples_someFn { | ||||
|     #[tokio::test(flavor = "multi_thread")] | ||||
|     async fn test_mock_example_someFn0() { | ||||
|         let program = crate::Program::parse("someFn()").unwrap(); | ||||
|         let program = crate::Program::parse_no_errs("someFn()").unwrap(); | ||||
|         let ctx = crate::executor::ExecutorContext { | ||||
|             engine: std::sync::Arc::new(Box::new( | ||||
|                 crate::engine::conn_mock::EngineConnection::new() | ||||
| @ -14,7 +14,7 @@ mod test_examples_someFn { | ||||
|             settings: Default::default(), | ||||
|             context_type: crate::executor::ContextType::Mock, | ||||
|         }; | ||||
|         ctx.run(&program, &mut crate::ExecState::default()) | ||||
|         ctx.run(program.into(), &mut crate::ExecState::default()) | ||||
|             .await | ||||
|             .unwrap(); | ||||
|     } | ||||
| @ -74,6 +74,10 @@ impl crate::docs::StdLibFn for SomeFn { | ||||
|         vec![] | ||||
|     } | ||||
|  | ||||
|     fn keyword_arguments(&self) -> bool { | ||||
|         false | ||||
|     } | ||||
|  | ||||
|     fn args(&self, inline_subschemas: bool) -> Vec<crate::docs::StdLibFnArg> { | ||||
|         let mut settings = schemars::gen::SchemaSettings::openapi3(); | ||||
|         settings.inline_subschemas = inline_subschemas; | ||||
| @ -83,6 +87,7 @@ impl crate::docs::StdLibFn for SomeFn { | ||||
|             type_: "Foo".to_string(), | ||||
|             schema: generator.root_schema_for::<Foo>(), | ||||
|             required: true, | ||||
|             label_required: true, | ||||
|         }] | ||||
|     } | ||||
|  | ||||
| @ -96,6 +101,7 @@ impl crate::docs::StdLibFn for SomeFn { | ||||
|             type_: "i32".to_string(), | ||||
|             schema, | ||||
|             required: true, | ||||
|             label_required: true, | ||||
|         }) | ||||
|     } | ||||
|  | ||||
| @ -112,8 +118,8 @@ impl crate::docs::StdLibFn for SomeFn { | ||||
|         code_blocks | ||||
|             .iter() | ||||
|             .map(|cb| { | ||||
|                 let program = crate::Program::parse(cb).unwrap(); | ||||
|                 let mut options: crate::ast::types::FormatOptions = Default::default(); | ||||
|                 let program = crate::Program::parse_no_errs(cb).unwrap(); | ||||
|                 let mut options: crate::parsing::ast::types::FormatOptions = Default::default(); | ||||
|                 options.insert_final_newline = false; | ||||
|                 program.ast.recast(&options, 0) | ||||
|             }) | ||||
|  | ||||
| @ -2,7 +2,7 @@ | ||||
| mod test_examples_someFn { | ||||
|     #[tokio::test(flavor = "multi_thread")] | ||||
|     async fn test_mock_example_someFn0() { | ||||
|         let program = crate::Program::parse("someFn()").unwrap(); | ||||
|         let program = crate::Program::parse_no_errs("someFn()").unwrap(); | ||||
|         let ctx = crate::executor::ExecutorContext { | ||||
|             engine: std::sync::Arc::new(Box::new( | ||||
|                 crate::engine::conn_mock::EngineConnection::new() | ||||
| @ -14,7 +14,7 @@ mod test_examples_someFn { | ||||
|             settings: Default::default(), | ||||
|             context_type: crate::executor::ContextType::Mock, | ||||
|         }; | ||||
|         ctx.run(&program, &mut crate::ExecState::default()) | ||||
|         ctx.run(program.into(), &mut crate::ExecState::default()) | ||||
|             .await | ||||
|             .unwrap(); | ||||
|     } | ||||
| @ -74,6 +74,10 @@ impl crate::docs::StdLibFn for SomeFn { | ||||
|         vec![] | ||||
|     } | ||||
|  | ||||
|     fn keyword_arguments(&self) -> bool { | ||||
|         false | ||||
|     } | ||||
|  | ||||
|     fn args(&self, inline_subschemas: bool) -> Vec<crate::docs::StdLibFnArg> { | ||||
|         let mut settings = schemars::gen::SchemaSettings::openapi3(); | ||||
|         settings.inline_subschemas = inline_subschemas; | ||||
| @ -83,6 +87,7 @@ impl crate::docs::StdLibFn for SomeFn { | ||||
|             type_: "string".to_string(), | ||||
|             schema: generator.root_schema_for::<str>(), | ||||
|             required: true, | ||||
|             label_required: true, | ||||
|         }] | ||||
|     } | ||||
|  | ||||
| @ -96,6 +101,7 @@ impl crate::docs::StdLibFn for SomeFn { | ||||
|             type_: "i32".to_string(), | ||||
|             schema, | ||||
|             required: true, | ||||
|             label_required: true, | ||||
|         }) | ||||
|     } | ||||
|  | ||||
| @ -112,8 +118,8 @@ impl crate::docs::StdLibFn for SomeFn { | ||||
|         code_blocks | ||||
|             .iter() | ||||
|             .map(|cb| { | ||||
|                 let program = crate::Program::parse(cb).unwrap(); | ||||
|                 let mut options: crate::ast::types::FormatOptions = Default::default(); | ||||
|                 let program = crate::Program::parse_no_errs(cb).unwrap(); | ||||
|                 let mut options: crate::parsing::ast::types::FormatOptions = Default::default(); | ||||
|                 options.insert_final_newline = false; | ||||
|                 program.ast.recast(&options, 0) | ||||
|             }) | ||||
|  | ||||
| @ -3,7 +3,7 @@ mod test_examples_show { | ||||
|     #[tokio::test(flavor = "multi_thread")] | ||||
|     async fn test_mock_example_show0() { | ||||
|         let program = | ||||
|             crate::Program::parse("This is another code block.\nyes sirrr.\nshow").unwrap(); | ||||
|             crate::Program::parse_no_errs("This is another code block.\nyes sirrr.\nshow").unwrap(); | ||||
|         let ctx = crate::executor::ExecutorContext { | ||||
|             engine: std::sync::Arc::new(Box::new( | ||||
|                 crate::engine::conn_mock::EngineConnection::new() | ||||
| @ -15,7 +15,7 @@ mod test_examples_show { | ||||
|             settings: Default::default(), | ||||
|             context_type: crate::executor::ContextType::Mock, | ||||
|         }; | ||||
|         ctx.run(&program, &mut crate::ExecState::default()) | ||||
|         ctx.run(program.into(), &mut crate::ExecState::default()) | ||||
|             .await | ||||
|             .unwrap(); | ||||
|     } | ||||
| @ -36,7 +36,8 @@ mod test_examples_show { | ||||
|  | ||||
|     #[tokio::test(flavor = "multi_thread")] | ||||
|     async fn test_mock_example_show1() { | ||||
|         let program = crate::Program::parse("This is code.\nIt does other shit.\nshow").unwrap(); | ||||
|         let program = | ||||
|             crate::Program::parse_no_errs("This is code.\nIt does other shit.\nshow").unwrap(); | ||||
|         let ctx = crate::executor::ExecutorContext { | ||||
|             engine: std::sync::Arc::new(Box::new( | ||||
|                 crate::engine::conn_mock::EngineConnection::new() | ||||
| @ -48,7 +49,7 @@ mod test_examples_show { | ||||
|             settings: Default::default(), | ||||
|             context_type: crate::executor::ContextType::Mock, | ||||
|         }; | ||||
|         ctx.run(&program, &mut crate::ExecState::default()) | ||||
|         ctx.run(program.into(), &mut crate::ExecState::default()) | ||||
|             .await | ||||
|             .unwrap(); | ||||
|     } | ||||
| @ -108,6 +109,10 @@ impl crate::docs::StdLibFn for Show { | ||||
|         vec![] | ||||
|     } | ||||
|  | ||||
|     fn keyword_arguments(&self) -> bool { | ||||
|         false | ||||
|     } | ||||
|  | ||||
|     fn args(&self, inline_subschemas: bool) -> Vec<crate::docs::StdLibFnArg> { | ||||
|         let mut settings = schemars::gen::SchemaSettings::openapi3(); | ||||
|         settings.inline_subschemas = inline_subschemas; | ||||
| @ -117,6 +122,7 @@ impl crate::docs::StdLibFn for Show { | ||||
|             type_: "[number]".to_string(), | ||||
|             schema: generator.root_schema_for::<[f64; 2usize]>(), | ||||
|             required: true, | ||||
|             label_required: true, | ||||
|         }] | ||||
|     } | ||||
|  | ||||
| @ -130,6 +136,7 @@ impl crate::docs::StdLibFn for Show { | ||||
|             type_: "number".to_string(), | ||||
|             schema, | ||||
|             required: true, | ||||
|             label_required: true, | ||||
|         }) | ||||
|     } | ||||
|  | ||||
| @ -149,8 +156,8 @@ impl crate::docs::StdLibFn for Show { | ||||
|         code_blocks | ||||
|             .iter() | ||||
|             .map(|cb| { | ||||
|                 let program = crate::Program::parse(cb).unwrap(); | ||||
|                 let mut options: crate::ast::types::FormatOptions = Default::default(); | ||||
|                 let program = crate::Program::parse_no_errs(cb).unwrap(); | ||||
|                 let mut options: crate::parsing::ast::types::FormatOptions = Default::default(); | ||||
|                 options.insert_final_newline = false; | ||||
|                 program.ast.recast(&options, 0) | ||||
|             }) | ||||
|  | ||||
| @ -2,7 +2,8 @@ | ||||
| mod test_examples_show { | ||||
|     #[tokio::test(flavor = "multi_thread")] | ||||
|     async fn test_mock_example_show0() { | ||||
|         let program = crate::Program::parse("This is code.\nIt does other shit.\nshow").unwrap(); | ||||
|         let program = | ||||
|             crate::Program::parse_no_errs("This is code.\nIt does other shit.\nshow").unwrap(); | ||||
|         let ctx = crate::executor::ExecutorContext { | ||||
|             engine: std::sync::Arc::new(Box::new( | ||||
|                 crate::engine::conn_mock::EngineConnection::new() | ||||
| @ -14,7 +15,7 @@ mod test_examples_show { | ||||
|             settings: Default::default(), | ||||
|             context_type: crate::executor::ContextType::Mock, | ||||
|         }; | ||||
|         ctx.run(&program, &mut crate::ExecState::default()) | ||||
|         ctx.run(program.into(), &mut crate::ExecState::default()) | ||||
|             .await | ||||
|             .unwrap(); | ||||
|     } | ||||
| @ -74,6 +75,10 @@ impl crate::docs::StdLibFn for Show { | ||||
|         vec![] | ||||
|     } | ||||
|  | ||||
|     fn keyword_arguments(&self) -> bool { | ||||
|         false | ||||
|     } | ||||
|  | ||||
|     fn args(&self, inline_subschemas: bool) -> Vec<crate::docs::StdLibFnArg> { | ||||
|         let mut settings = schemars::gen::SchemaSettings::openapi3(); | ||||
|         settings.inline_subschemas = inline_subschemas; | ||||
| @ -83,6 +88,7 @@ impl crate::docs::StdLibFn for Show { | ||||
|             type_: "number".to_string(), | ||||
|             schema: generator.root_schema_for::<f64>(), | ||||
|             required: true, | ||||
|             label_required: true, | ||||
|         }] | ||||
|     } | ||||
|  | ||||
| @ -96,6 +102,7 @@ impl crate::docs::StdLibFn for Show { | ||||
|             type_: "number".to_string(), | ||||
|             schema, | ||||
|             required: true, | ||||
|             label_required: true, | ||||
|         }) | ||||
|     } | ||||
|  | ||||
| @ -112,8 +119,8 @@ impl crate::docs::StdLibFn for Show { | ||||
|         code_blocks | ||||
|             .iter() | ||||
|             .map(|cb| { | ||||
|                 let program = crate::Program::parse(cb).unwrap(); | ||||
|                 let mut options: crate::ast::types::FormatOptions = Default::default(); | ||||
|                 let program = crate::Program::parse_no_errs(cb).unwrap(); | ||||
|                 let mut options: crate::parsing::ast::types::FormatOptions = Default::default(); | ||||
|                 options.insert_final_newline = false; | ||||
|                 program.ast.recast(&options, 0) | ||||
|             }) | ||||
|  | ||||
| @ -3,7 +3,8 @@ mod test_examples_my_func { | ||||
|     #[tokio::test(flavor = "multi_thread")] | ||||
|     async fn test_mock_example_my_func0() { | ||||
|         let program = | ||||
|             crate::Program::parse("This is another code block.\nyes sirrr.\nmyFunc").unwrap(); | ||||
|             crate::Program::parse_no_errs("This is another code block.\nyes sirrr.\nmyFunc") | ||||
|                 .unwrap(); | ||||
|         let ctx = crate::executor::ExecutorContext { | ||||
|             engine: std::sync::Arc::new(Box::new( | ||||
|                 crate::engine::conn_mock::EngineConnection::new() | ||||
| @ -15,7 +16,7 @@ mod test_examples_my_func { | ||||
|             settings: Default::default(), | ||||
|             context_type: crate::executor::ContextType::Mock, | ||||
|         }; | ||||
|         ctx.run(&program, &mut crate::ExecState::default()) | ||||
|         ctx.run(program.into(), &mut crate::ExecState::default()) | ||||
|             .await | ||||
|             .unwrap(); | ||||
|     } | ||||
| @ -36,7 +37,8 @@ mod test_examples_my_func { | ||||
|  | ||||
|     #[tokio::test(flavor = "multi_thread")] | ||||
|     async fn test_mock_example_my_func1() { | ||||
|         let program = crate::Program::parse("This is code.\nIt does other shit.\nmyFunc").unwrap(); | ||||
|         let program = | ||||
|             crate::Program::parse_no_errs("This is code.\nIt does other shit.\nmyFunc").unwrap(); | ||||
|         let ctx = crate::executor::ExecutorContext { | ||||
|             engine: std::sync::Arc::new(Box::new( | ||||
|                 crate::engine::conn_mock::EngineConnection::new() | ||||
| @ -48,7 +50,7 @@ mod test_examples_my_func { | ||||
|             settings: Default::default(), | ||||
|             context_type: crate::executor::ContextType::Mock, | ||||
|         }; | ||||
|         ctx.run(&program, &mut crate::ExecState::default()) | ||||
|         ctx.run(program.into(), &mut crate::ExecState::default()) | ||||
|             .await | ||||
|             .unwrap(); | ||||
|     } | ||||
| @ -108,6 +110,10 @@ impl crate::docs::StdLibFn for MyFunc { | ||||
|         vec![] | ||||
|     } | ||||
|  | ||||
|     fn keyword_arguments(&self) -> bool { | ||||
|         false | ||||
|     } | ||||
|  | ||||
|     fn args(&self, inline_subschemas: bool) -> Vec<crate::docs::StdLibFnArg> { | ||||
|         let mut settings = schemars::gen::SchemaSettings::openapi3(); | ||||
|         settings.inline_subschemas = inline_subschemas; | ||||
| @ -117,6 +123,7 @@ impl crate::docs::StdLibFn for MyFunc { | ||||
|             type_: "kittycad::types::InputFormat".to_string(), | ||||
|             schema: generator.root_schema_for::<Option<kittycad::types::InputFormat>>(), | ||||
|             required: false, | ||||
|             label_required: true, | ||||
|         }] | ||||
|     } | ||||
|  | ||||
| @ -130,6 +137,7 @@ impl crate::docs::StdLibFn for MyFunc { | ||||
|             type_: "[Sketch]".to_string(), | ||||
|             schema, | ||||
|             required: true, | ||||
|             label_required: true, | ||||
|         }) | ||||
|     } | ||||
|  | ||||
| @ -149,8 +157,8 @@ impl crate::docs::StdLibFn for MyFunc { | ||||
|         code_blocks | ||||
|             .iter() | ||||
|             .map(|cb| { | ||||
|                 let program = crate::Program::parse(cb).unwrap(); | ||||
|                 let mut options: crate::ast::types::FormatOptions = Default::default(); | ||||
|                 let program = crate::Program::parse_no_errs(cb).unwrap(); | ||||
|                 let mut options: crate::parsing::ast::types::FormatOptions = Default::default(); | ||||
|                 options.insert_final_newline = false; | ||||
|                 program.ast.recast(&options, 0) | ||||
|             }) | ||||
|  | ||||
| @ -3,7 +3,8 @@ mod test_examples_line_to { | ||||
|     #[tokio::test(flavor = "multi_thread")] | ||||
|     async fn test_mock_example_line_to0() { | ||||
|         let program = | ||||
|             crate::Program::parse("This is another code block.\nyes sirrr.\nlineTo").unwrap(); | ||||
|             crate::Program::parse_no_errs("This is another code block.\nyes sirrr.\nlineTo") | ||||
|                 .unwrap(); | ||||
|         let ctx = crate::executor::ExecutorContext { | ||||
|             engine: std::sync::Arc::new(Box::new( | ||||
|                 crate::engine::conn_mock::EngineConnection::new() | ||||
| @ -15,7 +16,7 @@ mod test_examples_line_to { | ||||
|             settings: Default::default(), | ||||
|             context_type: crate::executor::ContextType::Mock, | ||||
|         }; | ||||
|         ctx.run(&program, &mut crate::ExecState::default()) | ||||
|         ctx.run(program.into(), &mut crate::ExecState::default()) | ||||
|             .await | ||||
|             .unwrap(); | ||||
|     } | ||||
| @ -36,7 +37,8 @@ mod test_examples_line_to { | ||||
|  | ||||
|     #[tokio::test(flavor = "multi_thread")] | ||||
|     async fn test_mock_example_line_to1() { | ||||
|         let program = crate::Program::parse("This is code.\nIt does other shit.\nlineTo").unwrap(); | ||||
|         let program = | ||||
|             crate::Program::parse_no_errs("This is code.\nIt does other shit.\nlineTo").unwrap(); | ||||
|         let ctx = crate::executor::ExecutorContext { | ||||
|             engine: std::sync::Arc::new(Box::new( | ||||
|                 crate::engine::conn_mock::EngineConnection::new() | ||||
| @ -48,7 +50,7 @@ mod test_examples_line_to { | ||||
|             settings: Default::default(), | ||||
|             context_type: crate::executor::ContextType::Mock, | ||||
|         }; | ||||
|         ctx.run(&program, &mut crate::ExecState::default()) | ||||
|         ctx.run(program.into(), &mut crate::ExecState::default()) | ||||
|             .await | ||||
|             .unwrap(); | ||||
|     } | ||||
| @ -108,6 +110,10 @@ impl crate::docs::StdLibFn for LineTo { | ||||
|         vec![] | ||||
|     } | ||||
|  | ||||
|     fn keyword_arguments(&self) -> bool { | ||||
|         false | ||||
|     } | ||||
|  | ||||
|     fn args(&self, inline_subschemas: bool) -> Vec<crate::docs::StdLibFnArg> { | ||||
|         let mut settings = schemars::gen::SchemaSettings::openapi3(); | ||||
|         settings.inline_subschemas = inline_subschemas; | ||||
| @ -118,12 +124,14 @@ impl crate::docs::StdLibFn for LineTo { | ||||
|                 type_: "LineToData".to_string(), | ||||
|                 schema: generator.root_schema_for::<LineToData>(), | ||||
|                 required: true, | ||||
|                 label_required: true, | ||||
|             }, | ||||
|             crate::docs::StdLibFnArg { | ||||
|                 name: "sketch".to_string(), | ||||
|                 type_: "Sketch".to_string(), | ||||
|                 schema: generator.root_schema_for::<Sketch>(), | ||||
|                 required: true, | ||||
|                 label_required: true, | ||||
|             }, | ||||
|         ] | ||||
|     } | ||||
| @ -138,6 +146,7 @@ impl crate::docs::StdLibFn for LineTo { | ||||
|             type_: "Sketch".to_string(), | ||||
|             schema, | ||||
|             required: true, | ||||
|             label_required: true, | ||||
|         }) | ||||
|     } | ||||
|  | ||||
| @ -157,8 +166,8 @@ impl crate::docs::StdLibFn for LineTo { | ||||
|         code_blocks | ||||
|             .iter() | ||||
|             .map(|cb| { | ||||
|                 let program = crate::Program::parse(cb).unwrap(); | ||||
|                 let mut options: crate::ast::types::FormatOptions = Default::default(); | ||||
|                 let program = crate::Program::parse_no_errs(cb).unwrap(); | ||||
|                 let mut options: crate::parsing::ast::types::FormatOptions = Default::default(); | ||||
|                 options.insert_final_newline = false; | ||||
|                 program.ast.recast(&options, 0) | ||||
|             }) | ||||
|  | ||||
