import { doExport, getUtils, makeTemplate } from '@e2e/playwright/test-utils' import { expect, test } from '@e2e/playwright/zoo-test' test('Units menu', async ({ page, homePage }) => { await page.setBodyDimensions({ width: 1200, height: 500 }) await homePage.goToModelingScene() const unitsMenuButton = page.getByRole('button', { name: 'Current Units', exact: false, }) await expect(unitsMenuButton).toBeVisible() await expect(unitsMenuButton).toContainText('in') await unitsMenuButton.click() const millimetersButton = page.getByRole('button', { name: 'Millimeters' }) await expect(millimetersButton).toBeVisible() await millimetersButton.click() // Look out for the toast message const toastMessage = page.getByText('Updated per-file units to mm') await expect(toastMessage).toBeVisible() // Verify that the popover has closed await expect(millimetersButton).not.toBeAttached() // Verify that the button label has updated await expect(unitsMenuButton).toContainText('mm') }) test( 'Successful export shows a success toast', { tag: '@skipLocalEngine' }, async ({ page, homePage, tronApp }) => { // FYI this test doesn't work with only engine running locally // And you will need to have the KittyCAD CLI installed const u = await getUtils(page) await page.addInitScript(async () => { ;(window as any).playwrightSkipFilePicker = true localStorage.setItem( 'persistCode', `topAng = 25 bottomAng = 35 baseLen = 3.5 baseHeight = 1 totalHeightHalf = 2 armThick = 0.5 totalLen = 9.5 part001 = startSketchOn(-XZ) |> startProfile(at = [0, 0]) |> yLine(length = baseHeight) |> xLine(length = baseLen) |> angledLine( angle = topAng, endAbsoluteY = totalHeightHalf, tag = $seg04, ) |> xLine(endAbsolute = totalLen, tag = $seg03) |> yLine(length = -armThick, tag = $seg01) |> angledLineThatIntersects(angle = turns::HALF_TURN, offset = -armThick, intersectTag = seg04) |> angledLine(angle = segAng(seg04) + 180, endAbsoluteY = turns::ZERO) |> angledLine( angle = -bottomAng, endAbsoluteY = -totalHeightHalf - armThick, tag = $seg02, ) |> xLine(endAbsolute = segEndX(seg03) + 0) |> yLine(length = -segLen(seg01)) |> angledLineThatIntersects(angle = turns::HALF_TURN, offset = -armThick, intersectTag = seg02) |> angledLine(angle = segAng(seg02) + 180, endAbsoluteY = -baseHeight) |> xLine(endAbsolute = turns::ZERO) |> close() |> extrude(length = 4)` ) }) await page.setBodyDimensions({ width: 1200, height: 500 }) await homePage.goToModelingScene() await u.openDebugPanel() await u.expectCmdLog('[data-message-type="execution-done"]') await u.waitForCmdReceive('extrude') await page.waitForTimeout(1000) await u.clearAndCloseDebugPanel() if (!tronApp?.projectDirName) { fail() } await doExport( { type: 'gltf', storage: 'embedded', presentation: 'pretty', }, tronApp?.projectDirName, page ) } ) test('Paste should not work unless an input is focused', async ({ page, homePage, }) => { await page.setBodyDimensions({ width: 1200, height: 500 }) await homePage.goToModelingScene() await page .getByRole('button', { name: 'Start Sketch' }) .waitFor({ state: 'visible' }) const codeEditorText = page.locator('.cm-content') const pasteContent = `// was this pasted?` const typeContent = `// this should be typed` // Load text into the clipboard await page.evaluate((t) => navigator.clipboard.writeText(t), pasteContent) // Focus the text editor await codeEditorText.focus() // Show that we can type into it await page.keyboard.type(typeContent) await page.keyboard.press('Enter') // Paste without the code pane focused await codeEditorText.blur() await page.keyboard.press('ControlOrMeta+KeyV') // Show that the paste didn't work but typing did await expect(codeEditorText).not.toContainText(pasteContent) await expect(codeEditorText).toContainText(typeContent) // Paste with the code editor focused // Following this guidance: https://github.com/microsoft/playwright/issues/8114 await codeEditorText.focus() await page.keyboard.press('ControlOrMeta+KeyV') await expect( await page.evaluate( () => document.querySelector('.cm-content')?.textContent ) ).toContain(pasteContent) }) test('Keyboard shortcuts can be viewed through the help menu', async ({ page, homePage, }) => { await page.setBodyDimensions({ width: 1200, height: 500 }) await homePage.goToModelingScene() await page.waitForURL('file:///**', { waitUntil: 'domcontentloaded' }) await page .getByRole('button', { name: 'Start Sketch' }) .waitFor({ state: 'visible' }) // Open the help menu await page.getByRole('button', { name: 'Help and resources' }).click() // Open the keyboard shortcuts await page.getByRole('button', { name: 'Keyboard Shortcuts' }).click() // Verify the URL and that you can see a list of shortcuts await expect.poll(() => page.url()).toContain('?tab=keybindings') await expect( page.getByRole('heading', { name: 'Enter Sketch Mode' }) ).toBeAttached() }) test('First escape in tool pops you out of tool, second exits sketch mode', async ({ page, homePage, toolbar, }) => { // Wait for the app to be ready for use const u = await getUtils(page) await page.setBodyDimensions({ width: 1200, height: 500 }) await homePage.goToModelingScene() await u.openDebugPanel() await u.expectCmdLog('[data-message-type="execution-done"]') await u.closeDebugPanel() // Test these hotkeys perform actions when // focus is on the canvas await page.mouse.move(600, 250) await page.mouse.click(600, 250) // Start a sketch await page.keyboard.press('s') await page.mouse.move(800, 300) await page.mouse.click(800, 300) await page.waitForTimeout(1000) await expect(toolbar.lineBtn).toBeVisible() await expect(toolbar.lineBtn).toHaveAttribute('aria-pressed', 'true') // Draw a line await page.mouse.move(700, 200, { steps: 5 }) await page.mouse.click(700, 200) const secondMousePosition = { x: 800, y: 250 } await page.mouse.move(secondMousePosition.x, secondMousePosition.y, { steps: 5, }) await page.mouse.click(secondMousePosition.x, secondMousePosition.y) // Unequip line tool await page.keyboard.press('Escape') // Make sure we didn't pop out of sketch mode. await expect(page.getByRole('button', { name: 'Exit Sketch' })).toBeVisible() await expect(toolbar.lineBtn).not.toHaveAttribute('aria-pressed', 'true') // Equip arc tool await toolbar.selectTangentialArc() // click in the same position again to continue the profile await page.mouse.move(secondMousePosition.x, secondMousePosition.y, { steps: 5, }) await page.mouse.click(secondMousePosition.x, secondMousePosition.y) await page.mouse.move(1000, 100, { steps: 5 }) await page.mouse.click(1000, 100) await page.keyboard.press('Escape') await expect(toolbar.tangentialArcBtn).toHaveAttribute( 'aria-pressed', 'false' ) await expect .poll(async () => { await page.keyboard.press('l') return toolbar.lineBtn.getAttribute('aria-pressed') }) .toBe('true') // Do not close the sketch. // On close it will exit sketch mode. // Unequip line tool await page.keyboard.press('Escape') await expect(toolbar.lineBtn).toHaveAttribute('aria-pressed', 'false') await expect(toolbar.tangentialArcBtn).toHaveAttribute( 'aria-pressed', 'false' ) // Make sure we didn't pop out of sketch mode. await expect(page.getByRole('button', { name: 'Exit Sketch' })).toBeVisible() // Exit sketch await page.keyboard.press('Escape') await expect( page.getByRole('button', { name: 'Exit Sketch' }) ).not.toBeVisible() }) test('Basic default modeling and sketch hotkeys work', async ({ page, homePage, }) => { const u = await getUtils(page) await test.step(`Set up test`, async () => { await page.addInitScript(async () => { localStorage.setItem( 'store', JSON.stringify({ state: { openPanes: ['code'], }, version: 0, }) ) }) await page.setBodyDimensions({ width: 1200, height: 500 }) await homePage.goToModelingScene() await u.openDebugPanel() await u.expectCmdLog('[data-message-type="execution-done"]') await u.closeDebugPanel() }) const codePane = page.locator('.cm-content') const lineButton = page.getByRole('button', { name: 'line Line', exact: true, }) const arcButton = page.getByRole('button', { name: 'arc Tangential Arc', exact: true, }) const extrudeButton = page.getByRole('button', { name: 'Extrude' }) const commandBarComboBox = page.getByPlaceholder('Search commands') const exitSketchButton = page.getByRole('button', { name: 'Exit Sketch' }) await test.step(`Type code with modeling hotkeys, shouldn't fire`, async () => { await codePane.click() await page.keyboard.type('//') await page.keyboard.press('s') await expect(commandBarComboBox).not.toBeVisible() await page.keyboard.press('e') await expect(commandBarComboBox).not.toBeVisible() await expect(codePane).toHaveText('//se') }) // Blur focus from the code editor, use the s command to sketch await test.step(`Blur editor focus, enter sketch`, async () => { /** * TODO: There is a bug somewhere that causes this test to fail * if you toggle the codePane closed before your trigger the * start of the sketch. * and a separate Safari-only bug that causes the test to fail * if the pane is open the entire test. The maintainer of CodeMirror * has pinpointed this to the unusual browser behavior: * https://discuss.codemirror.net/t/how-to-force-unfocus-of-the-codemirror-element-in-safari/8095/3 */ await blurCodeEditor() await page.waitForTimeout(1000) await page.keyboard.press('s') await page.waitForTimeout(1000) await page.mouse.move(800, 300, { steps: 5 }) await page.mouse.click(800, 300) await page.waitForTimeout(1000) await expect(lineButton).toHaveAttribute('aria-pressed', 'true', { timeout: 15_000, }) }) // Use some sketch hotkeys to create a sketch (l and a for now) await test.step(`Incomplete sketch with hotkeys`, async () => { await test.step(`Draw a line`, async () => { await page.mouse.move(700, 200, { steps: 5 }) await page.mouse.click(700, 200) await page.mouse.move(800, 250, { steps: 5 }) await page.mouse.click(800, 250) }) await test.step(`Unequip line tool`, async () => { await page.keyboard.press('l') await expect(lineButton).not.toHaveAttribute('aria-pressed', 'true') }) await test.step(`Draw a tangential arc`, async () => { await page.keyboard.press('a') await expect(arcButton).toHaveAttribute('aria-pressed', 'true', { timeout: 10_000, }) await page.mouse.move(1000, 100, { steps: 5 }) await page.mouse.click(1000, 100) }) await test.step(`Unequip with escape, equip line tool`, async () => { await page.keyboard.press('Escape') await page.keyboard.press('l') await page.waitForTimeout(50) await expect(lineButton).toHaveAttribute('aria-pressed', 'true') }) }) await test.step(`Type code with sketch hotkeys, shouldn't fire`, async () => { // Since there's code now, we have to get to the end of the line await page.locator('.cm-line').last().click() await page.keyboard.down('ControlOrMeta') await page.keyboard.press('ArrowRight') await page.keyboard.up('ControlOrMeta') await page.keyboard.press('Enter') await page.keyboard.type('//') await page.keyboard.press('l') await expect(lineButton).toHaveAttribute('aria-pressed', 'true') await page.keyboard.press('a') await expect(lineButton).toHaveAttribute('aria-pressed', 'true') await expect(codePane).toContainText('//la') await page.keyboard.press('Backspace') await page.keyboard.press('Backspace') await page.keyboard.press('Backspace') await page.keyboard.press('Backspace') }) await test.step(`Close profile and exit sketch`, async () => { await blurCodeEditor() await page.mouse.move(700, 200, { steps: 5 }) await page.mouse.click(700, 200) // On close it will unequip the line tool. await expect(lineButton).toHaveAttribute('aria-pressed', 'false') await expect(exitSketchButton).toBeEnabled() await page.keyboard.press('Escape') await expect( page.getByRole('button', { name: 'Exit Sketch' }) ).not.toBeVisible() }) // Extrude with e await test.step(`Extrude the sketch`, async () => { await page.mouse.click(750, 150) await blurCodeEditor() await expect(extrudeButton).toBeEnabled() await page.keyboard.press('e') await page.waitForTimeout(500) await page.mouse.move(800, 200, { steps: 5 }) await page.mouse.click(800, 200) await expect(page.getByRole('button', { name: 'Continue' })).toBeVisible({ timeout: 20_000, }) await page.getByRole('button', { name: 'Continue' }).click() await expect( page.getByRole('button', { name: 'Submit command' }) ).toBeVisible() await page.getByRole('button', { name: 'Submit command' }).click() await expect(page.locator('.cm-content')).toContainText('extrude(') }) // await codePaneButton.click() // await expect(u.codeLocator).not.toBeVisible() /** * work-around: to stop `keyboard.press()` from typing in the editor even when it should be blurred */ async function blurCodeEditor() { await page.getByRole('button', { name: 'Commands' }).click() await page.waitForTimeout(100) await page.keyboard.press('Escape') await page.waitForTimeout(100) } }) test('Delete key does not navigate back', async ({ page, homePage }) => { await page.setBodyDimensions({ width: 1200, height: 500 }) await homePage.goToModelingScene() await page.waitForURL('file:///**', { waitUntil: 'domcontentloaded' }) const settingsButton = page.getByRole('link', { name: 'Settings', exact: false, }) const settingsCloseButton = page.getByTestId('settings-close-button') await settingsButton.click() await expect.poll(() => page.url()).toContain('/settings') // Make sure that delete doesn't go back from settings await page.keyboard.press('Delete') await expect.poll(() => page.url()).toContain('/settings') // Now close the settings and try delete again, // make sure it doesn't go back to settings await settingsCloseButton.click() await page.keyboard.press('Delete') await expect.poll(() => page.url()).not.toContain('/settings') }) test('Sketch on face', async ({ page, homePage, scene, cmdBar, toolbar }) => { test.setTimeout(90_000) const u = await getUtils(page) await page.addInitScript(async () => { localStorage.setItem( 'persistCode', `@settings(defaultLengthUnit = in) sketch001 = startSketchOn(XZ) |> startProfile(at = [3.29, 7.86]) |> line(end = [2.48, 2.44]) |> line(end = [2.66, 1.17]) |> line(end = [3.75, 0.46]) |> line(end = [4.99, -0.46]) |> line(end = [3.3, -2.12]) |> line(end = [2.16, -3.33]) |> line(end = [0.85, -3.08]) |> line(end = [-0.18, -3.36]) |> line(end = [-3.86, -2.73]) |> line(end = [-17.67, 0.85]) |> close() extrude001 = extrude(sketch001, length = 5 + 7)` ) }) await page.setBodyDimensions({ width: 1200, height: 500 }) await homePage.goToModelingScene() await scene.connectionEstablished() await scene.settled(cmdBar) let previousCodeContent = await page.locator('.cm-content').innerText() await toolbar.startSketchThenCallbackThenWaitUntilReady(async () => { await u.openAndClearDebugPanel() await u.doAndWaitForCmd( () => page.mouse.click(625, 165), 'default_camera_get_settings', true ) await page.waitForTimeout(150) await u.closeDebugPanel() }) await page.waitForTimeout(300) const firstClickPosition = [612, 238] const secondClickPosition = [661, 242] const thirdClickPosition = [609, 267] await page.mouse.click(firstClickPosition[0], firstClickPosition[1]) await expect(page.locator('.cm-content')).not.toHaveText(previousCodeContent) previousCodeContent = await page.locator('.cm-content').innerText() await page.waitForTimeout(100) await page.mouse.click(secondClickPosition[0], secondClickPosition[1]) await expect(page.locator('.cm-content')).not.toHaveText(previousCodeContent) previousCodeContent = await page.locator('.cm-content').innerText() await page.waitForTimeout(100) await page.mouse.click(thirdClickPosition[0], thirdClickPosition[1]) await expect(page.locator('.cm-content')).not.toHaveText(previousCodeContent) previousCodeContent = await page.locator('.cm-content').innerText() await page.waitForTimeout(100) await page.mouse.click(firstClickPosition[0], firstClickPosition[1]) await expect(page.locator('.cm-content')).not.toHaveText(previousCodeContent) previousCodeContent = await page.locator('.cm-content').innerText() await expect.poll(u.normalisedEditorCode).toContain( u.normalisedCode(`sketch002 = startSketchOn(extrude001, face = seg01) profile001 = startProfile(sketch002, at = [-12.34, 12.34]) |> line(end = [12.34, -12.34]) |> line(end = [-12.34, -12.34]) |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) |> close() `) ) await u.openAndClearDebugPanel() await page.getByRole('button', { name: 'Exit Sketch' }).click() await u.expectCmdLog('[data-message-type="execution-done"]') await u.updateCamPosition([1049, 239, 686]) await u.closeDebugPanel() await page.getByText('startProfile(sketch002, at = [-12').click() await expect(page.getByRole('button', { name: 'Edit Sketch' })).toBeVisible() await page.getByRole('button', { name: 'Edit Sketch' }).click() await page.waitForTimeout(500) await page.setViewportSize({ width: 1200, height: 1200 }) await u.openAndClearDebugPanel() await u.updateCamPosition([452, -152, 1166]) await u.closeDebugPanel() await page.waitForTimeout(200) const pointToDragFirst = [787, 565] await page.mouse.move(pointToDragFirst[0], pointToDragFirst[1]) await page.mouse.down() await page.mouse.move(pointToDragFirst[0] - 20, pointToDragFirst[1], { steps: 5, }) await page.mouse.up() await page.waitForTimeout(100) await expect(page.locator('.cm-content')).not.toHaveText(previousCodeContent) previousCodeContent = await page.locator('.cm-content').innerText() const result = makeTemplate`sketch002 = startSketchOn(extrude001, face = seg01) |> startProfile(at = [-12.83, 6.7]) |> line(end = [${[2.28, 2.35]}, -${0.07}]) |> line(end = [-3.05, -1.47]) |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) |> close()` await expect(page.locator('.cm-content')).toHaveText(result.regExp) // exit sketch await u.openAndClearDebugPanel() await page.getByRole('button', { name: 'Exit Sketch' }).click() await u.expectCmdLog('[data-message-type="execution-done"]') await page.getByText('startProfile(sketch002, at = [-12').click() await expect(page.getByRole('button', { name: 'Extrude' })).not.toBeDisabled() await page.waitForTimeout(100) await page.getByRole('button', { name: 'Extrude' }).click() await expect(page.getByTestId('command-bar')).toBeVisible() await page.waitForTimeout(100) await cmdBar.progressCmdBar() await expect(page.getByText('Confirm Extrude')).toBeVisible() await cmdBar.progressCmdBar() const result2 = result.genNext` const sketch002 = extrude(sketch002, length = ${[5, 5]} + 7)` await expect(page.locator('.cm-content')).toHaveText(result2.regExp) }) test.fixme( `Opening a share link in the web isn't blocked by the web warning banner`, async () => { // This test is not able to be run right now since we don't have a web-only setup for Playwright. // @franknoirot can implement it when that testing infra is set up. It should be a test to cover the fix from // modeling-app issue #6172. } )