* fix: Needed to increase timeout for this test. waitForExecutionDone does not work since there are errors in the KCL code * fix: again another wait for execution does not work * fix: bug that desyncs codeManager, executeCode, lspPlugin * fix: fixing react race condition on parsing numeric literals in command bar on open * fix: adding comment to clarify the gotcha * fix: saving off debugging... * fix: added wait for execution done * fix: removing testing code * fix: adding wait for execution done * fix: adding execution done wait * fix: only fixes the chamfer point and click delete * fix: add 1250ms before every progresscmdbar, removed model loaded scene state that is flaky, added toast if someone presses the command bar too quickly * fix: adding a wait for execution * fix: updating wait for execution * fix: wait for execution done * fix: wait for execution done * fix: not waiting for scene, not waiting for command bar * fix: restoring name * fix: auto fixes * A snapshot a day keeps the bugs away! 📷🐛 (OS: namespace-profile-ubuntu-8-cores) * A snapshot a day keeps the bugs away! 📷🐛 (OS: namespace-profile-ubuntu-8-cores) * A snapshot a day keeps the bugs away! 📷🐛 (OS: namespace-profile-ubuntu-8-cores) * A snapshot a day keeps the bugs away! 📷🐛 (OS: namespace-profile-ubuntu-8-cores) * A snapshot a day keeps the bugs away! 📷🐛 (OS: namespace-profile-ubuntu-8-cores) * A snapshot a day keeps the bugs away! 📷🐛 (OS: namespace-profile-ubuntu-8-cores) * fix: bad prompt fix * fix: Fixed testing selections with wait * fix: last wait fix * fix: trying to resolve more flakes when running with more workers * chore: adding a skipLocalEngine tag * fix: fixing test when using local engine * fix: codespell * fix: big if true * chore: skipping one more local engine tests that fails * fix: auto fix: * fix: restoring this testing code --------- Co-authored-by: github-actions[bot] <github-actions[bot]@users.noreply.github.com>
		
			
				
	
	
		
			1200 lines
		
	
	
		
			39 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			1200 lines
		
	
	
		
			39 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import { test, expect } from './zoo-test'
 | |
| import fsp from 'fs/promises'
 | |
| import { uuidv4 } from 'lib/utils'
 | |
| import {
 | |
|   darkModeBgColor,
 | |
|   darkModePlaneColorXZ,
 | |
|   executorInputPath,
 | |
|   getUtils,
 | |
| } from './test-utils'
 | |
| 
 | |
| import { join } from 'path'
 | |
| 
 | |
