2024-08-07 19:27:32 +10:00
|
|
|
import { test, expect } from '@playwright/test'
|
2024-10-18 10:43:01 -04:00
|
|
|
import fsp from 'fs/promises'
|
2024-08-07 19:27:32 +10:00
|
|
|
import { uuidv4 } from 'lib/utils'
|
2024-10-18 10:43:01 -04:00
|
|
|
import {
|
|
|
|
darkModeBgColor,
|
|
|
|
darkModePlaneColorXZ,
|
|
|
|
executorInputPath,
|
|
|
|
getUtils,
|
|
|
|
setup,
|
|
|
|
setupElectron,
|
|
|
|
tearDown,
|
|
|
|
} from './test-utils'
|
|
|
|
import { join } from 'path'
|
2024-08-07 19:27:32 +10:00
|
|
|
|
2024-09-16 07:32:33 +10:00
|
|
|
test.beforeEach(async ({ context, page }, testInfo) => {
|
|
|
|
await setup(context, page, testInfo)
|
2024-08-07 19:27:32 +10:00
|
|
|
})
|
|
|
|
|
|
|
|
test.afterEach(async ({ page }, testInfo) => {
|
|
|
|
await tearDown(page, testInfo)
|
|
|
|
})
|
|
|
|
|
|
|
|
test.describe('Editor tests', () => {
|
|
|
|
test('can comment out code with ctrl+/', async ({ page }) => {
|
|
|
|
const u = await getUtils(page)
|
|
|
|
await page.setViewportSize({ width: 1000, height: 500 })
|
|
|
|
|
|
|
|
await u.waitForAuthSkipAppStart()
|
|
|
|
|
|
|
|
// check no error to begin with
|
|
|
|
await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible()
|
|
|
|
|
|
|
|
await u.codeLocator.click()
|
2024-10-02 14:19:40 -05:00
|
|
|
await page.keyboard.type(`sketch001 = startSketchOn('XY')
|
2024-08-07 19:27:32 +10:00
|
|
|
|> startProfileAt([-10, -10], %)
|
|
|
|
|> line([20, 0], %)
|
|
|
|
|> line([0, 20], %)
|
|
|
|
|> line([-20, 0], %)
|
|
|
|
|> close(%)`)
|
|
|
|
|
2024-08-22 19:13:27 -04:00
|
|
|
await page.keyboard.down('ControlOrMeta')
|
2024-08-07 19:27:32 +10:00
|
|
|
await page.keyboard.press('/')
|
2024-08-22 19:13:27 -04:00
|
|
|
await page.keyboard.up('ControlOrMeta')
|
2024-08-07 19:27:32 +10:00
|
|
|
|
|
|
|
await expect(page.locator('.cm-content'))
|
2024-10-02 14:19:40 -05:00
|
|
|
.toHaveText(`sketch001 = startSketchOn('XY')
|
2024-08-07 19:27:32 +10:00
|
|
|
|> startProfileAt([-10, -10], %)
|
|
|
|
|> line([20, 0], %)
|
|
|
|
|> line([0, 20], %)
|
|
|
|
|> line([-20, 0], %)
|
|
|
|
// |> close(%)`)
|
|
|
|
|
|
|
|
// uncomment the code
|
2024-08-22 19:13:27 -04:00
|
|
|
await page.keyboard.down('ControlOrMeta')
|
2024-08-07 19:27:32 +10:00
|
|
|
await page.keyboard.press('/')
|
2024-08-22 19:13:27 -04:00
|
|
|
await page.keyboard.up('ControlOrMeta')
|
2024-08-07 19:27:32 +10:00
|
|
|
|
|
|
|
await expect(page.locator('.cm-content'))
|
2024-10-02 14:19:40 -05:00
|
|
|
.toHaveText(`sketch001 = startSketchOn('XY')
|
2024-08-07 19:27:32 +10:00
|
|
|
|> startProfileAt([-10, -10], %)
|
|
|
|
|> line([20, 0], %)
|
|
|
|
|> line([0, 20], %)
|
|
|
|
|> line([-20, 0], %)
|
|
|
|
|> close(%)`)
|
|
|
|
})
|
|
|
|
|
|
|
|
test('if you click the format button it formats your code', async ({
|
|
|
|
page,
|
|
|
|
}) => {
|
|
|
|
const u = await getUtils(page)
|
|
|
|
await page.setViewportSize({ width: 1000, height: 500 })
|
|
|
|
|
|
|
|
await u.waitForAuthSkipAppStart()
|
|
|
|
|
|
|
|
// check no error to begin with
|
|
|
|
await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible()
|
|
|
|
|
|
|
|
await u.codeLocator.click()
|
2024-10-02 14:19:40 -05:00
|
|
|
await page.keyboard.type(`sketch001 = startSketchOn('XY')
|
2024-08-07 19:27:32 +10:00
|
|
|
|> startProfileAt([-10, -10], %)
|
|
|
|
|> line([20, 0], %)
|
|
|
|
|> line([0, 20], %)
|
|
|
|
|> line([-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'))
|
2024-10-02 14:19:40 -05:00
|
|
|
.toHaveText(`sketch001 = startSketchOn('XY')
|
2024-08-07 19:27:32 +10:00
|
|
|
|> startProfileAt([-10, -10], %)
|
|
|
|
|> line([20, 0], %)
|
|
|
|
|> line([0, 20], %)
|
|
|
|
|> line([-20, 0], %)
|
|
|
|
|> close(%)`)
|
|
|
|
})
|
|
|
|
|
2024-12-12 13:37:23 -08:00
|
|
|
test('ensure we use the cache, and do not re-execute', async ({ page }) => {
|
|
|
|
const u = await getUtils(page)
|
|
|
|
await page.setViewportSize({ width: 1000, height: 500 })
|
|
|
|
|
|
|
|
await u.waitForAuthSkipAppStart()
|
|
|
|
|
|
|
|
await u.codeLocator.click()
|
|
|
|
await page.keyboard.type(`sketch001 = startSketchOn('XY')
|
|
|
|
|> startProfileAt([-10, -10], %)
|
|
|
|
|> line([20, 0], %)
|
|
|
|
|> line([0, 20], %)
|
|
|
|
|> line([-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(1)
|
|
|
|
|
|
|
|
// 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(2)
|
|
|
|
await expect(
|
|
|
|
page.locator('[data-receive-command-type="scene_clear_all"]')
|
|
|
|
).toHaveCount(1)
|
|
|
|
})
|
|
|
|
|
2024-08-25 15:14:38 -07:00
|
|
|
test('if you click the format button it formats your code and executes so lints are still there', async ({
|
|
|
|
page,
|
|
|
|
}) => {
|
|
|
|
const u = await getUtils(page)
|
|
|
|
await page.setViewportSize({ width: 1000, height: 500 })
|
|
|
|
|
|
|
|
await u.waitForAuthSkipAppStart()
|
|
|
|
|
|
|
|
// check no error to begin with
|
|
|
|
await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible()
|
|
|
|
|
|
|
|
await u.codeLocator.click()
|
2024-10-02 14:19:40 -05:00
|
|
|
await page.keyboard.type(`sketch_001 = startSketchOn('XY')
|
2024-08-25 15:14:38 -07:00
|
|
|
|> startProfileAt([-10, -10], %)
|
|
|
|
|> line([20, 0], %)
|
|
|
|
|> line([0, 20], %)
|
|
|
|
|> line([-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'))
|
2024-10-02 14:19:40 -05:00
|
|
|
.toHaveText(`sketch_001 = startSketchOn('XY')
|
2024-08-25 15:14:38 -07:00
|
|
|
|> startProfileAt([-10, -10], %)
|
|
|
|
|> line([20, 0], %)
|
|
|
|
|> line([0, 20], %)
|
|
|
|
|> line([-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()
|
|
|
|
})
|
|
|
|
|
2024-08-07 19:27:32 +10:00
|
|
|
test('fold gutters work', async ({ page }) => {
|
|
|
|
const u = await getUtils(page)
|
|
|
|
|
2024-10-02 14:19:40 -05:00
|
|
|
const fullCode = `sketch001 = startSketchOn('XY')
|
2024-08-07 19:27:32 +10:00
|
|
|
|> startProfileAt([-10, -10], %)
|
|
|
|
|> line([20, 0], %)
|
|
|
|
|> line([0, 20], %)
|
|
|
|
|> line([-20, 0], %)
|
|
|
|
|> close(%)`
|
|
|
|
await page.addInitScript(async () => {
|
|
|
|
localStorage.setItem(
|
|
|
|
'persistCode',
|
2024-10-02 14:19:40 -05:00
|
|
|
`sketch001 = startSketchOn('XY')
|
2024-08-07 19:27:32 +10:00
|
|
|
|> startProfileAt([-10, -10], %)
|
|
|
|
|> line([20, 0], %)
|
|
|
|
|> line([0, 20], %)
|
|
|
|
|> line([-20, 0], %)
|
|
|
|
|> close(%)`
|
|
|
|
)
|
|
|
|
})
|
|
|
|
await page.setViewportSize({ width: 1000, height: 500 })
|
|
|
|
|
|
|
|
await u.waitForAuthSkipAppStart()
|
|
|
|
|
|
|
|
// 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(
|
2024-10-02 14:19:40 -05:00
|
|
|
`sketch001 = startSketchOn('XY')… `
|
2024-08-07 19:27:32 +10:00
|
|
|
)
|
|
|
|
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
|
2024-08-22 19:13:27 -04:00
|
|
|
await page.keyboard.press('ControlOrMeta+A')
|
2024-08-07 19:27:32 +10:00
|
|
|
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 }) => {
|
|
|
|
const u = await getUtils(page)
|
|
|
|
await page.addInitScript(async () => {
|
|
|
|
localStorage.setItem(
|
|
|
|
'persistCode',
|
2024-10-02 14:19:40 -05:00
|
|
|
`sketch001 = startSketchOn('XY')
|
2024-08-07 19:27:32 +10:00
|
|
|
|> startProfileAt([-10, -10], %)
|
|
|
|
|> line([20, 0], %)
|
|
|
|
|> line([0, 20], %)
|
|
|
|
|> line([-20, 0], %)
|
|
|
|
|> close(%)`
|
|
|
|
)
|
|
|
|
})
|
|
|
|
await page.setViewportSize({ width: 1000, height: 500 })
|
|
|
|
|
|
|
|
await u.waitForAuthSkipAppStart()
|
|
|
|
|
|
|
|
// 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('Draw a line')).toBeVisible()
|
|
|
|
})
|
|
|
|
|
|
|
|
test('if you use the format keyboard binding it formats your code', async ({
|
|
|
|
page,
|
|
|
|
}) => {
|
|
|
|
const u = await getUtils(page)
|
|
|
|
await page.addInitScript(async () => {
|
|
|
|
localStorage.setItem(
|
|
|
|
'persistCode',
|
2024-10-02 14:19:40 -05:00
|
|
|
`sketch001 = startSketchOn('XY')
|
2024-08-07 19:27:32 +10:00
|
|
|
|> startProfileAt([-10, -10], %)
|
|
|
|
|> line([20, 0], %)
|
|
|
|
|> line([0, 20], %)
|
|
|
|
|> line([-20, 0], %)
|
|
|
|
|> close(%)`
|
|
|
|
)
|
|
|
|
localStorage.setItem('disableAxis', 'true')
|
|
|
|
})
|
|
|
|
await page.setViewportSize({ width: 1000, height: 500 })
|
|
|
|
|
|
|
|
await u.waitForAuthSkipAppStart()
|
|
|
|
|
|
|
|
// 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'))
|
2024-10-02 14:19:40 -05:00
|
|
|
.toHaveText(`sketch001 = startSketchOn('XY')
|
2024-08-07 19:27:32 +10:00
|
|
|
|> startProfileAt([-10, -10], %)
|
|
|
|
|> line([20, 0], %)
|
|
|
|
|> line([0, 20], %)
|
|
|
|
|> line([-20, 0], %)
|
|
|
|
|> close(%)`)
|
|
|
|
})
|
|
|
|
|
2024-08-25 15:14:38 -07:00
|
|
|
test('if you use the format keyboard binding it formats your code and executes so lints are shown', async ({
|
|
|
|
page,
|
|
|
|
}) => {
|
|
|
|
const u = await getUtils(page)
|
|
|
|
await page.addInitScript(async () => {
|
|
|
|
localStorage.setItem(
|
|
|
|
'persistCode',
|
2024-10-02 14:19:40 -05:00
|
|
|
`sketch_001 = startSketchOn('XY')
|
2024-08-25 15:14:38 -07:00
|
|
|
|> startProfileAt([-10, -10], %)
|
|
|
|
|> line([20, 0], %)
|
|
|
|
|> line([0, 20], %)
|
|
|
|
|> line([-20, 0], %)
|
|
|
|
|> close(%)`
|
|
|
|
)
|
|
|
|
localStorage.setItem('disableAxis', 'true')
|
|
|
|
})
|
|
|
|
await page.setViewportSize({ width: 1000, height: 500 })
|
|
|
|
|
|
|
|
await u.waitForAuthSkipAppStart()
|
|
|
|
|
|
|
|
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'))
|
2024-10-02 14:19:40 -05:00
|
|
|
.toHaveText(`sketch_001 = startSketchOn('XY')
|
2024-08-25 15:14:38 -07:00
|
|
|
|> startProfileAt([-10, -10], %)
|
|
|
|
|> line([20, 0], %)
|
|
|
|
|> line([0, 20], %)
|
|
|
|
|> line([-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()
|
|
|
|
})
|
|
|
|
|
2024-08-07 19:27:32 +10:00
|
|
|
test('if you write kcl with lint errors you get lints', async ({ page }) => {
|
|
|
|
const u = await getUtils(page)
|
|
|
|
await page.setViewportSize({ width: 1000, height: 500 })
|
|
|
|
|
|
|
|
await u.waitForAuthSkipAppStart()
|
|
|
|
|
|
|
|
// check no error to begin with
|
|
|
|
await expect(page.locator('.cm-lint-marker-info')).not.toBeVisible()
|
|
|
|
|
|
|
|
await u.codeLocator.click()
|
2024-10-02 14:19:40 -05:00
|
|
|
await page.keyboard.type('my_snake_case_var = 5')
|
2024-08-07 19:27:32 +10:00
|
|
|
await page.keyboard.press('Enter')
|
2024-10-02 14:19:40 -05:00
|
|
|
await page.keyboard.type('myCamelCaseVar = 5')
|
2024-08-07 19:27:32 +10:00
|
|
|
await page.keyboard.press('Enter')
|
|
|
|
|
|
|
|
// press arrows to clear autocomplete
|
|
|
|
await page.keyboard.press('ArrowLeft')
|
|
|
|
await page.keyboard.press('ArrowRight')
|
|
|
|
|
|
|
|
// 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
|
2024-10-02 14:19:40 -05:00
|
|
|
await page.getByText('my_snake_case_var = 5').click()
|
2024-08-07 19:27:32 +10:00
|
|
|
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 }) => {
|
|
|
|
const u = await getUtils(page)
|
|
|
|
await page.addInitScript(async () => {
|
|
|
|
localStorage.setItem(
|
|
|
|
'persistCode',
|
2024-10-02 14:19:40 -05:00
|
|
|
`sketch001 = startSketchOn('XZ')
|
2024-08-07 19:27:32 +10:00
|
|
|
|> startProfileAt([3.29, 7.86], %)
|
|
|
|
|> line([2.48, 2.44], %)
|
|
|
|
|> line([2.66, 1.17], %)
|
|
|
|
|> close(%)
|
|
|
|
`
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
await page.setViewportSize({ width: 1000, height: 500 })
|
|
|
|
|
|
|
|
await u.waitForAuthSkipAppStart()
|
|
|
|
|
|
|
|
// check no error to begin with
|
|
|
|
await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible()
|
|
|
|
|
|
|
|
await u.codeLocator.click()
|
|
|
|
|
|
|
|
await page.getByText(' |> line([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 }) => {
|
|
|
|
const u = await getUtils(page)
|
|
|
|
await page.setViewportSize({ width: 1000, height: 500 })
|
|
|
|
|
|
|
|
await u.waitForAuthSkipAppStart()
|
|
|
|
|
|
|
|
// 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)
|
|
|
|
$ error
|
2024-12-06 13:12:20 -06:00
|
|
|
topAng = 30
|
|
|
|
bottomAng = 25
|
2024-08-07 19:27:32 +10:00
|
|
|
*/
|
|
|
|
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')
|
2024-10-02 14:19:40 -05:00
|
|
|
await page.keyboard.type('topAng = 30')
|
2024-08-07 19:27:32 +10:00
|
|
|
await page.keyboard.press('Enter')
|
2024-10-02 14:19:40 -05:00
|
|
|
await page.keyboard.type('bottomAng = 25')
|
2024-08-07 19:27:32 +10:00
|
|
|
await page.keyboard.press('Enter')
|
|
|
|
|
2024-12-06 13:12:20 -06:00
|
|
|
// error in gutter
|
2024-08-07 19:27:32 +10:00
|
|
|
await expect(page.locator('.cm-lint-marker-error')).toBeVisible()
|
|
|
|
|
|
|
|
// error text on hover
|
|
|
|
await page.hover('.cm-lint-marker-error')
|
2024-12-06 13:12:20 -06:00
|
|
|
await expect(
|
|
|
|
page.getByText('Tag names must not be empty').first()
|
|
|
|
).toBeVisible()
|
2024-08-07 19:27:32 +10:00
|
|
|
|
|
|
|
// 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
|
2024-10-02 14:19:40 -05:00
|
|
|
await page.getByText('bottomAng = 25').click()
|
2024-08-07 19:27:32 +10:00
|
|
|
await page.keyboard.press('Enter')
|
|
|
|
await page.keyboard.type("// Let's define the same thing twice")
|
|
|
|
await page.keyboard.press('Enter')
|
2024-10-02 14:19:40 -05:00
|
|
|
await page.keyboard.type('topAng = 42')
|
2024-08-07 19:27:32 +10:00
|
|
|
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()
|
|
|
|
})
|
|
|
|
|
2024-12-06 13:57:31 +13:00
|
|
|
// TODO currently multiple source ranges are not supported
|
|
|
|
test.skip('error with 2 source ranges gets 2 diagnostics', async ({
|
|
|
|
page,
|
|
|
|
}) => {
|
2024-08-07 19:27:32 +10:00
|
|
|
const u = await getUtils(page)
|
|
|
|
await page.addInitScript(async () => {
|
|
|
|
localStorage.setItem(
|
|
|
|
'persistCode',
|
2024-10-02 14:19:40 -05:00
|
|
|
`length = .750
|
|
|
|
width = 0.500
|
|
|
|
height = 0.500
|
|
|
|
dia = 4
|
2024-08-25 15:14:38 -07:00
|
|
|
|
2024-08-07 19:27:32 +10:00
|
|
|
fn squareHole = (l, w) => {
|
2024-10-02 14:19:40 -05:00
|
|
|
squareHoleSketch = startSketchOn('XY')
|
2024-08-07 19:27:32 +10:00
|
|
|
|> startProfileAt([-width / 2, -length / 2], %)
|
|
|
|
|> lineTo([width / 2, -length / 2], %)
|
|
|
|
|> lineTo([width / 2, length / 2], %)
|
|
|
|
|> lineTo([-width / 2, length / 2], %)
|
|
|
|
|> close(%)
|
|
|
|
return squareHoleSketch
|
|
|
|
}
|
|
|
|
`
|
|
|
|
)
|
|
|
|
})
|
|
|
|
await page.setViewportSize({ width: 1000, height: 500 })
|
|
|
|
|
|
|
|
await u.waitForAuthSkipAppStart()
|
|
|
|
|
|
|
|
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')
|
2024-10-02 14:19:40 -05:00
|
|
|
await page.keyboard.type(`extrusion = startSketchOn('XY')
|
2024-11-25 09:21:55 +13:00
|
|
|
|> circle({ center = [0, 0], radius = dia/2 }, %)
|
2024-08-07 19:27:32 +10:00
|
|
|
|> hole(squareHole(length, width, height), %)
|
|
|
|
|> extrude(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 ({
|
|
|
|
page,
|
|
|
|
}) => {
|
|
|
|
const u = await getUtils(page)
|
|
|
|
await page.addInitScript(async () => {
|
|
|
|
localStorage.setItem(
|
|
|
|
'persistCode',
|
2024-10-02 14:19:40 -05:00
|
|
|
`box = startSketchOn('XY')
|
2024-08-07 19:27:32 +10:00
|
|
|
|> startProfileAt([0, 0], %)
|
|
|
|
|> line([0, 10], %)
|
|
|
|
|> line([10, 0], %)
|
|
|
|
|> line([0, -10], %, $revolveAxis)
|
|
|
|
|> close(%)
|
|
|
|
|> extrude(10, %)
|
|
|
|
|
2024-10-02 14:19:40 -05:00
|
|
|
sketch001 = startSketchOn(box, revolveAxis)
|
2024-08-07 19:27:32 +10:00
|
|
|
|> startProfileAt([5, 10], %)
|
|
|
|
|> line([0, -10], %)
|
|
|
|
|> line([2, 0], %)
|
|
|
|
|> line([0, -10], %)
|
|
|
|
|> close(%)
|
|
|
|
|> revolve({
|
2024-11-25 09:21:55 +13:00
|
|
|
axis = revolveAxis,
|
|
|
|
angle = 90
|
2024-08-07 19:27:32 +10:00
|
|
|
}, %)
|
|
|
|
`
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
await page.setViewportSize({ width: 1000, height: 500 })
|
|
|
|
|
|
|
|
await page.goto('/')
|
|
|
|
await u.waitForPageLoad()
|
|
|
|
|
|
|
|
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 }) => {
|
|
|
|
const u = await getUtils(page)
|
|
|
|
// const PUR = 400 / 37.5 //pixeltoUnitRatio
|
|
|
|
await page.setViewportSize({ width: 1200, height: 500 })
|
|
|
|
|
|
|
|
await u.waitForAuthSkipAppStart()
|
|
|
|
|
|
|
|
// tests clicking on an option, selection the first option
|
|
|
|
// and arrowing down to an option
|
|
|
|
|
|
|
|
await u.codeLocator.click()
|
2024-10-02 14:19:40 -05:00
|
|
|
await page.keyboard.type('sketch001 = start')
|
2024-08-07 19:27:32 +10:00
|
|
|
|
2024-11-05 14:10:35 -06:00
|
|
|
// 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()
|
2024-08-07 19:27:32 +10:00
|
|
|
// 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 twice then enter to accept xLine
|
|
|
|
await page.keyboard.press('ArrowDown')
|
|
|
|
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'))
|
2024-10-02 14:19:40 -05:00
|
|
|
.toHaveText(`sketch001 = startSketchOn('XZ')
|
2024-08-07 19:27:32 +10:00
|
|
|
|> startProfileAt([3.14, 12], %)
|
|
|
|
|> xLine(5, %) // lin`)
|
2024-11-12 20:01:47 -05:00
|
|
|
|
|
|
|
// expect there to be no KCL errors
|
|
|
|
await expect(page.locator('.cm-lint-marker-error')).toHaveCount(0)
|
2024-08-07 19:27:32 +10:00
|
|
|
})
|
|
|
|
|
|
|
|
test('with tab to accept the completion', async ({ page }) => {
|
|
|
|
const u = await getUtils(page)
|
|
|
|
// const PUR = 400 / 37.5 //pixeltoUnitRatio
|
|
|
|
await page.setViewportSize({ width: 1200, height: 500 })
|
|
|
|
|
|
|
|
await u.waitForAuthSkipAppStart()
|
|
|
|
|
|
|
|
// 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()
|
2024-10-02 14:19:40 -05:00
|
|
|
await page.keyboard.type('sketch001 = startSketchO')
|
2024-08-07 19:27:32 +10:00
|
|
|
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 twice then tab to accept xLine
|
|
|
|
await page.keyboard.press('ArrowDown')
|
|
|
|
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'))
|
2024-10-02 14:19:40 -05:00
|
|
|
.toHaveText(`sketch001 = startSketchOn('XZ')
|
2024-08-07 19:27:32 +10:00
|
|
|
|> startProfileAt([3.14, 12], %)
|
|
|
|
|> xLine(5, %) // lin`)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
test('Can undo a click and point extrude with ctrl+z', async ({ page }) => {
|
|
|
|
const u = await getUtils(page)
|
|
|
|
await page.addInitScript(async () => {
|
|
|
|
localStorage.setItem(
|
|
|
|
'persistCode',
|
2024-10-02 14:19:40 -05:00
|
|
|
`sketch001 = startSketchOn('XZ')
|
2024-08-07 19:27:32 +10:00
|
|
|
|> startProfileAt([4.61, -14.01], %)
|
|
|
|
|> line([12.73, -0.09], %)
|
|
|
|
|> tangentialArcTo([24.95, -5.38], %)
|
|
|
|
|> close(%)`
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
await page.setViewportSize({ width: 1200, height: 500 })
|
|
|
|
|
|
|
|
await u.waitForAuthSkipAppStart()
|
|
|
|
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)
|
|
|
|
|
2024-08-16 07:15:42 -04:00
|
|
|
await page.getByRole('button', { name: 'arrow right Continue' }).click()
|
2024-08-07 19:27:32 +10:00
|
|
|
await page.waitForTimeout(100)
|
|
|
|
await expect(page.getByText('Confirm Extrude')).toBeVisible()
|
2024-08-16 07:15:42 -04:00
|
|
|
await page.getByRole('button', { name: 'checkmark Submit command' }).click()
|
2024-08-07 19:27:32 +10:00
|
|
|
await page.waitForTimeout(100)
|
|
|
|
|
|
|
|
// expect the code to have changed
|
|
|
|
await expect(page.locator('.cm-content')).toHaveText(
|
2024-10-02 14:19:40 -05:00
|
|
|
`sketch001 = startSketchOn('XZ') |> startProfileAt([4.61, -14.01], %) |> line([12.73, -0.09], %) |> tangentialArcTo([24.95, -5.38], %) |> close(%)extrude001 = extrude(5, sketch001)`
|
2024-08-07 19:27:32 +10:00
|
|
|
)
|
|
|
|
|
|
|
|
// 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'))
|
2024-10-02 14:19:40 -05:00
|
|
|
.toHaveText(`sketch001 = startSketchOn('XZ')
|
2024-08-07 19:27:32 +10:00
|
|
|
|> startProfileAt([4.61, -14.01], %)
|
|
|
|
|> line([12.73, -0.09], %)
|
|
|
|
|> tangentialArcTo([24.95, -5.38], %)
|
|
|
|
|> close(%)`)
|
|
|
|
})
|
|
|
|
|
2024-08-20 14:13:56 +10:00
|
|
|
test('Can undo a sketch modification with ctrl+z', async ({ page }) => {
|
2024-08-07 19:27:32 +10:00
|
|
|
const u = await getUtils(page)
|
|
|
|
await page.addInitScript(async () => {
|
|
|
|
localStorage.setItem(
|
|
|
|
'persistCode',
|
2024-10-02 14:19:40 -05:00
|
|
|
`sketch001 = startSketchOn('XZ')
|
2024-08-20 14:13:56 +10:00
|
|
|
|> startProfileAt([4.61, -10.01], %)
|
2024-08-07 19:27:32 +10:00
|
|
|
|> line([12.73, -0.09], %)
|
2024-08-20 14:13:56 +10:00
|
|
|
|> tangentialArcTo([24.95, -0.38], %)
|
2024-08-07 19:27:32 +10:00
|
|
|
|> close(%)
|
|
|
|
|> extrude(5, %)`
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
await page.setViewportSize({ width: 1200, height: 500 })
|
|
|
|
|
|
|
|
await u.waitForAuthSkipAppStart()
|
|
|
|
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)
|
|
|
|
|
2024-08-20 14:13:56 +10:00
|
|
|
const startPX = [665, 397]
|
2024-08-07 19:27:32 +10:00
|
|
|
|
|
|
|
const dragPX = 40
|
|
|
|
|
2024-08-20 14:13:56 +10:00
|
|
|
await page.getByText('startProfileAt([4.61, -10.01], %)').click()
|
2024-08-07 19:27:32 +10:00
|
|
|
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 startProfieAt handle
|
|
|
|
await page.dragAndDrop('#stream', '#stream', {
|
|
|
|
sourcePosition: { x: startPX[0], y: startPX[1] },
|
|
|
|
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', {
|
2024-08-20 14:13:56 +10:00
|
|
|
sourcePosition: { x: tangentEnd.x + 10, y: tangentEnd.y - 5 },
|
2024-08-07 19:27:32 +10:00
|
|
|
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'))
|
2024-10-02 14:19:40 -05:00
|
|
|
.toHaveText(`sketch001 = startSketchOn('XZ')
|
2024-08-20 14:13:56 +10:00
|
|
|
|> startProfileAt([7.12, -12.68], %)
|
|
|
|
|> line([15.39, -2.78], %)
|
|
|
|
|> tangentialArcTo([27.6, -3.05], %)
|
|
|
|
|> close(%)
|
|
|
|
|> extrude(5, %)
|
|
|
|
`)
|
2024-08-07 19:27:32 +10:00
|
|
|
|
|
|
|
// Hit undo
|
|
|
|
await page.keyboard.down('Control')
|
|
|
|
await page.keyboard.press('KeyZ')
|
|
|
|
await page.keyboard.up('Control')
|
|
|
|
|
|
|
|
await expect(page.locator('.cm-content'))
|
2024-10-02 14:19:40 -05:00
|
|
|
.toHaveText(`sketch001 = startSketchOn('XZ')
|
2024-08-20 14:13:56 +10:00
|
|
|
|> startProfileAt([7.12, -12.68], %)
|
|
|
|
|> line([15.39, -2.78], %)
|
|
|
|
|> tangentialArcTo([24.95, -0.38], %)
|
|
|
|
|> close(%)
|
|
|
|
|> extrude(5, %)`)
|
2024-08-07 19:27:32 +10:00
|
|
|
|
|
|
|
// Hit undo again.
|
|
|
|
await page.keyboard.down('Control')
|
|
|
|
await page.keyboard.press('KeyZ')
|
|
|
|
await page.keyboard.up('Control')
|
|
|
|
|
|
|
|
await expect(page.locator('.cm-content'))
|
2024-10-02 14:19:40 -05:00
|
|
|
.toHaveText(`sketch001 = startSketchOn('XZ')
|
2024-08-20 14:13:56 +10:00
|
|
|
|> startProfileAt([7.12, -12.68], %)
|
|
|
|
|> line([12.73, -0.09], %)
|
|
|
|
|> tangentialArcTo([24.95, -0.38], %)
|
|
|
|
|> close(%)
|
|
|
|
|> extrude(5, %)
|
|
|
|
`)
|
2024-08-07 19:27:32 +10:00
|
|
|
|
|
|
|
// 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'))
|
2024-10-02 14:19:40 -05:00
|
|
|
.toHaveText(`sketch001 = startSketchOn('XZ')
|
2024-08-20 14:13:56 +10:00
|
|
|
|> startProfileAt([4.61, -10.01], %)
|
2024-08-07 19:27:32 +10:00
|
|
|
|> line([12.73, -0.09], %)
|
2024-08-20 14:13:56 +10:00
|
|
|
|> tangentialArcTo([24.95, -0.38], %)
|
2024-08-07 19:27:32 +10:00
|
|
|
|> close(%)
|
|
|
|
|> extrude(5, %)`)
|
|
|
|
})
|
2024-10-18 10:43:01 -04:00
|
|
|
|
2024-11-04 20:34:22 -06:00
|
|
|
test.fixme(
|
2024-10-18 10:43:01 -04:00
|
|
|
`Can use the import stdlib function on a local OBJ file`,
|
|
|
|
{ tag: '@electron' },
|
|
|
|
async ({ browserName }, testInfo) => {
|
|
|
|
const { electronApp, page } = await setupElectron({
|
|
|
|
testInfo,
|
|
|
|
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.setViewportSize(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)
|
|
|
|
})
|
|
|
|
|
|
|
|
await electronApp.close()
|
|
|
|
}
|
|
|
|
)
|
2024-08-07 19:27:32 +10:00
|
|
|
})
|