import type { Page, Locator } from '@playwright/test' import { expect } from '@playwright/test' type CmdBarSerialised = | { stage: 'commandBarClosed' } | { stage: 'pickCommand' // TODO this will need more properties when implemented in _serialiseCmdBar } | { stage: 'arguments' currentArgKey: string currentArgValue: string headerArguments: Record highlightedHeaderArg: string commandName: string } | { stage: 'review' headerArguments: Record commandName: string } export class CmdBarFixture { public page: Page cmdBarOpenBtn!: Locator cmdBarElement!: Locator constructor(page: Page) { this.page = page this.cmdBarOpenBtn = page.getByTestId('command-bar-open-button') this.cmdBarElement = page.getByTestId('command-bar') } get currentArgumentInput() { return this.page.getByTestId('cmd-bar-arg-value') } reConstruct = (page: Page) => { this.page = page } private _serialiseCmdBar = async (): Promise => { if (!(await this.page.getByTestId('command-bar-wrapper').isVisible())) { return { stage: 'commandBarClosed' } } const reviewForm = this.page.locator('#review-form') const getHeaderArgs = async () => { const inputs = await this.page.getByTestId('cmd-bar-input-tab').all() const entries = await Promise.all( inputs.map((input) => { const key = input .locator('[data-test-name="arg-name"]') .innerText() .then((a) => a.trim()) const value = input .getByTestId('header-arg-value') .innerText() .then((a) => a.trim()) return Promise.all([key, value]) }) ) return Object.fromEntries(entries) } const getCommandName = () => this.page.getByTestId('command-name').textContent() if (await reviewForm.isVisible()) { const [headerArguments, commandName] = await Promise.all([ getHeaderArgs(), getCommandName(), ]) return { stage: 'review', headerArguments, commandName: commandName || '', } } const [ currentArgKey, currentArgValue, headerArguments, highlightedHeaderArg, commandName, ] = await Promise.all([ this.page.getByTestId('cmd-bar-arg-name').textContent(), this.page.getByTestId('cmd-bar-arg-value').textContent(), getHeaderArgs(), this.page .locator('[data-is-current-arg="true"]') .locator('[data-test-name="arg-name"]') .textContent(), getCommandName(), ]) return { stage: 'arguments', currentArgKey: currentArgKey || '', currentArgValue: currentArgValue || '', headerArguments, highlightedHeaderArg: highlightedHeaderArg || '', commandName: commandName || '', } } expectState = async (expected: CmdBarSerialised) => { return expect.poll(() => this._serialiseCmdBar()).toEqual(expected) } /** The method will use buttons OR press enter randomly to progress the cmdbar, * this could have unexpected results depending on what's focused * * TODO: This method assumes the user has a valid input to the current stage, * and assumes we are past the `pickCommand` step. */ progressCmdBar = async (shouldFuzzProgressMethod = true) => { // FIXME: Progressing the command bar is a race condition. We have an async useEffect that reports the final state via useCalculateKclExpression. If this does not run quickly enough, it will not "fail" the continue because you can press continue if the state is not ready. E2E tests do not know this. // Wait 1250ms to assume the await executeAst of the KCL input field is finished await this.page.waitForTimeout(1250) if (shouldFuzzProgressMethod || Math.random() > 0.5) { const arrowButton = this.page.getByRole('button', { name: 'arrow right Continue', }) if (await arrowButton.isVisible()) { await arrowButton.click() } else { await this.page .getByRole('button', { name: 'checkmark Submit command' }) .click() } } else { await this.page.keyboard.press('Enter') } } // Added data-testid to the command bar buttons // command-bar-continue are the buttons to go to the next step // does not include the submit which is the final button press // aka the right arrow button continue = async () => { const continueButton = this.page.getByTestId('command-bar-continue') await continueButton.click() } // Added data-testid to the command bar buttons // command-bar-submit is the button for the final step to submit // the command bar flow aka the checkmark button. submit = async () => { const submitButton = this.page.getByTestId('command-bar-submit') await submitButton.click() } openCmdBar = async (selectCmd?: 'promptToEdit') => { // TODO why does this button not work in electron tests? // await this.cmdBarOpenBtn.click() await this.page.keyboard.down('ControlOrMeta') await this.page.keyboard.press('KeyK') await this.page.keyboard.up('ControlOrMeta') await expect(this.page.getByPlaceholder('Search commands')).toBeVisible() if (selectCmd === 'promptToEdit') { const promptEditCommand = this.page.getByText( 'Use Zoo AI to edit your kcl' ) await expect(promptEditCommand.first()).toBeVisible() await promptEditCommand.first().scrollIntoViewIfNeeded() await promptEditCommand.first().click() } } get cmdSearchInput() { return this.page.getByTestId('cmd-bar-search') } get argumentInput() { return this.page.getByTestId('cmd-bar-arg-value') } get cmdOptions() { return this.page.getByTestId('cmd-bar-option') } chooseCommand = async (commandName: string) => { await this.cmdOptions.getByText(commandName).click() } /** * Select an option from the command bar */ selectOption = (options: Parameters[1]) => { return this.page.getByRole('option', options) } }