/**
 * Copyright 2018 Google Inc. All rights reserved.
 * Modifications 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 as it, expect } from './pageTest';
it('should work @smoke', async ({ page }) => {
  await page.setContent(`
yo
ya
\nye  
`);
  expect(await page.$eval(`text=ya`, e => e.outerHTML)).toBe('ya
');
  expect(await page.$eval(`text="ya"`, e => e.outerHTML)).toBe('ya
');
  expect(await page.$eval(`text=/^[ay]+$/`, e => e.outerHTML)).toBe('ya
');
  expect(await page.$eval(`text=/Ya/i`, e => e.outerHTML)).toBe('ya
');
  expect(await page.$eval(`text=ye`, e => e.outerHTML)).toBe('\nye  
');
  await page.setContent(` ye 
ye
`);
  expect(await page.$eval(`text="ye"`, e => e.outerHTML)).toBe(' ye 
');
  await page.setContent(`yo
"ya
 hello world! 
`);
  expect(await page.$eval(`text="\\"ya"`, e => e.outerHTML)).toBe('"ya
');
  expect(await page.$eval(`text=/hello/`, e => e.outerHTML)).toBe(' hello world! 
');
  expect(await page.$eval(`text=/^\\s*heLLo/i`, e => e.outerHTML)).toBe(' hello world! 
');
  await page.setContent(``);
  expect(await page.$eval(`text=hey`, e => e.outerHTML)).toBe('hey
');
  expect(await page.$eval(`text=yo>>text="ya"`, e => e.outerHTML)).toBe('ya
');
  expect(await page.$eval(`text=yo>> text="ya"`, e => e.outerHTML)).toBe('ya
');
  expect(await page.$eval(`text=yo >>text='ya'`, e => e.outerHTML)).toBe('ya
');
  expect(await page.$eval(`text=yo >> text='ya'`, e => e.outerHTML)).toBe('ya
');
  expect(await page.$eval(`'yo'>>"ya"`, e => e.outerHTML)).toBe('ya
');
  expect(await page.$eval(`"yo" >> 'ya'`, e => e.outerHTML)).toBe('ya
');
  await page.setContent(`yo
yo
`);
  expect(await page.$$eval(`text=yo`, es => es.map(e => e.outerHTML).join('\n'))).toBe('yo
\nyo
');
  await page.setContent(`'
"
\\
x
`);
  expect(await page.$eval(`text='\\''`, e => e.outerHTML)).toBe('\'
');
  expect(await page.$eval(`text='"'`, e => e.outerHTML)).toBe('"
');
  expect(await page.$eval(`text="\\""`, e => e.outerHTML)).toBe('"
');
  expect(await page.$eval(`text="'"`, e => e.outerHTML)).toBe('\'
');
  expect(await page.$eval(`text="\\x"`, e => e.outerHTML)).toBe('x
');
  expect(await page.$eval(`text='\\x'`, e => e.outerHTML)).toBe('x
');
  expect(await page.$eval(`text='\\\\'`, e => e.outerHTML)).toBe('\\
');
  expect(await page.$eval(`text="\\\\"`, e => e.outerHTML)).toBe('\\
');
  expect(await page.$eval(`text="`, e => e.outerHTML)).toBe('"
');
  expect(await page.$eval(`text='`, e => e.outerHTML)).toBe('\'
');
  expect(await page.$eval(`"x"`, e => e.outerHTML)).toBe('x
');
  expect(await page.$eval(`'x'`, e => e.outerHTML)).toBe('x
');
  let error = await page.$(`"`).catch(e => e);
  expect(error).toBeInstanceOf(Error);
  error = await page.$(`'`).catch(e => e);
  expect(error).toBeInstanceOf(Error);
  await page.setContent(` ' 
 " 
`);
  expect(await page.$eval(`text="`, e => e.outerHTML)).toBe(' " 
');
  expect(await page.$eval(`text='`, e => e.outerHTML)).toBe(' \' 
');
  await page.setContent(`Hi''>>foo=bar
`);
  expect(await page.$eval(`text="Hi''>>foo=bar"`, e => e.outerHTML)).toBe(`Hi''>>foo=bar
`);
  await page.setContent(`Hi'">>foo=bar
`);
  expect(await page.$eval(`text="Hi'\\">>foo=bar"`, e => e.outerHTML)).toBe(`Hi'">>foo=bar
`);
  await page.setContent(`Hi>>
`);
  expect(await page.$eval(`text="Hi>>">>span`, e => e.outerHTML)).toBe(`a
a
`);
  expect(await page.$eval(`text=a`, e => e.outerHTML)).toBe('a
');
  expect(await page.$eval(`text=b`, e => e.outerHTML)).toBe('a
');
  expect(await page.$eval(`text=ab`, e => e.outerHTML)).toBe('a
');
  expect(await page.$(`text=abc`)).toBe(null);
  expect(await page.$$eval(`text=a`, els => els.length)).toBe(2);
  expect(await page.$$eval(`text=b`, els => els.length)).toBe(1);
  expect(await page.$$eval(`text=ab`, els => els.length)).toBe(1);
  expect(await page.$$eval(`text=abc`, els => els.length)).toBe(0);
  await page.setContent(`
helloworld
');
  expect(await page.$$eval(`text=lowo`, els => els.map(e => e.outerHTML).join(''))).toBe('helloworld
helloworld ');
  await page.setContent(`Sign in Hello\n \nworld `);
  expect(await page.$eval(`text=Sign in`, e => e.outerHTML)).toBe('Sign in ');
  expect((await page.$$(`text=Sign \tin`)).length).toBe(1);
  expect((await page.$$(`text="Sign in"`)).length).toBe(1);
  expect(await page.$eval(`text=lo wo`, e => e.outerHTML)).toBe('Hello\n \nworld ');
  expect(await page.$eval(`text="Hello world"`, e => e.outerHTML)).toBe('Hello\n \nworld ');
  expect(await page.$(`text="lo wo"`)).toBe(null);
  expect((await page.$$(`text=lo \nwo`)).length).toBe(1);
  expect((await page.$$(`text="lo \nwo"`)).length).toBe(0);
});
it('should work with :text', async ({ page }) => {
  await page.setContent(`yo
ya
\nHELLO   \n world  
`);
  expect(await page.$eval(`:text("ya")`, e => e.outerHTML)).toBe('ya
');
  expect(await page.$eval(`:text-is("ya")`, e => e.outerHTML)).toBe('ya
');
  expect(await page.$eval(`:text("y")`, e => e.outerHTML)).toBe('yo
');
  expect(await page.$(`:text-is("Y")`)).toBe(null);
  expect(await page.$eval(`:text("hello world")`, e => e.outerHTML)).toBe('\nHELLO   \n world  
');
  expect(await page.$eval(`:text-is("HELLO world")`, e => e.outerHTML)).toBe('\nHELLO   \n world  
');
  expect(await page.$eval(`:text("lo wo")`, e => e.outerHTML)).toBe('\nHELLO   \n world  
');
  expect(await page.$(`:text-is("lo wo")`)).toBe(null);
  expect(await page.$eval(`:text-matches("^[ay]+$")`, e => e.outerHTML)).toBe('ya
');
  expect(await page.$eval(`:text-matches("y", "g")`, e => e.outerHTML)).toBe('yo
');
  expect(await page.$eval(`:text-matches("Y", "i")`, e => e.outerHTML)).toBe('yo
');
  expect(await page.$(`:text-matches("^y$")`)).toBe(null);
  const error1 = await page.$(`:text("foo", "bar")`).catch(e => e);
  expect(error1.message).toContain(`"text" engine expects a single string`);
  const error2 = await page.$(`:text(foo > bar)`).catch(e => e);
  expect(error2.message).toContain(`"text" engine expects a single string`);
});
it('should support empty string', async ({ page }) => {
  await page.setContent(`
ya
\nHELLO   \n world  
`);
  expect(await page.$eval(`div:text-is("")`, e => e.outerHTML)).toBe('
');
  expect(await page.$$eval(`div:text-is("")`, els => els.length)).toBe(1);
  expect(await page.$eval(`div:text("")`, e => e.outerHTML)).toBe('
');
  expect(await page.$$eval(`div:text("")`, els => els.length)).toBe(3);
  expect(await page.$eval(`div >> text=""`, e => e.outerHTML)).toBe('
');
  expect(await page.$$eval(`div >> text=""`, els => els.length)).toBe(1);
  expect(await page.$eval(`div >> text=/^$/`, e => e.outerHTML)).toBe('
');
  expect(await page.$$eval(`div >> text=/^$/`, els => els.length)).toBe(1);
  expect(await page.$eval(`div:text-matches("")`, e => e.outerHTML)).toBe('
');
  expect(await page.$$eval(`div:text-matches("")`, els => els.length)).toBe(3);
});
it('should work across nodes', async ({ page }) => {
  await page.setContent(`Hello,  world ! 
`);
  expect(await page.$eval(`:text("Hello, world!")`, e => e.id)).toBe('target1');
  expect(await page.$eval(`:text("Hello")`, e => e.id)).toBe('target1');
  expect(await page.$eval(`:text("world")`, e => e.id)).toBe('target2');
  expect(await page.$$eval(`:text("world")`, els => els.length)).toBe(1);
  expect(await page.$(`:text("hello world")`)).toBe(null);
  expect(await page.$(`div:text("world")`)).toBe(null);
  expect(await page.$eval(`text=Hello, world!`, e => e.id)).toBe('target1');
  expect(await page.$eval(`text=Hello`, e => e.id)).toBe('target1');
  expect(await page.$eval(`text=world`, e => e.id)).toBe('target2');
  expect(await page.$$eval(`text=world`, els => els.length)).toBe(1);
  expect(await page.$(`text=hello world`)).toBe(null);
  expect(await page.$(`:text-is("Hello, world!")`)).toBe(null);
  expect(await page.$eval(`:text-is("Hello")`, e => e.id)).toBe('target1');
  expect(await page.$eval(`:text-is("world")`, e => e.id)).toBe('target2');
  expect(await page.$$eval(`:text-is("world")`, els => els.length)).toBe(1);
  expect(await page.$(`text="Hello, world!"`)).toBe(null);
  expect(await page.$eval(`text="Hello"`, e => e.id)).toBe('target1');
  expect(await page.$eval(`text="world"`, e => e.id)).toBe('target2');
  expect(await page.$$eval(`text="world"`, els => els.length)).toBe(1);
  expect(await page.$eval(`:text-matches(".*")`, e => e.nodeName)).toBe('I');
  expect(await page.$eval(`:text-matches("world?")`, e => e.id)).toBe('target2');
  expect(await page.$$eval(`:text-matches("world")`, els => els.length)).toBe(1);
  expect(await page.$(`div:text(".*")`)).toBe(null);
  expect(await page.$eval(`text=/.*/`, e => e.nodeName)).toBe('I');
  expect(await page.$eval(`text=/world?/`, e => e.id)).toBe('target2');
  expect(await page.$$eval(`text=/world/`, els => els.length)).toBe(1);
});
it('should work with text nodes in quoted mode', async ({ page }) => {
  await page.setContent(`Hellowo  rld     Hi again  