| test.describe('Editor tests', { tag: ['@skipWin'] }, () => {
 | |
|   test('can comment out code with ctrl+/', async ({ page, homePage }) => {
 | |
|     const u = await getUtils(page)
 | |
|     await page.setBodyDimensions({ width: 1000, height: 500 })
 | |
| 
 | |
|     await homePage.goToModelingScene()
 | |
| 
 | |
|     // check no error to begin with
 | |
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible()
 | |
| 
 | |
|     await u.codeLocator.click()
 | |
|     await page.keyboard.type(`sketch001 = startSketchOn('XY')
 | |
|     |> startProfileAt([-10, -10], %)
 | |
|     |> line(end = [20, 0])
 | |
|     |> line(end = [0, 20])
 | |
|     |> line(end = [-20, 0])
 | |
|     |> close()`)
 | |
| 
 | |
|     await page.keyboard.down('ControlOrMeta')
 | |
|     await page.keyboard.press('/')
 | |
|     await page.keyboard.up('ControlOrMeta')
 | |
| 
 | |
|     await expect(page.locator('.cm-content'))
 | |
|       .toHaveText(`sketch001 = startSketchOn('XY')
 | |
|   |> startProfileAt([-10, -10], %)
 | |
|   |> line(end = [20, 0])
 | |
|   |> line(end = [0, 20])
 | |
|   |> line(end = [-20, 0])
 | |
|   // |> close()`)
 | |
| 
 | |
|     // uncomment the code
 | |
|     await page.keyboard.down('ControlOrMeta')
 | |
|     await page.keyboard.press('/')
 | |
|     await page.keyboard.up('ControlOrMeta')
 | |
| 
 | |
|     await expect(page.locator('.cm-content'))
 | |
|       .toHaveText(`sketch001 = startSketchOn('XY')
 | |
|   |> startProfileAt([-10, -10], %)
 | |
|   |> line(end = [20, 0])
 | |
|   |> line(end = [0, 20])
 | |
|   |> line(end = [-20, 0])
 | |
|   |> close()`)
 | |
|   })
 | |
| 
 | |
|   test('ensure we use the cache, and do not re-execute', async ({
 | |
|     homePage,
 | |
|     page,
 | |
|   }) => {
 | |
|     const u = await getUtils(page)
 | |
|     await page.setBodyDimensions({ width: 1000, height: 500 })
 | |
| 
 | |
|     await homePage.goToModelingScene()
 | |
|     await u.waitForPageLoad()
 | |
| 
 | |
|     await u.codeLocator.click()
 | |
|     await page.keyboard.type(`sketch001 = startSketchOn('XY')
 | |
|   |> startProfileAt([-10, -10], %)
 | |
|   |> line(end = [20, 0])
 | |
|   |> line(end = [0, 20])
 | |
|   |> line(end = [-20, 0])
 | |
|   |> close()`)
 | |
| 
 | |
|     // Ensure we execute the first time.
 | |
|     await u.openDebugPanel()
 | |
|     await expect(
 | |
|       page.locator('[data-receive-command-type="scene_clear_all"]')
 | |
|     ).toHaveCount(1)
 | |
|     await expect(
 | |
|       page.locator('[data-message-type="execution-done"]')
 | |
|     ).toHaveCount(2)
 | |
| 
 | |
|     // Add whitespace to the end of the code.
 | |
|     await u.codeLocator.click()
 | |
|     await page.keyboard.press('ArrowUp')
 | |
|     await page.keyboard.press('ArrowUp')
 | |
|     await page.keyboard.press('ArrowUp')
 | |
|     await page.keyboard.press('ArrowUp')
 | |
|     await page.keyboard.press('Home')
 | |
|     await page.keyboard.type('    ')
 | |
|     await page.keyboard.press('Enter')
 | |
|     await page.keyboard.type('    ')
 | |
| 
 | |
|     // Ensure we don't execute the second time.
 | |
|     await u.openDebugPanel()
 | |
|     // Make sure we didn't clear the scene.
 | |
|     await expect(
 | |
|       page.locator('[data-message-type="execution-done"]')
 | |
|     ).toHaveCount(3)
 | |
|     await expect(
 | |
|       page.locator('[data-receive-command-type="scene_clear_all"]')
 | |
|     ).toHaveCount(1)
 | |
|   })
 | |
| 
 | |
|   test('ensure we use the cache, and do not clear on append', async ({
 | |
|     homePage,
 | |
|     page,
 | |
|   }) => {
 | |
|     const u = await getUtils(page)
 | |
|     await page.setBodyDimensions({ width: 1000, height: 500 })
 | |
| 
 | |
|     await homePage.goToModelingScene()
 | |
|     await u.waitForPageLoad()
 | |
| 
 | |
|     await u.codeLocator.click()
 | |
|     await page.keyboard.type(`sketch001 = startSketchOn('XY')
 | |
|   |> startProfileAt([-10, -10], %)
 | |
|   |> line(end = [20, 0])
 | |
|   |> line(end = [0, 20])
 | |
|   |> line(end = [-20, 0])
 | |
|   |> close()`)
 | |
| 
 | |
|     // Ensure we execute the first time.
 | |
|     await u.openDebugPanel()
 | |
|     await expect(
 | |
|       page.locator('[data-receive-command-type="scene_clear_all"]')
 | |
|     ).toHaveCount(1)
 | |
|     await expect(
 | |
|       page.locator('[data-message-type="execution-done"]')
 | |
|     ).toHaveCount(2)
 | |
| 
 | |
|     // Add whitespace to the end of the code.
 | |
|     await u.codeLocator.click()
 | |
|     await page.keyboard.press('ArrowDown')
 | |
|     await page.keyboard.press('ArrowDown')
 | |
|     await page.keyboard.press('ArrowDown')
 | |
|     await page.keyboard.press('ArrowDown')
 | |
|     await page.keyboard.press('ArrowDown')
 | |
|     await page.keyboard.press('ArrowDown')
 | |
|     await page.keyboard.press('ArrowDown')
 | |
|     await page.keyboard.press('ArrowDown')
 | |
|     await page.keyboard.press('ArrowDown')
 | |
|     await page.keyboard.press('End')
 | |
|     await page.keyboard.press('Enter')
 | |
|     await page.keyboard.press('Enter')
 | |
|     await page.keyboard.type('const x = 1')
 | |
|     await page.keyboard.press('Enter')
 | |
| 
 | |
|     await u.openDebugPanel()
 | |
|     await expect(
 | |
|       page.locator('[data-message-type="execution-done"]')
 | |
|     ).toHaveCount(3)
 | |
|     await expect(
 | |
|       page.locator('[data-receive-command-type="scene_clear_all"]')
 | |
|     ).toHaveCount(1)
 | |
|   })
 | |
| 
 | |
|   test('if you click the format button it formats your code', async ({
 | |
|     page,
 | |
|     homePage,
 | |
|   }) => {
 | |
|     const u = await getUtils(page)
 | |
|     await page.setBodyDimensions({ width: 1000, height: 500 })
 | |
| 
 | |
|     await homePage.goToModelingScene()
 | |
| 
 | |
|     // check no error to begin with
 | |
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible()
 | |
| 
 | |
|     await u.codeLocator.click()
 | |
|     await page.keyboard.type(`sketch001 = startSketchOn('XY')
 | |
|     |> startProfileAt([-10, -10], %)
 | |
|     |> line(end = [20, 0])
 | |
|     |> line(end = [0, 20])
 | |
|     |> line(end = [-20, 0])
 | |
|     |> close()`)
 | |
|     await page.locator('#code-pane button:first-child').click()
 | |
|     await page.locator('button:has-text("Format code")').click()
 | |
| 
 | |
|     await expect(page.locator('.cm-content'))
 | |
|       .toHaveText(`sketch001 = startSketchOn('XY')
 | |
|   |> startProfileAt([-10, -10], %)
 | |
|   |> line(end = [20, 0])
 | |
|   |> line(end = [0, 20])
 | |
|   |> line(end = [-20, 0])
 | |
|   |> close()`)
 | |
|   })
 | |
| 
 | |
|   test('if you click the format button it formats your code and executes so lints are still there', async ({
 | |
|     page,
 | |
|     homePage,
 | |
|   }) => {
 | |
|     const u = await getUtils(page)
 | |
|     await page.setBodyDimensions({ width: 1000, height: 500 })
 | |
| 
 | |
|     await homePage.goToModelingScene()
 | |
| 
 | |
|     // check no error to begin with
 | |
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible()
 | |
| 
 | |
|     await u.codeLocator.click()
 | |
|     await page.keyboard.type(`sketch_001 = startSketchOn('XY')
 | |
|     |> startProfileAt([-10, -10], %)
 | |
|     |> line(end = [20, 0])
 | |
|     |> line(end = [0, 20])
 | |
|     |> line(end = [-20, 0])
 | |
|     |> close()`)
 | |
| 
 | |
|     await u.openDebugPanel()
 | |
|     await u.expectCmdLog('[data-message-type="execution-done"]')
 | |
|     await u.closeDebugPanel()
 | |
| 
 | |
|     // error in guter
 | |
|     await expect(page.locator('.cm-lint-marker-info').first()).toBeVisible()
 | |
| 
 | |
|     // error text on hover
 | |
|     await page.hover('.cm-lint-marker-info')
 | |
|     await expect(
 | |
|       page.getByText('Identifiers must be lowerCamelCase').first()
 | |
|     ).toBeVisible()
 | |
| 
 | |
|     await page.locator('#code-pane button:first-child').click()
 | |
|     await page.locator('button:has-text("Format code")').click()
 | |
| 
 | |
|     await u.openDebugPanel()
 | |
|     await u.expectCmdLog('[data-message-type="execution-done"]')
 | |
|     await u.closeDebugPanel()
 | |
| 
 | |
|     await expect(page.locator('.cm-content'))
 | |
|       .toHaveText(`sketch_001 = startSketchOn('XY')
 | |
|   |> startProfileAt([-10, -10], %)
 | |
|   |> line(end = [20, 0])
 | |
|   |> line(end = [0, 20])
 | |
|   |> line(end = [-20, 0])
 | |
|   |> close()`)
 | |
| 
 | |
|     // error in guter
 | |
|     await expect(page.locator('.cm-lint-marker-info').first()).toBeVisible()
 | |
| 
 | |
|     // error text on hover
 | |
|     await page.hover('.cm-lint-marker-info')
 | |
|     await expect(
 | |
|       page.getByText('Identifiers must be lowerCamelCase').first()
 | |
|     ).toBeVisible()
 | |
|   })
 | |
| 
 | |
|   test('fold gutters work', async ({ page, homePage }) => {
 | |
|     const fullCode = `sketch001 = startSketchOn('XY')
 | |
|    |> startProfileAt([-10, -10], %)
 | |
|    |> line(end = [20, 0])
 | |
|    |> line(end = [0, 20])
 | |
|    |> line(end = [-20, 0])
 | |
|    |> close()`
 | |
|     await page.addInitScript(async () => {
 | |
|       localStorage.setItem(
 | |
|         'persistCode',
 | |
|         `sketch001 = startSketchOn('XY')
 | |
|    |> startProfileAt([-10, -10], %)
 | |
|    |> line(end = [20, 0])
 | |
|    |> line(end = [0, 20])
 | |
|    |> line(end = [-20, 0])
 | |
|    |> close()`
 | |
|       )
 | |
|     })
 | |
|     await page.setBodyDimensions({ width: 1000, height: 500 })
 | |
| 
 | |
|     await homePage.goToModelingScene()
 | |
| 
 | |
|     // TODO: Jess needs to fix this but you have to mod the code to get them to show
 | |
|     // up, its an annoying codemirror thing.
 | |
|     await page.locator('.cm-content').click()
 | |
|     await page.keyboard.press('ArrowDown')
 | |
|     await page.keyboard.press('ArrowDown')
 | |
|     await page.keyboard.press('ArrowDown')
 | |
|     await page.keyboard.press('ArrowDown')
 | |
|     await page.keyboard.press('ArrowDown')
 | |
|     await page.keyboard.press('Enter')
 | |
| 
 | |
|     const foldGutterFoldLine = page.locator('[title="Fold line"]')
 | |
|     const foldGutterUnfoldLine = page.locator('[title="Unfold line"]')
 | |
| 
 | |
|     await expect(page.locator('.cm-content')).toHaveText(fullCode)
 | |
| 
 | |
|     // check no error to begin with
 | |
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible()
 | |
| 
 | |
|     // Make sure we have a fold gutter
 | |
|     await expect(foldGutterFoldLine).toBeVisible()
 | |
|     await expect(foldGutterUnfoldLine).not.toBeVisible()
 | |
| 
 | |
|     // Collapse the code
 | |
|     await foldGutterFoldLine.click()
 | |
| 
 | |
|     await expect(page.locator('.cm-content')).toHaveText(
 | |
|       `sketch001 = startSketchOn('XY')…   `
 | |
|     )
 | |
|     await expect(page.locator('.cm-content')).not.toHaveText(fullCode)
 | |
|     await expect(foldGutterFoldLine).not.toBeVisible()
 | |
|     await expect(foldGutterUnfoldLine.nth(1)).toBeVisible()
 | |
| 
 | |
|     // Expand the code
 | |
|     await foldGutterUnfoldLine.nth(1).click()
 | |
|     await expect(page.locator('.cm-content')).toHaveText(fullCode)
 | |
| 
 | |
|     // Delete all the code.
 | |
|     await page.locator('.cm-content').click()
 | |
|     // Select all
 | |
|     await page.keyboard.press('ControlOrMeta+A')
 | |
|     await page.keyboard.press('Backspace')
 | |
| 
 | |
|     await expect(page.locator('.cm-content')).toHaveText(``)
 | |
|     await expect(page.locator('.cm-content')).not.toHaveText(fullCode)
 | |
| 
 | |
|     await expect(foldGutterUnfoldLine).not.toBeVisible()
 | |
|     await expect(foldGutterFoldLine).not.toBeVisible()
 | |
|   })
 | |
| 
 | |
|   test('hover over functions shows function description', async ({
 | |
|     page,
 | |
|     homePage,
 | |
|   }) => {
 | |
|     const u = await getUtils(page)
 | |
|     await page.addInitScript(async () => {
 | |
|       localStorage.setItem(
 | |
|         'persistCode',
 | |
|         `sketch001 = startSketchOn('XY')
 | |
|     |> startProfileAt([-10, -10], %)
 | |
|     |> line(end = [20, 0])
 | |
|     |> line(end = [0, 20])
 | |
|     |> line(end = [-20, 0])
 | |
|     |> close()`
 | |
|       )
 | |
|     })
 | |
|     await page.setBodyDimensions({ width: 1000, height: 500 })
 | |
| 
 | |
|     await homePage.goToModelingScene()
 | |
| 
 | |
|     // check no error to begin with
 | |
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible()
 | |
| 
 | |
|     await u.openDebugPanel()
 | |
|     await u.expectCmdLog('[data-message-type="execution-done"]')
 | |
|     await u.closeDebugPanel()
 | |
| 
 | |
|     // focus the editor
 | |
|     await u.codeLocator.click()
 | |
| 
 | |
|     // Hover over  the startSketchOn function
 | |
|     await page.getByText('startSketchOn').hover()
 | |
|     await expect(page.locator('.hover-tooltip')).toBeVisible()
 | |
|     await expect(
 | |
|       page.getByText(
 | |
|         'Start a new 2-dimensional sketch on a specific plane or face'
 | |
|       )
 | |
|     ).toBeVisible()
 | |
| 
 | |
|     // Hover over the line function
 | |
|     await page.getByText('line').first().hover()
 | |
|     await expect(page.locator('.hover-tooltip')).toBeVisible()
 | |
|     await expect(
 | |
|       page.getByText('Extend the current sketch with a new straight line.')
 | |
|     ).toBeVisible()
 | |
|   })
 | |
| 
 | |
|   test('if you use the format keyboard binding it formats your code', async ({
 | |
|     page,
 | |
|     homePage,
 | |
|   }) => {
 | |
|     const u = await getUtils(page)
 | |
|     await page.addInitScript(async () => {
 | |
|       localStorage.setItem(
 | |
|         'persistCode',
 | |
|         `sketch001 = startSketchOn('XY')
 | |
|     |> startProfileAt([-10, -10], %)
 | |
|     |> line(end = [20, 0])
 | |
|     |> line(end = [0, 20])
 | |
|     |> line(end = [-20, 0])
 | |
|     |> close()`
 | |
|       )
 | |
|       localStorage.setItem('disableAxis', 'true')
 | |
|     })
 | |
|     await page.setBodyDimensions({ width: 1000, height: 500 })
 | |
| 
 | |
|     await homePage.goToModelingScene()
 | |
| 
 | |
|     // check no error to begin with
 | |
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible()
 | |
| 
 | |
|     await u.openDebugPanel()
 | |
|     await u.expectCmdLog('[data-message-type="execution-done"]')
 | |
|     await u.closeDebugPanel()
 | |
| 
 | |
|     // focus the editor
 | |
|     await u.codeLocator.click()
 | |
| 
 | |
|     // Hit alt+shift+f to format the code
 | |
|     await page.keyboard.press('Alt+Shift+KeyF')
 | |
| 
 | |
|     await expect(page.locator('.cm-content'))
 | |
|       .toHaveText(`sketch001 = startSketchOn('XY')
 | |
|   |> startProfileAt([-10, -10], %)
 | |
|   |> line(end = [20, 0])
 | |
|   |> line(end = [0, 20])
 | |
|   |> line(end = [-20, 0])
 | |
|   |> close()`)
 | |
|   })
 | |
| 
 | |
|   test('if you use the format keyboard binding it formats your code and executes so lints are shown', async ({
 | |
|     page,
 | |
|     homePage,
 | |
|   }) => {
 | |
|     const u = await getUtils(page)
 | |
|     await page.addInitScript(async () => {
 | |
|       localStorage.setItem(
 | |
|         'persistCode',
 | |
|         `sketch_001 = startSketchOn('XY')
 | |
|     |> startProfileAt([-10, -10], %)
 | |
|     |> line(end = [20, 0])
 | |
|     |> line(end = [0, 20])
 | |
|     |> line(end = [-20, 0])
 | |
|     |> close()`
 | |
|       )
 | |
|       localStorage.setItem('disableAxis', 'true')
 | |
|     })
 | |
|     await page.setBodyDimensions({ width: 1000, height: 500 })
 | |
| 
 | |
|     await homePage.goToModelingScene()
 | |
| 
 | |
|     await u.openDebugPanel()
 | |
|     await u.expectCmdLog('[data-message-type="execution-done"]')
 | |
|     await u.closeDebugPanel()
 | |
| 
 | |
|     // error in guter
 | |
|     await expect(page.locator('.cm-lint-marker-info').first()).toBeVisible()
 | |
| 
 | |
|     // error text on hover
 | |
|     await page.hover('.cm-lint-marker-info')
 | |
|     await expect(
 | |
|       page.getByText('Identifiers must be lowerCamelCase').first()
 | |
|     ).toBeVisible()
 | |
| 
 | |
|     // focus the editor
 | |
|     await u.codeLocator.click()
 | |
| 
 | |
|     // Hit alt+shift+f to format the code
 | |
|     await page.keyboard.press('Alt+Shift+KeyF')
 | |
| 
 | |
|     await u.openDebugPanel()
 | |
|     await u.expectCmdLog('[data-message-type="execution-done"]')
 | |
|     await u.closeDebugPanel()
 | |
| 
 | |
|     await expect(page.locator('.cm-content'))
 | |
|       .toHaveText(`sketch_001 = startSketchOn('XY')
 | |
|   |> startProfileAt([-10, -10], %)
 | |
|   |> line(end = [20, 0])
 | |
|   |> line(end = [0, 20])
 | |
|   |> line(end = [-20, 0])
 | |
|   |> close()`)
 | |
| 
 | |
|     // error in guter
 | |
|     await expect(page.locator('.cm-lint-marker-info').first()).toBeVisible()
 | |
| 
 | |
|     // error text on hover
 | |
|     await page.hover('.cm-lint-marker-info')
 | |
|     await expect(
 | |
|       page.getByText('Identifiers must be lowerCamelCase').first()
 | |
|     ).toBeVisible()
 | |
|   })
 | |
| 
 | |
|   test('if you write kcl with lint errors you get lints', async ({
 | |
|     page,
 | |
|     homePage,
 | |
|   }) => {
 | |
|     const u = await getUtils(page)
 | |
|     await page.setBodyDimensions({ width: 1000, height: 500 })
 | |
| 
 | |
|     await homePage.goToModelingScene()
 | |
| 
 | |
|     // check no error to begin with
 | |
|     await expect(page.locator('.cm-lint-marker-info')).not.toBeVisible()
 | |
| 
 | |
|     await u.codeLocator.click()
 | |
|     await page.keyboard.type('my_snake_case_var = 5')
 | |
|     await page.keyboard.press('Enter')
 | |
|     await page.keyboard.type('myCamelCaseVar = 5')
 | |
|     await page.keyboard.press('Enter')
 | |
| 
 | |
|     // press arrows to clear autocomplete
 | |
|     await page.keyboard.press('ArrowLeft')
 | |
|     await page.keyboard.press('ArrowRight')
 | |
| 
 | |
|     // FIXME: lsp errors do not propagate to the frontend until engine is connected and code is executed
 | |
|     // This timeout is to wait for engine connection. LSP and code execution errors should be handled differently
 | |
|     // LSP can emit errors as fast as it waits and show them in the editor
 | |
|     await page.waitForTimeout(10000)
 | |
| 
 | |
|     // error in guter
 | |
|     await expect(page.locator('.cm-lint-marker-info').first()).toBeVisible()
 | |
| 
 | |
|     // error text on hover
 | |
|     await page.hover('.cm-lint-marker-info')
 | |
|     await expect(
 | |
|       page.getByText('Identifiers must be lowerCamelCase').first()
 | |
|     ).toBeVisible()
 | |
| 
 | |
|     // select the line that's causing the error and delete it
 | |
|     await page.getByText('my_snake_case_var = 5').click()
 | |
|     await page.keyboard.press('End')
 | |
|     await page.keyboard.down('Shift')
 | |
|     await page.keyboard.press('Home')
 | |
|     await page.keyboard.up('Shift')
 | |
|     await page.keyboard.press('Backspace')
 | |
| 
 | |
|     // wait for .cm-lint-marker-info not to be visible
 | |
|     await expect(page.locator('.cm-lint-marker-info')).not.toBeVisible()
 | |
|   })
 | |
| 
 | |
|   test('if you fixup kcl errors you clear lints', async ({
 | |
|     page,
 | |
|     homePage,
 | |
|   }) => {
 | |
|     const u = await getUtils(page)
 | |
|     await page.addInitScript(async () => {
 | |
|       localStorage.setItem(
 | |
|         'persistCode',
 | |
|         `sketch001 = startSketchOn('XZ')
 | |
|     |> startProfileAt([3.29, 7.86], %)
 | |
|     |> line(end = [2.48, 2.44])
 | |
|     |> line(end = [2.66, 1.17])
 | |
|     |> close()
 | |
|     `
 | |
|       )
 | |
|     })
 | |
| 
 | |
|     await page.setBodyDimensions({ width: 1000, height: 500 })
 | |
| 
 | |
|     await homePage.goToModelingScene()
 | |
| 
 | |
|     // check no error to begin with
 | |
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible()
 | |
| 
 | |
|     await u.codeLocator.click()
 | |
| 
 | |
|     await page.getByText(' |> line(end = [2.48, 2.44])').click()
 | |
| 
 | |
|     await expect(
 | |
|       page.locator('.cm-lint-marker-error').first()
 | |
|     ).not.toBeVisible()
 | |
|     await page.keyboard.press('End')
 | |
|     await page.keyboard.press('Backspace')
 | |
| 
 | |
|     await expect(page.locator('.cm-lint-marker-error').first()).toBeVisible()
 | |
|     await page.keyboard.type(')')
 | |
|     await expect(
 | |
|       page.locator('.cm-lint-marker-error').first()
 | |
|     ).not.toBeVisible()
 | |
|   })
 | |
| 
 | |
|   test('if you write invalid kcl you get inlined errors', async ({
 | |
|     page,
 | |
|     homePage,
 | |
|   }) => {
 | |
|     const u = await getUtils(page)
 | |
|     await page.setBodyDimensions({ width: 1200, height: 500 })
 | |
| 
 | |
|     await homePage.goToModelingScene()
 | |
| 
 | |
|     // check no error to begin with
 | |
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible()
 | |
| 
 | |
|     /* add the following code to the editor (~ error is not a valid line)
 | |
|       * the old check here used $ but this is for tags so it changed meaning.
 | |
|       * hopefully ~ doesn't change meaning
 | |
|     ~ error
 | |
|     const topAng = 30
 | |
|     const bottomAng = 25
 | |
|    */
 | |
|     await u.codeLocator.click()
 | |
|     await page.keyboard.type('~ error')
 | |
| 
 | |
|     // press arrows to clear autocomplete
 | |
|     await page.keyboard.press('ArrowLeft')
 | |
|     await page.keyboard.press('ArrowRight')
 | |
| 
 | |
|     await page.keyboard.press('Enter')
 | |
|     await page.keyboard.type('topAng = 30')
 | |
|     await page.keyboard.press('Enter')
 | |
|     await page.keyboard.type('bottomAng = 25')
 | |
|     await page.keyboard.press('Enter')
 | |
| 
 | |
|     // error in guter
 | |
|     await expect(page.locator('.cm-lint-marker-error')).toBeVisible()
 | |
| 
 | |
|     // error text on hover
 | |
|     await page.hover('.cm-lint-marker-error')
 | |
|     await expect(
 | |
|       page.getByText("found unknown token '~'").first()
 | |
|     ).toBeVisible()
 | |
| 
 | |
|     // select the line that's causing the error and delete it
 | |
|     await page.getByText('~ error').click()
 | |
|     await page.keyboard.press('End')
 | |
|     await page.keyboard.down('Shift')
 | |
|     await page.keyboard.press('Home')
 | |
|     await page.keyboard.up('Shift')
 | |
|     await page.keyboard.press('Backspace')
 | |
| 
 | |
|     // wait for .cm-lint-marker-error not to be visible
 | |
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible()
 | |
| 
 | |
|     // let's check we get an error when defining the same variable twice
 | |
|     await page.getByText('bottomAng = 25').click()
 | |
|     await page.keyboard.press('Enter')
 | |
|     await page.keyboard.type("// Let's define the same thing twice")
 | |
|     await page.keyboard.press('Enter')
 | |
|     await page.keyboard.type('topAng = 42')
 | |
|     await page.keyboard.press('ArrowLeft')
 | |
| 
 | |
|     await expect(page.locator('.cm-lint-marker-error')).toBeVisible()
 | |
|     await expect(
 | |
|       page.locator('.cm-lint-marker.cm-lint-marker-error')
 | |
|     ).toBeVisible()
 | |
| 
 | |
|     await page.locator('.cm-lint-marker.cm-lint-marker-error').hover()
 | |
|     await expect(page.locator('.cm-diagnosticText').first()).toBeVisible()
 | |
|     await expect(
 | |
|       page.getByText('Cannot redefine `topAng`').first()
 | |
|     ).toBeVisible()
 | |
| 
 | |
|     const secondTopAng = page.getByText('topAng').first()
 | |
|     await secondTopAng?.dblclick()
 | |
|     await page.keyboard.type('otherAng')
 | |
| 
 | |
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible()
 | |
|   })
 | |
| 
 | |
|   test.fixme(
 | |
|     'error with 2 source ranges gets 2 diagnostics',
 | |
|     async ({ page, homePage }) => {
 | |
|       const u = await getUtils(page)
 | |
|       await page.addInitScript(async () => {
 | |
|         localStorage.setItem(
 | |
|           'persistCode',
 | |
|           `length = .750
 | |
|     width = 0.500
 | |
|     height = 0.500
 | |
|     dia = 4
 | |
| 
 | |
|     fn squareHole = (l, w) => {
 | |
|   squareHoleSketch = startSketchOn('XY')
 | |
|   |> startProfileAt([-width / 2, -length / 2], %)
 | |
|   |> line(endAbsolute = [width / 2, -length / 2])
 | |
|   |> line(endAbsolute = [width / 2, length / 2])
 | |
|   |> line(endAbsolute = [-width / 2, length / 2])
 | |
|   |> close()
 | |
|   return squareHoleSketch
 | |
|     }
 | |
|     `
 | |
|         )
 | |
|       })
 | |
|       await page.setBodyDimensions({ width: 1000, height: 500 })
 | |
| 
 | |
|       await homePage.goToModelingScene()
 | |
|       await u.waitForPageLoad()
 | |
|       await page.waitForTimeout(1000)
 | |
| 
 | |
|       await u.openDebugPanel()
 | |
|       await u.expectCmdLog('[data-message-type="execution-done"]')
 | |
|       await u.closeDebugPanel()
 | |
| 
 | |
|       // check no error to begin with
 | |
|       await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible()
 | |
| 
 | |
|       // Click on the bottom of the code editor to add a new line
 | |
|       await u.codeLocator.click()
 | |
|       await page.keyboard.press('ArrowDown')
 | |
|       await page.keyboard.press('ArrowDown')
 | |
|       await page.keyboard.press('ArrowDown')
 | |
|       await page.keyboard.press('ArrowDown')
 | |
|       await page.keyboard.press('ArrowDown')
 | |
|       await page.keyboard.press('ArrowDown')
 | |
|       await page.keyboard.press('ArrowDown')
 | |
|       await page.keyboard.press('ArrowDown')
 | |
|       await page.keyboard.press('ArrowDown')
 | |
|       await page.keyboard.press('ArrowDown')
 | |
|       await page.keyboard.press('ArrowDown')
 | |
|       await page.keyboard.press('ArrowDown')
 | |
|       await page.keyboard.press('ArrowDown')
 | |
|       await page.keyboard.press('Enter')
 | |
|       await page.keyboard.type(`extrusion = startSketchOn('XY')
 | |
|   |> circle({ center: [0, 0], radius: dia/2 }, %)
 | |
|     |> hole(squareHole(length, width, height), %)
 | |
|     |> extrude(length = height)`)
 | |
| 
 | |
|       // error in gutter
 | |
|       await expect(page.locator('.cm-lint-marker-error').first()).toBeVisible()
 | |
|       await page.hover('.cm-lint-marker-error:first-child')
 | |
|       await expect(
 | |
|         page.getByText('Expected 2 arguments, got 3').first()
 | |
|       ).toBeVisible()
 | |
| 
 | |
|       // Make sure there are two diagnostics
 | |
|       await expect(page.locator('.cm-lint-marker-error')).toHaveCount(2)
 | |
|     }
 | |
|   )
 | |
|   test('if your kcl gets an error from the engine it is inlined', async ({
 | |
|     context,
 | |
|     page,
 | |
|     homePage,
 | |
|   }) => {
 | |
|     await context.addInitScript(async () => {
 | |
|       localStorage.setItem(
 | |
|         'persistCode',
 | |
|         `box = startSketchOn('XY')
 | |
|     |> startProfileAt([0, 0], %)
 | |
|     |> line(end = [0, 10])
 | |
|     |> line(end = [10, 0])
 | |
|     |> line(end = [0, -10], tag = $revolveAxis)
 | |
|     |> close()
 | |
|     |> extrude(length = 10)
 | |
| 
 | |
|     sketch001 = startSketchOn(box, revolveAxis)
 | |
|     |> startProfileAt([5, 10], %)
 | |
|     |> line(end = [0, -10])
 | |
|     |> line(end = [2, 0])
 | |
|     |> line(end = [0, -10])
 | |
|     |> close()
 | |
|     |> revolve({
 | |
|     axis: revolveAxis,
 | |
|     angle: 90
 | |
|     }, %)
 | |
|     `
 | |
|       )
 | |
|     })
 | |
| 
 | |
|     await page.setBodyDimensions({ width: 1000, height: 500 })
 | |
| 
 | |
|     await homePage.goToModelingScene()
 | |
| 
 | |
|     await expect(page.locator('.cm-lint-marker-error')).toBeVisible()
 | |
| 
 | |
|     // error text on hover
 | |
|     await page.hover('.cm-lint-marker-error')
 | |
|     const searchText =
 | |
|       'sketch profile must lie entirely on one side of the revolution axis'
 | |
|     await expect(page.getByText(searchText)).toBeVisible()
 | |
|   })
 | |
|   test.describe('Autocomplete works', () => {
 | |
|     test('with enter/click to accept the completion', async ({
 | |
|       page,
 | |
|       homePage,
 | |
|     }) => {
 | |
|       const u = await getUtils(page)
 | |
|       // const PUR = 400 / 37.5 //pixeltoUnitRatio
 | |
|       await page.setBodyDimensions({ width: 1200, height: 500 })
 | |
| 
 | |
|       await homePage.goToModelingScene()
 | |
| 
 | |
|       // tests clicking on an option, selection the first option
 | |
|       // and arrowing down to an option
 | |
| 
 | |
|       await u.codeLocator.click()
 | |
|       await page.keyboard.type('sketch001 = start')
 | |
| 
 | |
|       // expect there to be some auto complete options
 | |
|       // exact number depends on the KCL stdlib, so let's just check it's > 0 for now.
 | |
|       await expect(async () => {
 | |
|         const children = await page.locator('.cm-completionLabel').count()
 | |
|         expect(children).toBeGreaterThan(0)
 | |
|       }).toPass()
 | |
|       // this makes sure we can accept a completion with click
 | |
|       await page.getByText('startSketchOn').click()
 | |
|       await page.keyboard.type("'XZ'")
 | |
|       await page.keyboard.press('Tab')
 | |
|       await page.keyboard.press('Enter')
 | |
|       await page.keyboard.type('  |> startProfi')
 | |
|       // expect there be a single auto complete option that we can just hit enter on
 | |
|       await expect(page.locator('.cm-completionLabel')).toBeVisible()
 | |
|       await page.waitForTimeout(100)
 | |
|       await page.keyboard.press('Enter') // accepting the auto complete, not a new line
 | |
| 
 | |
|       await page.keyboard.press('Tab')
 | |
|       await page.waitForTimeout(100)
 | |
|       await page.keyboard.type('12')
 | |
|       await page.waitForTimeout(100)
 | |
|       await page.keyboard.press('Tab')
 | |
|       await page.waitForTimeout(100)
 | |
|       await page.keyboard.press('Tab')
 | |
|       await page.waitForTimeout(100)
 | |
|       await page.keyboard.press('Tab')
 | |
|       await page.waitForTimeout(100)
 | |
|       await page.keyboard.press('Enter')
 | |
|       await page.waitForTimeout(100)
 | |
|       await page.keyboard.type('  |> lin')
 | |
| 
 | |
|       await expect(page.locator('.cm-tooltip-autocomplete')).toBeVisible()
 | |
|       await page.waitForTimeout(100)
 | |
|       // press arrow down then enter to accept xLine
 | |
|       await page.keyboard.press('ArrowDown')
 | |
|       await page.keyboard.press('Enter')
 | |
|       // finish line with comment
 | |
|       await page.keyboard.type('5')
 | |
|       await page.waitForTimeout(100)
 | |
|       await page.keyboard.press('Tab')
 | |
|       await page.waitForTimeout(100)
 | |
|       await page.keyboard.press('Tab')
 | |
| 
 | |
|       await page.keyboard.type(' // ')
 | |
|       // Since we need to parse the ast to know we are in a comment we gotta hang tight.
 | |
|       await page.waitForTimeout(700)
 | |
|       await page.keyboard.type('lin ')
 | |
|       await page.waitForTimeout(200)
 | |
|       // there shouldn't be any auto complete options for 'lin' in the comment
 | |
|       await expect(page.locator('.cm-completionLabel')).not.toBeVisible()
 | |
| 
 | |
|       await expect(page.locator('.cm-content'))
 | |
|         .toHaveText(`sketch001 = startSketchOn('XZ')
 | |
|         |> startProfileAt([3.14, 12], %)
 | |
|         |> xLine(5, %) // lin`)
 | |
| 
 | |
|       // expect there to be no KCL errors
 | |
|       await expect(page.locator('.cm-lint-marker-error')).toHaveCount(0)
 | |
|     })
 | |
| 
 | |
|     test('with tab to accept the completion', async ({ page, homePage }) => {
 | |
|       const u = await getUtils(page)
 | |
|       // const PUR = 400 / 37.5 //pixeltoUnitRatio
 | |
|       await page.setBodyDimensions({ width: 1200, height: 500 })
 | |
| 
 | |
|       await homePage.goToModelingScene()
 | |
| 
 | |
|       // this test might be brittle as we add and remove functions
 | |
|       // but should also be easy to update.
 | |
|       // tests clicking on an option, selection the first option
 | |
|       // and arrowing down to an option
 | |
| 
 | |
|       await u.codeLocator.click()
 | |
|       await page.keyboard.type('sketch001 = startSketchO')
 | |
|       await page.waitForTimeout(100)
 | |
| 
 | |
|       // Make sure just hitting tab will take the only one left
 | |
|       await expect(page.locator('.cm-completionLabel')).toHaveCount(1)
 | |
|       await page.waitForTimeout(500)
 | |
|       await page.keyboard.press('ArrowDown')
 | |
|       await page.keyboard.press('Tab')
 | |
|       await page.waitForTimeout(500)
 | |
|       await page.keyboard.type("'XZ'")
 | |
|       await page.keyboard.press('Tab')
 | |
|       await page.keyboard.press('Enter')
 | |
|       await page.keyboard.type('  |> startProfi')
 | |
|       // expect there be a single auto complete option that we can just hit enter on
 | |
|       await expect(page.locator('.cm-completionLabel')).toBeVisible()
 | |
|       await page.waitForTimeout(100)
 | |
|       await page.keyboard.press('Tab') // accepting the auto complete, not a new line
 | |
| 
 | |
|       await page.keyboard.press('Tab')
 | |
|       await page.keyboard.type('12')
 | |
|       await page.waitForTimeout(100)
 | |
|       await page.keyboard.press('Tab')
 | |
|       await page.waitForTimeout(100)
 | |
|       await page.keyboard.press('Tab')
 | |
|       await page.waitForTimeout(100)
 | |
|       await page.keyboard.press('Tab')
 | |
|       await page.waitForTimeout(100)
 | |
|       await page.keyboard.press('Enter')
 | |
|       await page.waitForTimeout(100)
 | |
|       await page.keyboard.type('  |> lin')
 | |
| 
 | |
|       await expect(page.locator('.cm-tooltip-autocomplete')).toBeVisible()
 | |
|       await page.waitForTimeout(100)
 | |
|       // press arrow down then tab to accept xLine
 | |
|       await page.keyboard.press('ArrowDown')
 | |
|       await page.keyboard.press('Tab')
 | |
|       // finish line with comment
 | |
|       await page.keyboard.type('5')
 | |
|       await page.waitForTimeout(100)
 | |
|       await page.keyboard.press('Tab')
 | |
|       await page.waitForTimeout(100)
 | |
|       await page.keyboard.press('Tab')
 | |
| 
 | |
|       await page.keyboard.type(' // ')
 | |
|       // Since we need to parse the ast to know we are in a comment we gotta hang tight.
 | |
|       await page.waitForTimeout(700)
 | |
|       await page.keyboard.type('lin ')
 | |
|       await page.waitForTimeout(200)
 | |
|       // there shouldn't be any auto complete options for 'lin' in the comment
 | |
|       await expect(page.locator('.cm-completionLabel')).not.toBeVisible()
 | |
| 
 | |
|       await expect(page.locator('.cm-content'))
 | |
|         .toHaveText(`sketch001 = startSketchOn('XZ')
 | |
|         |> startProfileAt([3.14, 12], %)
 | |
|         |> xLine(5, %) // lin`)
 | |
|     })
 | |
|   })
 | |
|   test('Can undo a click and point extrude with ctrl+z', async ({
 | |
|     page,
 | |
|     context,
 | |
|     homePage,
 | |
|   }) => {
 | |
|     const u = await getUtils(page)
 | |
|     await context.addInitScript(async () => {
 | |
|       localStorage.setItem(
 | |
|         'persistCode',
 | |
|         `sketch001 = startSketchOn('XZ')
 | |
|   |> startProfileAt([4.61, -14.01], %)
 | |
|   |> line(end = [12.73, -0.09])
 | |
|   |> tangentialArcTo([24.95, -5.38], %)
 | |
|   |> close()`
 | |
|       )
 | |
|     })
 | |
| 
 | |
|     await page.setBodyDimensions({ width: 1200, height: 500 })
 | |
| 
 | |
|     await homePage.goToModelingScene()
 | |
|     await expect(
 | |
|       page.getByRole('button', { name: 'Start Sketch' })
 | |
|     ).not.toBeDisabled()
 | |
| 
 | |
|     await page.waitForTimeout(100)
 | |
|     await u.openAndClearDebugPanel()
 | |
|     await u.sendCustomCmd({
 | |
|       type: 'modeling_cmd_req',
 | |
|       cmd_id: uuidv4(),
 | |
|       cmd: {
 | |
|         type: 'default_camera_look_at',
 | |
|         vantage: { x: 0, y: -1250, z: 580 },
 | |
|         center: { x: 0, y: 0, z: 0 },
 | |
|         up: { x: 0, y: 0, z: 1 },
 | |
|       },
 | |
|     })
 | |
|     await page.waitForTimeout(100)
 | |
|     await u.sendCustomCmd({
 | |
|       type: 'modeling_cmd_req',
 | |
|       cmd_id: uuidv4(),
 | |
|       cmd: {
 | |
|         type: 'default_camera_get_settings',
 | |
|       },
 | |
|     })
 | |
|     await page.waitForTimeout(100)
 | |
| 
 | |
|     await page.getByText('startProfileAt([4.61, -14.01], %)').click()
 | |
|     await expect(page.getByRole('button', { name: 'Extrude' })).toBeVisible()
 | |
|     await page.getByRole('button', { name: 'Extrude' }).click()
 | |
| 
 | |
|     await expect(page.getByTestId('command-bar')).toBeVisible()
 | |
|     await page.waitForTimeout(100)
 | |
| 
 | |
|     await page.getByRole('button', { name: 'arrow right Continue' }).click()
 | |
|     await page.waitForTimeout(100)
 | |
|     await expect(page.getByText('Confirm Extrude')).toBeVisible()
 | |
|     await page.getByRole('button', { name: 'checkmark Submit command' }).click()
 | |
|     await page.waitForTimeout(100)
 | |
| 
 | |
|     // expect the code to have changed
 | |
|     await expect(page.locator('.cm-content')).toHaveText(
 | |
|       `sketch001 = startSketchOn('XZ')  |> startProfileAt([4.61, -14.01], %)  |> line(end = [12.73, -0.09])  |> tangentialArcTo([24.95, -5.38], %)  |> close()extrude001 = extrude(sketch001, length = 5)`
 | |
|     )
 | |
| 
 | |
|     // Now hit undo
 | |
|     await page.keyboard.down('Control')
 | |
|     await page.keyboard.press('KeyZ')
 | |
|     await page.keyboard.up('Control')
 | |
| 
 | |
|     await page.waitForTimeout(100)
 | |
|     await expect(page.locator('.cm-content'))
 | |
|       .toHaveText(`sketch001 = startSketchOn('XZ')
 | |
|   |> startProfileAt([4.61, -14.01], %)
 | |
|   |> line(end = [12.73, -0.09])
 | |
|   |> tangentialArcTo([24.95, -5.38], %)
 | |
|   |> close()`)
 | |
|   })
 | |
| 
 | |
|   test(
 | |
|     'Can undo a sketch modification with ctrl+z',
 | |
|     { tag: ['@skipWin'] },
 | |
|     async ({ page, homePage }) => {
 | |
|       const u = await getUtils(page)
 | |
|       await page.addInitScript(async () => {
 | |
|         localStorage.setItem(
 | |
|           'persistCode',
 | |
|           `sketch001 = startSketchOn('XZ')
 | |
|   |> startProfileAt([4.61, -10.01], %)
 | |
|   |> line(end = [12.73, -0.09])
 | |
|   |> tangentialArcTo([24.95, -0.38], %)
 | |
|   |> close()
 | |
|   |> extrude(length = 5)`
 | |
|         )
 | |
|       })
 | |
| 
 | |
|       await page.setBodyDimensions({ width: 1200, height: 500 })
 | |
| 
 | |
|       await homePage.goToModelingScene()
 | |
|       await expect(
 | |
|         page.getByRole('button', { name: 'Start Sketch' })
 | |
|       ).not.toBeDisabled()
 | |
| 
 | |
|       await page.waitForTimeout(100)
 | |
|       await u.openAndClearDebugPanel()
 | |
|       await u.sendCustomCmd({
 | |
|         type: 'modeling_cmd_req',
 | |
|         cmd_id: uuidv4(),
 | |
|         cmd: {
 | |
|           type: 'default_camera_look_at',
 | |
|           vantage: { x: 0, y: -1250, z: 580 },
 | |
|           center: { x: 0, y: 0, z: 0 },
 | |
|           up: { x: 0, y: 0, z: 1 },
 | |
|         },
 | |
|       })
 | |
|       await page.waitForTimeout(100)
 | |
|       await u.sendCustomCmd({
 | |
|         type: 'modeling_cmd_req',
 | |
|         cmd_id: uuidv4(),
 | |
|         cmd: {
 | |
|           type: 'default_camera_get_settings',
 | |
|         },
 | |
|       })
 | |
|       await page.waitForTimeout(100)
 | |
| 
 | |
|       const startPX = [1200 / 2, 500 / 2]
 | |
| 
 | |
|       const dragPX = 40
 | |
| 
 | |
|       await page.getByText('startProfileAt([4.61, -10.01], %)').click()
 | |
|       await expect(
 | |
|         page.getByRole('button', { name: 'Edit Sketch' })
 | |
|       ).toBeVisible()
 | |
|       await page.getByRole('button', { name: 'Edit Sketch' }).click()
 | |
|       await page.waitForTimeout(400)
 | |
|       let prevContent = await page.locator('.cm-content').innerText()
 | |
| 
 | |
|       await expect(page.getByTestId('segment-overlay')).toHaveCount(2)
 | |
| 
 | |
|       // drag startProfileAt handle
 | |
|       await page.dragAndDrop('#stream', '#stream', {
 | |
|         sourcePosition: { x: startPX[0] + 68, y: startPX[1] + 147 },
 | |
|         targetPosition: { x: startPX[0] + dragPX, y: startPX[1] + dragPX },
 | |
|       })
 | |
|       await page.waitForTimeout(100)
 | |
|       await expect(page.locator('.cm-content')).not.toHaveText(prevContent)
 | |
|       prevContent = await page.locator('.cm-content').innerText()
 | |
| 
 | |
|       // drag line handle
 | |
|       // we wait so it saves the code
 | |
|       await page.waitForTimeout(800)
 | |
| 
 | |
|       const lineEnd = await u.getBoundingBox('[data-overlay-index="0"]')
 | |
|       await page.waitForTimeout(100)
 | |
|       await page.dragAndDrop('#stream', '#stream', {
 | |
|         sourcePosition: { x: lineEnd.x - 5, y: lineEnd.y },
 | |
|         targetPosition: { x: lineEnd.x + dragPX, y: lineEnd.y + dragPX },
 | |
|       })
 | |
|       await expect(page.locator('.cm-content')).not.toHaveText(prevContent)
 | |
|       prevContent = await page.locator('.cm-content').innerText()
 | |
| 
 | |
|       // we wait so it saves the code
 | |
|       await page.waitForTimeout(800)
 | |
| 
 | |
|       // drag tangentialArcTo handle
 | |
|       const tangentEnd = await u.getBoundingBox('[data-overlay-index="1"]')
 | |
|       await page.dragAndDrop('#stream', '#stream', {
 | |
|         sourcePosition: { x: tangentEnd.x + 10, y: tangentEnd.y - 5 },
 | |
|         targetPosition: {
 | |
|           x: tangentEnd.x + dragPX,
 | |
|           y: tangentEnd.y + dragPX,
 | |
|         },
 | |
|       })
 | |
|       await page.waitForTimeout(100)
 | |
|       await expect(page.locator('.cm-content')).not.toHaveText(prevContent)
 | |
| 
 | |
|       // expect the code to have changed
 | |
|       await expect(page.locator('.cm-content'))
 | |
|         .toHaveText(`sketch001 = startSketchOn('XZ')
 | |
|     |> startProfileAt([2.71, -2.71], %)
 | |
|     |> line(end = [15.4, -2.78])
 | |
|     |> tangentialArcTo([27.6, -3.05], %)
 | |
|     |> close()
 | |
|     |> extrude(length = 5)
 | |
|   `)
 | |
| 
 | |
|       // Hit undo
 | |
|       await page.keyboard.down('Control')
 | |
|       await page.keyboard.press('KeyZ')
 | |
|       await page.keyboard.up('Control')
 | |
| 
 | |
|       await expect(page.locator('.cm-content'))
 | |
|         .toHaveText(`sketch001 = startSketchOn('XZ')
 | |
|     |> startProfileAt([2.71, -2.71], %)
 | |
|     |> line(end = [15.4, -2.78])
 | |
|     |> tangentialArcTo([24.95, -0.38], %)
 | |
|     |> close()
 | |
|     |> extrude(length = 5)`)
 | |
| 
 | |
|       // Hit undo again.
 | |
|       await page.keyboard.down('Control')
 | |
|       await page.keyboard.press('KeyZ')
 | |
|       await page.keyboard.up('Control')
 | |
| 
 | |
|       await expect(page.locator('.cm-content'))
 | |
|         .toHaveText(`sketch001 = startSketchOn('XZ')
 | |
|     |> startProfileAt([2.71, -2.71], %)
 | |
|     |> line(end = [12.73, -0.09])
 | |
|     |> tangentialArcTo([24.95, -0.38], %)
 | |
|     |> close()
 | |
|     |> extrude(length = 5)
 | |
|   `)
 | |
| 
 | |
|       // Hit undo again.
 | |
|       await page.keyboard.down('Control')
 | |
|       await page.keyboard.press('KeyZ')
 | |
|       await page.keyboard.up('Control')
 | |
| 
 | |
|       await page.waitForTimeout(100)
 | |
|       await expect(page.locator('.cm-content'))
 | |
|         .toHaveText(`sketch001 = startSketchOn('XZ')
 | |
|   |> startProfileAt([4.61, -10.01], %)
 | |
|   |> line(end = [12.73, -0.09])
 | |
|   |> tangentialArcTo([24.95, -0.38], %)
 | |
|   |> close()
 | |
|   |> extrude(length = 5)`)
 | |
|     }
 | |
|   )
 | |
| 
 | |
|   test.fixme(
 | |
|     `Can use the import stdlib function on a local OBJ file`,
 | |
|     { tag: '@electron' },
 | |
|     async ({ page, context }, testInfo) => {
 | |
|       await context.folderSetupFn(async (dir) => {
 | |
|         const bracketDir = join(dir, 'cube')
 | |
|         await fsp.mkdir(bracketDir, { recursive: true })
 | |
|         await fsp.copyFile(
 | |
|           executorInputPath('cube.obj'),
 | |
|           join(bracketDir, 'cube.obj')
 | |
|         )
 | |
|         await fsp.writeFile(join(bracketDir, 'main.kcl'), '')
 | |
|       })
 | |
| 
 | |
|       const viewportSize = { width: 1200, height: 500 }
 | |
|       await page.setBodyDimensions(viewportSize)
 | |
| 
 | |
|       // Locators and constants
 | |
|       const u = await getUtils(page)
 | |
|       const projectLink = page.getByRole('link', { name: 'cube' })
 | |
|       const gizmo = page.locator('[aria-label*=gizmo]')
 | |
|       const resetCameraButton = page.getByRole('button', { name: 'Reset view' })
 | |
|       const locationToHavColor = async (
 | |
|         position: { x: number; y: number },
 | |
|         color: [number, number, number]
 | |
|       ) => {
 | |
|         return u.getGreatestPixDiff(position, color)
 | |
|       }
 | |
|       const notTheOrigin = {
 | |
|         x: viewportSize.width * 0.55,
 | |
|         y: viewportSize.height * 0.3,
 | |
|       }
 | |
|       const origin = { x: viewportSize.width / 2, y: viewportSize.height / 2 }
 | |
|       const errorIndicators = page.locator('.cm-lint-marker-error')
 | |
| 
 | |
|       await test.step(`Open the empty file, see the default planes`, async () => {
 | |
|         await projectLink.click()
 | |
|         await u.waitForPageLoad()
 | |
|         await expect
 | |
|           .poll(
 | |
|             async () => locationToHavColor(notTheOrigin, darkModePlaneColorXZ),
 | |
|             {
 | |
|               timeout: 5000,
 | |
|               message: 'XZ plane color is visible',
 | |
|             }
 | |
|           )
 | |
|           .toBeLessThan(15)
 | |
|       })
 | |
|       await test.step(`Write the import function line`, async () => {
 | |
|         await u.codeLocator.fill(`import('cube.obj')`)
 | |
|         await page.waitForTimeout(800)
 | |
|       })
 | |
|       await test.step(`Reset the camera before checking`, async () => {
 | |
|         await u.doAndWaitForCmd(async () => {
 | |
|           await gizmo.click({ button: 'right' })
 | |
|           await resetCameraButton.click()
 | |
|         }, 'zoom_to_fit')
 | |
|       })
 | |
|       await test.step(`Verify that we see the imported geometry and no errors`, async () => {
 | |
|         await expect(errorIndicators).toHaveCount(0)
 | |
|         await expect
 | |
|           .poll(async () => locationToHavColor(origin, darkModePlaneColorXZ), {
 | |
|             timeout: 3000,
 | |
|             message: 'Plane color should not be visible',
 | |
|           })
 | |
|           .toBeGreaterThan(15)
 | |
|         await expect
 | |
|           .poll(async () => locationToHavColor(origin, darkModeBgColor), {
 | |
|             timeout: 3000,
 | |
|             message: 'Background color should not be visible',
 | |
|           })
 | |
|           .toBeGreaterThan(15)
 | |
|       })
 | |
|     }
 | |
|   )
 | |
| })
 |