/**
 * Copyright (c) Microsoft Corporation.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
import { test, expect } from './inspectorTest';
test.describe('cli codegen', () => {
  test.skip(({ mode }) => mode !== 'default');
  test('should click', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait(``);
    const locator = await recorder.hoverOverElement('button');
    expect(locator).toBe(`getByRole('button', { name: 'Submit' })`);
    const [message, sources] = await Promise.all([
      page.waitForEvent('console', msg => msg.type() !== 'error'),
      recorder.waitForOutput('JavaScript', 'click'),
      recorder.trustedClick(),
    ]);
    expect.soft(sources.get('JavaScript').text).toContain(`
  await page.getByRole('button', { name: 'Submit' }).click();`);
    expect.soft(sources.get('Python').text).toContain(`
    page.get_by_role("button", name="Submit").click()`);
    expect.soft(sources.get('Python Async').text).toContain(`
    await page.get_by_role("button", name="Submit").click()`);
    expect.soft(sources.get('Java').text).toContain(`
      page.getByRole(AriaRole.BUTTON, new Page.GetByRoleOptions().setName("Submit")).click()`);
    expect.soft(sources.get('C#').text).toContain(`
        await page.GetByRole(AriaRole.Button, new() { Name = "Submit" }).ClickAsync();`);
    expect(message.text()).toBe('click');
  });
  test('should ignore programmatic events', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait(``);
    const locator = await recorder.hoverOverElement('button');
    expect(locator).toBe(`getByRole('button', { name: 'Submit' })`);
    await page.dispatchEvent('button', 'click', { detail: 1 });
    await Promise.all([
      page.waitForEvent('console', msg => msg.type() !== 'error'),
      recorder.waitForOutput('JavaScript', 'click'),
      recorder.trustedClick()
    ]);
    const clicks = recorder.sources().get('Playwright Test').actions.filter(l => l.includes('Submit'));
    expect(clicks.length).toBe(1);
  });
  test('should click after same-document navigation', async ({ page, openRecorder, server }) => {
    const recorder = await openRecorder();
    server.setRoute('/foo.html', (req, res) => {
      res.setHeader('Content-Type', 'text/html; charset=utf-8');
      res.end('');
    });
    await recorder.setContentAndWait(``, server.PREFIX + '/foo.html');
    await Promise.all([
      page.waitForNavigation(),
      page.evaluate(() => history.pushState({}, '', '/url.html')),
    ]);
    // This is the only way to give recorder a chance to install
    // the second unnecessary copy of the recorder script.
    await page.waitForTimeout(1000);
    const locator = await recorder.hoverOverElement('button');
    expect(locator).toBe(`getByRole('button', { name: 'Submit' })`);
    const [message, sources] = await Promise.all([
      page.waitForEvent('console', msg => msg.type() !== 'error'),
      recorder.waitForOutput('JavaScript', 'click'),
      recorder.trustedClick(),
    ]);
    expect(sources.get('JavaScript').text).toContain(`
  await page.getByRole('button', { name: 'Submit' }).click();`);
    expect(message.text()).toBe('click');
  });
  test('should make a positioned click on a canvas', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait(`
      
      
    `);
    const locator = await recorder.hoverOverElement('canvas', {
      position: { x: 250, y: 250 },
    });
    expect(locator).toBe(`locator('canvas')`);
    const [message, sources] = await Promise.all([
      page.waitForEvent('console', msg => msg.type() !== 'error'),
      recorder.waitForOutput('JavaScript', 'click'),
      recorder.trustedClick(),
    ]);
    expect(sources.get('JavaScript').text).toContain(`
  await page.locator('canvas').click({
    position: {
      x: 250,
      y: 250
    }
  });`);
    expect(sources.get('Python').text).toContain(`
    page.locator("canvas").click(position={"x":250,"y":250})`);
    expect(sources.get('Python Async').text).toContain(`
    await page.locator("canvas").click(position={"x":250,"y":250})`);
    expect(sources.get('Java').text).toContain(`
      page.locator("canvas").click(new Locator.ClickOptions()
        .setPosition(250, 250));`);
    expect(sources.get('C#').text).toContain(`
        await page.Locator("canvas").ClickAsync(new LocatorClickOptions
        {
            Position = new Position
            {
                X = 250,
                Y = 250,
            },
        });`);
    expect(message.text()).toBe('click 250 250');
  });
  test('should work with TrustedTypes', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait(`
      
        
    
    
      
    `);
    const locator = await recorder.hoverOverElement('button');
    expect(locator).toBe(`getByRole('button', { name: 'Submit' })`);
    const [message, sources] = await Promise.all([
      page.waitForEvent('console', msg => msg.type() !== 'error'),
      recorder.waitForOutput('JavaScript', 'click'),
      recorder.trustedClick(),
    ]);
    expect.soft(sources.get('JavaScript').text).toContain(`
  await page.getByRole('button', { name: 'Submit' }).click();`);
    expect.soft(sources.get('Python').text).toContain(`
    page.get_by_role("button", name="Submit").click()`);
    expect.soft(sources.get('Python Async').text).toContain(`
    await page.get_by_role("button", name="Submit").click()`);
    expect.soft(sources.get('Java').text).toContain(`
      page.getByRole(AriaRole.BUTTON, new Page.GetByRoleOptions().setName("Submit")).click()`);
    expect.soft(sources.get('C#').text).toContain(`
        await page.GetByRole(AriaRole.Button, new() { Name = "Submit" }).ClickAsync();`);
    expect(message.text()).toBe('click');
  });
  test('should not target selector preview by text regexp', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait(`dummy`);
    // Force highlight.
    await recorder.hoverOverElement('span');
    // Append text after highlight.
    await page.evaluate(() => {
      const div = document.createElement('div');
      div.setAttribute('onclick', "console.log('click')");
      div.textContent = ' Some long text here ';
      document.body.appendChild(div);
    });
    const locator = await recorder.hoverOverElement('div');
    expect(locator).toBe(`getByText('Some long text here')`);
    const divContents = await page.$eval('div', div => div.outerHTML);
    expect(divContents.replace(/\s__playwright_target__="[^"]+"/, '')).toBe(` Some long text here 
`);
    const [message, sources] = await Promise.all([
      page.waitForEvent('console', msg => msg.type() !== 'error'),
      recorder.waitForOutput('JavaScript', 'click'),
      recorder.trustedMove('div').then(() => recorder.trustedClick()),
    ]);
    expect(sources.get('JavaScript').text).toContain(`
  await page.getByText('Some long text here').click();`);
    expect(message.text()).toBe('click');
  });
  test('should fill', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait(``);
    const locator = await recorder.focusElement('input');
    expect(locator).toBe(`locator('#input')`);
    const [message, sources] = await Promise.all([
      page.waitForEvent('console', msg => msg.type() !== 'error'),
      recorder.waitForOutput('JavaScript', 'fill'),
      page.fill('input', 'John')
    ]);
    expect(sources.get('JavaScript').text).toContain(`
  await page.locator('#input').fill('John');`);
    expect(sources.get('Java').text).toContain(`
      page.locator("#input").fill("John");`);
    expect(sources.get('Python').text).toContain(`
    page.locator("#input").fill(\"John\")`);
    expect(sources.get('Python Async').text).toContain(`
    await page.locator("#input").fill(\"John\")`);
    expect(sources.get('C#').text).toContain(`
        await page.Locator("#input").FillAsync(\"John\");`);
    expect(message.text()).toBe('John');
  });
  test('should fill japanese text', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    // In Japanese, "てすと" or "テスト" means "test".
    await recorder.setContentAndWait(``);
    const locator = await recorder.focusElement('input');
    expect(locator).toBe(`locator('#input')`);
    const [message, sources] = await Promise.all([
      page.waitForEvent('console', msg => msg.type() !== 'error'),
      recorder.waitForOutput('JavaScript', 'fill'),
      (async () => {
        await recorder.page.dispatchEvent('input', 'keydown', { key: 'Process' });
        await recorder.page.keyboard.insertText('てすと');
        await recorder.page.dispatchEvent('input', 'keyup', { key: 'Process' });
      })()
    ]);
    expect(sources.get('JavaScript').text).toContain(`
  await page.locator('#input').fill('てすと');`);
    expect(sources.get('Java').text).toContain(`
      page.locator("#input").fill("てすと");`);
    expect(sources.get('Python').text).toContain(`
    page.locator("#input").fill(\"てすと\")`);
    expect(sources.get('Python Async').text).toContain(`
    await page.locator("#input").fill(\"てすと\")`);
    expect(sources.get('C#').text).toContain(`
        await page.Locator("#input").FillAsync(\"てすと\");`);
    expect(message.text()).toBe('てすと');
  });
  test('should fill textarea', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait(``);
    const locator = await recorder.focusElement('textarea');
    expect(locator).toBe(`locator('#textarea')`);
    const [message, sources] = await Promise.all([
      page.waitForEvent('console', msg => msg.type() !== 'error'),
      recorder.waitForOutput('JavaScript', 'fill'),
      page.fill('textarea', 'John')
    ]);
    expect(sources.get('JavaScript').text).toContain(`
  await page.locator('#textarea').fill('John');`);
    expect(message.text()).toBe('John');
  });
  test('should fill [contentEditable]', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait(``);
    const locator = await recorder.focusElement('div');
    expect(locator).toBe(`locator('#content')`);
    const [message, sources] = await Promise.all([
      page.waitForEvent('console', msg => msg.type() !== 'error'),
      recorder.waitForOutput('JavaScript', 'fill'),
      page.fill('div', 'John Doe')
    ]);
    expect(sources.get('JavaScript').text).toContain(`
  await page.locator('#content').fill('John Doe');`);
    expect(message.text()).toBe('John Doe');
  });
  test('should press', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait(``);
    const locator = await recorder.focusElement('input');
    expect(locator).toBe(`getByRole('textbox')`);
    const messages: any[] = [];
    page.on('console', message => messages.push(message));
    const [, sources] = await Promise.all([
      recorder.waitForActionPerformed(),
      recorder.waitForOutput('JavaScript', 'press'),
      page.press('input', 'Shift+Enter')
    ]);
    expect(sources.get('JavaScript').text).toContain(`
  await page.getByRole('textbox').press('Shift+Enter');`);
    expect(sources.get('Java').text).toContain(`
      page.getByRole(AriaRole.TEXTBOX).press("Shift+Enter");`);
    expect(sources.get('Python').text).toContain(`
    page.get_by_role("textbox").press("Shift+Enter")`);
    expect(sources.get('Python Async').text).toContain(`
    await page.get_by_role("textbox").press("Shift+Enter")`);
    expect(sources.get('C#').text).toContain(`
        await page.GetByRole(AriaRole.Textbox).PressAsync("Shift+Enter");`);
    expect(messages[0].text()).toBe('press');
  });
  test('should update selected element after pressing Tab', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait(`
      
      
    `);
    await page.click('input[name="one"]');
    await recorder.waitForOutput('JavaScript', 'click');
    await page.keyboard.type('foobar123');
    await recorder.waitForOutput('JavaScript', 'foobar123');
    await page.keyboard.press('Tab');
    await recorder.waitForOutput('JavaScript', 'Tab');
    await page.keyboard.type('barfoo321');
    await recorder.waitForOutput('JavaScript', 'barfoo321');
    const text = recorder.sources().get('JavaScript').text;
    expect(text).toContain(`
  await page.locator('input[name="one"]').fill('foobar123');`);
    expect(text).toContain(`
  await page.locator('input[name="one"]').press('Tab');`);
    expect(text).toContain(`
  await page.locator('input[name="two"]').fill('barfoo321');`);
  });
  test('should record ArrowDown', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait(``);
    const locator = await recorder.focusElement('input');
    expect(locator).toBe(`getByRole('textbox')`);
    const messages: any[] = [];
    page.on('console', message => {
      messages.push(message);
    });
    const [, sources] = await Promise.all([
      recorder.waitForActionPerformed(),
      recorder.waitForOutput('JavaScript', 'press'),
      page.press('input', 'ArrowDown')
    ]);
    expect(sources.get('JavaScript').text).toContain(`
  await page.getByRole('textbox').press('ArrowDown');`);
    expect(messages[0].text()).toBe('press:ArrowDown');
  });
  test('should emit single keyup on ArrowDown', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait(``);
    const locator = await recorder.focusElement('input');
    expect(locator).toBe(`getByRole('textbox')`);
    const messages: any[] = [];
    page.on('console', message => {
      if (message.type() !== 'error')
        messages.push(message);
    });
    const [, sources] = await Promise.all([
      recorder.waitForActionPerformed(),
      recorder.waitForOutput('JavaScript', 'press'),
      page.press('input', 'ArrowDown')
    ]);
    expect(sources.get('JavaScript').text).toContain(`
  await page.getByRole('textbox').press('ArrowDown');`);
    expect(messages.length).toBe(2);
    expect(messages[0].text()).toBe('down:ArrowDown');
    expect(messages[1].text()).toBe('up:ArrowDown');
  });
  test('should check', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait(``);
    const locator = await recorder.focusElement('input');
    expect(locator).toBe(`locator('#checkbox')`);
    const [message, sources] = await Promise.all([
      page.waitForEvent('console', msg => msg.type() !== 'error'),
      recorder.waitForOutput('JavaScript', 'check'),
      page.click('input')
    ]);
    expect(sources.get('JavaScript').text).toContain(`
  await page.locator('#checkbox').check();`);
    expect(sources.get('Java').text).toContain(`
      page.locator("#checkbox").check();`);
    expect(sources.get('Python').text).toContain(`
    page.locator("#checkbox").check()`);
    expect(sources.get('Python Async').text).toContain(`
    await page.locator("#checkbox").check()`);
    expect(sources.get('C#').text).toContain(`
        await page.Locator("#checkbox").CheckAsync();`);
    expect(message.text()).toBe('true');
  });
  test('should check a radio button', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait(``);
    const locator = await recorder.focusElement('input');
    expect(locator).toBe(`locator('#checkbox')`);
    const [message, sources] = await Promise.all([
      page.waitForEvent('console', msg => msg.type() !== 'error'),
      recorder.waitForOutput('JavaScript', 'check'),
      page.click('input')
    ]);
    expect(sources.get('JavaScript').text).toContain(`
  await page.locator('#checkbox').check();`);
    expect(message.text()).toBe('true');
  });
  test('should check with keyboard', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait(``);
    const locator = await recorder.focusElement('input');
    expect(locator).toBe(`locator('#checkbox')`);
    const [message, sources] = await Promise.all([
      page.waitForEvent('console', msg => msg.type() !== 'error'),
      recorder.waitForOutput('JavaScript', 'check'),
      page.keyboard.press('Space')
    ]);
    expect(sources.get('JavaScript').text).toContain(`
  await page.locator('#checkbox').check();`);
    expect(message.text()).toBe('true');
  });
  test('should uncheck', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait(``);
    const locator = await recorder.focusElement('input');
    expect(locator).toBe(`locator('#checkbox')`);
    const [message, sources] = await Promise.all([
      page.waitForEvent('console', msg => msg.type() !== 'error'),
      recorder.waitForOutput('JavaScript', 'uncheck'),
      page.click('input')
    ]);
    expect(sources.get('JavaScript').text).toContain(`
  await page.locator('#checkbox').uncheck();`);
    expect(sources.get('Java').text).toContain(`
      page.locator("#checkbox").uncheck();`);
    expect(sources.get('Python').text).toContain(`
    page.locator("#checkbox").uncheck()`);
    expect(sources.get('Python Async').text).toContain(`
    await page.locator("#checkbox").uncheck()`);
    expect(sources.get('C#').text).toContain(`
        await page.Locator("#checkbox").UncheckAsync();`);
    expect(message.text()).toBe('false');
  });
  test('should select', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait('');
    const locator = await recorder.hoverOverElement('select');
    expect(locator).toBe(`locator('#age')`);
    const [message, sources] = await Promise.all([
      page.waitForEvent('console', msg => msg.type() !== 'error'),
      recorder.waitForOutput('JavaScript', 'select'),
      page.selectOption('select', '2')
    ]);
    expect(sources.get('JavaScript').text).toContain(`
  await page.locator('#age').selectOption('2');`);
    expect(sources.get('Java').text).toContain(`
      page.locator("#age").selectOption("2");`);
    expect(sources.get('Python').text).toContain(`
    page.locator("#age").select_option("2")`);
    expect(sources.get('Python Async').text).toContain(`
    await page.locator("#age").select_option("2")`);
    expect(sources.get('C#').text).toContain(`
        await page.Locator("#age").SelectOptionAsync(new[] { "2" });`);
    expect(message.text()).toBe('2');
  });
  test('should select with size attribute', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait(`
      
      
    `);
    const locator = await recorder.hoverOverElement('select');
    expect(locator).toBe(`locator('#age')`);
    const [message, sources] = await Promise.all([
      page.waitForEvent('console', msg => msg.type() !== 'error'),
      recorder.waitForOutput('JavaScript', 'select'),
      page.mouse.click(10, 25)
    ]);
    expect(sources.get('JavaScript').text).toContain(`
  await page.locator('#age').selectOption('2');`);
    expect(sources.get('Java').text).toContain(`
      page.locator("#age").selectOption("2");`);
    expect(sources.get('Python').text).toContain(`
    page.locator(\"#age\").select_option(\"2\")`);
    expect(sources.get('Python Async').text).toContain(`
    await page.locator(\"#age\").select_option(\"2\")`);
    expect(sources.get('C#').text).toContain(`
        await page.Locator(\"#age\").SelectOptionAsync(new[] { \"2\" });`);
    expect(message.text()).toBe('2');
  });
  test('should await popup', async ({ page, openRecorder, browserName, headless }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait('link');
    const locator = await recorder.hoverOverElement('a');
    expect(locator).toBe(`getByRole('link', { name: 'link' })`);
    const [popup, sources] = await Promise.all([
      page.context().waitForEvent('page'),
      recorder.waitForOutput('JavaScript', 'waitForEvent'),
      recorder.trustedClick(),
    ]);
    expect.soft(sources.get('JavaScript').text).toContain(`
  const page1Promise = page.waitForEvent('popup');
  await page.getByRole('link', { name: 'link' }).click();
  const page1 = await page1Promise;`);
    expect.soft(sources.get('Java').text).toContain(`
      Page page1 = page.waitForPopup(() -> {
        page.getByRole(AriaRole.LINK, new Page.GetByRoleOptions().setName("link")).click();
      });`);
    expect.soft(sources.get('Python').text).toContain(`
    with page.expect_popup() as page1_info:
        page.get_by_role("link", name="link").click()
    page1 = page1_info.value`);
    expect.soft(sources.get('Python Async').text).toContain(`
    async with page.expect_popup() as page1_info:
        await page.get_by_role("link", name="link").click()
    page1 = await page1_info.value`);
    expect.soft(sources.get('C#').text).toContain(`
        var page1 = await page.RunAndWaitForPopupAsync(async () =>
        {
            await page.GetByRole(AriaRole.Link, new() { Name = "link" }).ClickAsync();
        });`);
    expect(popup.url()).toBe('about:blank');
  });
  test('should assert navigation', async ({ page, openRecorder }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait(`link`);
    const locator = await recorder.hoverOverElement('a');
    expect(locator).toBe(`getByText('link')`);
    const [, sources] = await Promise.all([
      page.waitForNavigation(),
      recorder.waitForOutput('JavaScript', '.click()'),
      recorder.trustedClick(),
    ]);
    expect.soft(sources.get('JavaScript').text).toContain(`
  await page.getByText('link').click();`);
    expect.soft(sources.get('Playwright Test').text).toContain(`
  await page.getByText('link').click();`);
    expect.soft(sources.get('Java').text).toContain(`
      page.getByText("link").click();`);
    expect.soft(sources.get('Python').text).toContain(`
    page.get_by_text("link").click()`);
    expect.soft(sources.get('Python Async').text).toContain(`
    await page.get_by_text("link").click()`);
    expect.soft(sources.get('Pytest').text).toContain(`
    page.get_by_text("link").click()`);
    expect.soft(sources.get('C#').text).toContain(`
        await page.GetByText("link").ClickAsync();`);
    expect(page.url()).toContain('about:blank#foo');
  });
  test('should ignore AltGraph', async ({ openRecorder, browserName }) => {
    test.skip(browserName === 'firefox', 'The TextInputProcessor in Firefox does not work with AltGraph.');
    const recorder = await openRecorder();
    await recorder.setContentAndWait(``);
    await recorder.page.type('input', 'playwright');
    await recorder.page.keyboard.press('AltGraph');
    await recorder.page.keyboard.insertText('@');
    await recorder.page.keyboard.type('example.com');
    await recorder.waitForOutput('JavaScript', 'example.com');
    expect(recorder.sources().get('JavaScript').text).not.toContain(`await page.getByRole('textbox').press('AltGraph');`);
    expect(recorder.sources().get('JavaScript').text).toContain(`await page.getByRole('textbox').fill('playwright@example.com');`);
  });
  test('should middle click', async ({ page, openRecorder, server }) => {
    const recorder = await openRecorder();
    await recorder.setContentAndWait(`Click me`);
    const [sources] = await Promise.all([
      recorder.waitForOutput('JavaScript', 'click'),
      page.click('a', { button: 'middle' }),
    ]);
    expect(sources.get('JavaScript').text).toContain(`
  await page.getByText('Click me').click({
    button: 'middle'
  });`);
    expect(sources.get('Python').text).toContain(`
    page.get_by_text("Click me").click(button="middle")`);
    expect(sources.get('Python Async').text).toContain(`
    await page.get_by_text("Click me").click(button="middle")`);
    expect(sources.get('Java').text).toContain(`
      page.getByText("Click me").click(new Locator.ClickOptions()
        .setButton(MouseButton.MIDDLE));`);
    expect(sources.get('C#').text).toContain(`
        await page.GetByText("Click me").ClickAsync(new LocatorClickOptions
        {
            Button = MouseButton.Middle,
        });`);
  });
});