`);
  expect(await page.$eval(`text="Hello"`, e => e.id)).toBe('target1');
  expect(await page.$eval(`text="Hi again"`, e => e.id)).toBe('target1');
  expect(await page.$eval(`text="wo rld"`, e => e.id)).toBe('target2');
  expect(await page.$(`text="Hellowo rld Hi again"`)).toBe(null);
  expect(await page.$(`text="Hellowo"`)).toBe(null);
  expect(await page.$(`text="Hellowo rld"`)).toBe(null);
  expect(await page.$(`text="wo rld Hi ag"`)).toBe(null);
  expect(await page.$(`text="again"`)).toBe(null);
  expect(await page.$(`text="hi again"`)).toBe(null);
  expect(await page.$eval(`text=hi again`, e => e.id)).toBe('target1');
});
it('should clear caches', async ({ page }) => {
  await page.setContent(`text
text
`);
  const div = await page.$('#target1');
  await div.evaluate(div => div.textContent = 'text');
  expect(await page.$eval(`text=text`, e => e.id)).toBe('target1');
  await div.evaluate(div => div.textContent = 'foo');
  expect(await page.$eval(`text=text`, e => e.id)).toBe('target2');
  await div.evaluate(div => div.textContent = 'text');
  expect(await page.$eval(`:text("text")`, e => e.id)).toBe('target1');
  await div.evaluate(div => div.textContent = 'foo');
  expect(await page.$eval(`:text("text")`, e => e.id)).toBe('target2');
  await div.evaluate(div => div.textContent = 'text');
  expect(await page.$$eval(`text=text`, els => els.length)).toBe(2);
  await div.evaluate(div => div.textContent = 'foo');
  expect(await page.$$eval(`text=text`, els => els.length)).toBe(1);
  await div.evaluate(div => div.textContent = 'text');
  expect(await page.$$eval(`:text("text")`, els => els.length)).toBe(2);
  await div.evaluate(div => div.textContent = 'foo');
  expect(await page.$$eval(`:text("text")`, els => els.length)).toBe(1);
});
it('should work with :has-text', async ({ page }) => {
  await page.setContent(`
      Find me   ');
  expect(await page.$eval(`div:has-text("find me")`, e => e.id)).toBe('div1');
  expect(await page.$eval(`div:has-text("find me") input`, e => e.id)).toBe('input1');
  expect(await page.$eval(`:has-text("find me") input`, e => e.id)).toBe('input2');
  expect(await page.$eval(`div:has-text("find me or maybe me")`, e => e.id)).toBe('div1');
  expect(await page.$(`div:has-text("find noone")`)).toBe(null);
  expect(await page.$$eval(`:is(div,span):has-text("maybe")`, els => els.map(e => e.id).join(';'))).toBe('div1;span2');
  expect(await page.$eval(`div:has-text("find me") :has-text("maybe me")`, e => e.tagName)).toBe('WRAP');
  expect(await page.$eval(`div:has-text("find me") span:has-text("maybe me")`, e => e.id)).toBe('span2');
  const error1 = await page.$(`:has-text("foo", "bar")`).catch(e => e);
  expect(error1.message).toContain(`"has-text" engine expects a single string`);
  const error2 = await page.$(`:has-text(foo > bar)`).catch(e => e);
  expect(error2.message).toContain(`"has-text" engine expects a single string`);
});
it('should work with large DOM', async ({ page }) => {
  await page.evaluate(() => {
    let id = 0;
    const next = (tag: string) => {
      const e = document.createElement(tag);
      const eid = ++id;
      e.textContent = 'id' + eid;
      e.id = 'id' + eid;
      return e;
    };
    const generate = (depth: number) => {
      const div = next('div');
      const span1 = next('span');
      const span2 = next('span');
      div.appendChild(span1);
      div.appendChild(span2);
      if (depth > 0) {
        div.appendChild(generate(depth - 1));
        div.appendChild(generate(depth - 1));
      }
      return div;
    };
    document.body.appendChild(generate(12));
  });
  const selectors = [
    ':has-text("id18")',
    ':has-text("id12345")',
    ':has-text("id")',
    ':text("id18")',
    ':text("id12345")',
    ':text("id")',
    ':text-matches("id12345", "i")',
    'text=id18',
    'text=id12345',
    'text=id',
    '#id18',
    '#id12345',
    '*',
  ];
  const measure = false;
  for (const selector of selectors) {
    const time1 = Date.now();
    for (let i = 0; i < (measure ? 10 : 1); i++)
      await page.$$eval(selector, els => els.length);
    if (measure)
      console.log(`pw("${selector}"): ` + (Date.now() - time1));
    if (measure && !selector.includes('text')) {
      const time2 = Date.now();
      for (let i = 0; i < (measure ? 10 : 1); i++)
        await page.evaluate(selector => document.querySelectorAll(selector).length, selector);
      console.log(`qs("${selector}"): ` + (Date.now() - time2));
    }
  }
});
it('should be case sensitive if quotes are specified', async ({ page }) => {
  await page.setContent(`yo
ya
\nye  
`);
  expect(await page.$eval(`text=yA`, e => e.outerHTML)).toBe('ya
');
  expect(await page.$(`text="yA"`)).toBe(null);
  expect(await page.$(`text= "ya"`)).toBe(null);
});
it('should search for a substring without quotes', async ({ page }) => {
  await page.setContent(`textwithsubstring
`);
  expect(await page.$eval(`text=with`, e => e.outerHTML)).toBe('textwithsubstring
');
  expect(await page.$(`text="with"`)).toBe(null);
});
it('should skip head, script and style', async ({ page }) => {
  await page.setContent(`
    
      title 
      
      
    
    
      
      
      title script style
    `);
  const head = await page.$('head');
  const title = await page.$('title');
  const script = await page.$('body script');
  const style = await page.$('body style');
  for (const text of ['title', 'script', 'style']) {
    expect(await page.$eval(`text=${text}`, e => e.nodeName)).toBe('DIV');
    expect(await page.$$eval(`text=${text}`, els => els.map(e => e.nodeName).join('|'))).toBe('DIV');
    for (const root of [head, title, script, style]) {
      expect(await root.$(`text=${text}`)).toBe(null);
      expect(await root.$$eval(`text=${text}`, els => els.length)).toBe(0);
    }
  }
});
it('should match input[type=button|submit]', async ({ page }) => {
  await page.setContent(` hello world    hellow  world    `);
  expect(await page.$$eval('*css=button >> text=hello >> text=world', els => els.length)).toBe(2);
});
it('should work with leading and trailing spaces', async ({ page }) => {
  it.info().annotations.push({ type: 'issue', description: 'https://github.com/microsoft/playwright/issues/10997' });
  await page.setContent(` Add widget  `);
  await expect(page.locator('text=Add widget')).toBeVisible();
  await expect(page.locator('text= Add widget ')).toBeVisible();
});
it('should work with unpaired quotes when not at the start', async ({ page }) => {
  it.info().annotations.push({ type: 'issue', description: 'https://github.com/microsoft/playwright/issues/12719' });
  await page.setContent(`
    hello"worldyay 
    hello'worldnay 
    hello\`worldoh 
    hello\`worldoh2 
  `);
  expect(await page.$eval('text=lo" >> span', e => e.outerHTML)).toBe('yay ');
  expect(await page.$eval('  text=lo" >> span', e => e.outerHTML)).toBe('yay ');
  expect(await page.$eval('text  =lo" >> span', e => e.outerHTML)).toBe('yay ');
  expect(await page.$eval('text=  lo" >> span', e => e.outerHTML)).toBe('yay ');
  expect(await page.$eval(' text = lo" >> span', e => e.outerHTML)).toBe('yay ');
  expect(await page.$eval('text=o"wor >> span', e => e.outerHTML)).toBe('yay ');
  expect(await page.$eval(`text=lo'wor >> span`, e => e.outerHTML)).toBe('nay ');
  expect(await page.$eval(`text=o' >> span`, e => e.outerHTML)).toBe('nay ');
  expect(await page.$eval(`text=ello\`wor >> span`, e => e.outerHTML)).toBe('oh ');
  await expect(page.locator(`text=ello\`wor`).locator('span').first()).toHaveText('oh');
  await expect(page.locator(`text=ello\`wor`).locator('span').nth(1)).toHaveText('oh2');
  expect(await page.$(`text='wor >> span`)).toBe(null);
  expect(await page.$(`text=" >> span`)).toBe(null);
  expect(await page.$(`text=\` >> span`)).toBe(null);
});