import * as path from 'path';
import * as assert from 'assert';

import TestServer from './lib/TestServer';
import eventually from './eventually';
import { Builder, Lanthan } from 'lanthan';
import { WebDriver, Key } from 'selenium-webdriver';
import Page from './lib/Page';

describe("tab test", () => {
  const server = new TestServer().receiveContent('/*', 'ok');
  let lanthan: Lanthan;
  let webdriver: WebDriver;
  let browser: any;
  let win: any;
  let tabs: any[];

  before(async() => {
    lanthan = await Builder
      .forBrowser('firefox')
      .spyAddon(path.join(__dirname, '..'))
      .build();
    webdriver = lanthan.getWebDriver();
    browser = lanthan.getWebExtBrowser();
    await server.start();
  });

  after(async() => {
    await server.stop();
    if (lanthan) {
      await lanthan.quit();
    }
  });

  beforeEach(async() => {
    win = await browser.windows.create({ url: server.url('/#0') });
    for (let i = 1; i < 5; ++i) {
      await browser.tabs.create({ url: server.url('/#' + i), windowId: win.id });
      await webdriver.navigate().to(server.url('/#' + i));
    }
    tabs = await browser.tabs.query({ windowId: win.id });
    tabs.sort((t1: any, t2: any) => t1.index - t2.index);
  });

  afterEach(async() => {
    await browser.windows.remove(win.id);
  });

  it('deletes tab and selects right by d', async () => {
    await browser.tabs.update(tabs[3].id, { active: true });
    const page = await Page.currentContext(webdriver);
    await page.sendKeys('d');

    await eventually(async() => {
      const current = await browser.tabs.query({ windowId: win.id });
      assert.strictEqual(current.length, tabs.length - 1);
      assert.strictEqual(current[3].active, true);
      assert.strictEqual(current[3].id, tabs[4].id);
    });
  });

  it('deletes tab and selects left by D', async () => {
    await browser.tabs.update(tabs[3].id, { active: true });
    const page = await Page.currentContext(webdriver);
    await page.sendKeys(Key.SHIFT, 'D');

    await eventually(async() => {
      const current = await browser.tabs.query({ windowId: win.id });
      assert.strictEqual(current.length, tabs.length - 1);
      assert.strictEqual(current[2].active, true);
      assert.strictEqual(current[2].id, tabs[2].id);
    })
  });

  it('deletes all tabs to the right by x$', async () => {
    await browser.tabs.update(tabs[1].id, { active: true });
    const page = await Page.currentContext(webdriver);
    await page.sendKeys('x', '$');

    const current = await browser.tabs.query({ windowId: win.id });
    assert.strictEqual(current.length, 2);
  });

  it('duplicates tab by zd', async () => {
    await browser.tabs.update(tabs[0].id, { active: true });
    const page = await Page.currentContext(webdriver);
    await page.sendKeys('z', 'd');

    await eventually(async() => {
      const current = await browser.tabs.query({ windowId: win.id });
      current.sort((t1: any, t2: any) => t1.index - t2.index);
      assert.strictEqual(current.length, tabs.length + 1);
      assert.strictEqual(current[0].url, current[1].url);
    });
  });

  it('makes pinned by zp', async () => {
    await browser.tabs.update(tabs[0].id, { active: true });
    const page = await Page.currentContext(webdriver);
    await page.sendKeys('z', 'p');

    const current = await browser.tabs.query({ windowId: win.id });
    assert.strictEqual(current[0].pinned, true);
  });

  it('selects previous tab by K', async () => {
    await browser.tabs.update(tabs[2].id, { active: true });
    const page = await Page.currentContext(webdriver);
    await page.sendKeys(Key.SHIFT, 'K');

    const current = await browser.tabs.query({ windowId: win.id });
    assert.strictEqual(current[1].active, true);
  });

  it('selects previous tab by K rotatory', async () => {
    await browser.tabs.update(tabs[0].id, { active: true });
    const page = await Page.currentContext(webdriver);
    await page.sendKeys(Key.SHIFT, 'K');

    const current = await browser.tabs.query({ windowId: win.id });
    assert.strictEqual(current[current.length - 1].active, true)
  });

  it('selects next tab by J', async () => {
    await browser.tabs.update(tabs[2].id, { active: true });
    const page = await Page.currentContext(webdriver);
    await page.sendKeys(Key.SHIFT, 'J');

    const current = await browser.tabs.query({ windowId: win.id });
    assert.strictEqual(current[3].active, true);
  });

  it('selects previous tab by J rotatory', async () => {
    await browser.tabs.update(tabs[tabs.length - 1].id, { active: true });
    const page = await Page.currentContext(webdriver);
    await page.sendKeys(Key.SHIFT, 'J');

    const current = await browser.tabs.query({ windowId: win.id });
    assert.strictEqual(current[0].active, true)
  });

  it('selects first tab by g0', async () => {
    await browser.tabs.update(tabs[2].id, { active: true });
    const page = await Page.currentContext(webdriver);
    await page.sendKeys('g', '0');

    const current = await browser.tabs.query({ windowId: win.id });
    assert.strictEqual(current[0].active, true)
  });

  it('selects last tab by g$', async () => {
    await browser.tabs.update(tabs[2].id, { active: true });
    const page = await Page.currentContext(webdriver);
    await page.sendKeys('g', '$');

    const current = await browser.tabs.query({ windowId: win.id });
    assert.strictEqual(current[current.length - 1].active, true)
  });

  it('selects last selected tab by <C-6>', async () => {
    await browser.tabs.update(tabs[1].id, { active: true });
    await browser.tabs.update(tabs[4].id, { active: true });

    const page = await Page.currentContext(webdriver);
    await page.sendKeys(Key.CONTROL, '6');

    const current = await browser.tabs.query({ windowId: win.id });
    assert.strictEqual(current[1].active, true)
  });

  // browser.sessions.getRecentlyClosed() sometime throws "An unexpected error occurred"
  // This might be a bug in Firefox.
  it.skip('reopen tab by u', async () => {
    await browser.tabs.remove(tabs[1].id);
    const page = await Page.currentContext(webdriver);
    await page.sendKeys('u');

    const current = await browser.tabs.query({ windowId: win.id });
    assert.strictEqual(current.length, tabs.length);
  });

  it('does not delete pinned tab by d', async () => {
    await browser.tabs.update(tabs[0].id, { active: true, pinned: true });
    const page = await Page.currentContext(webdriver);
    await page.sendKeys('d');

    const current = await browser.tabs.query({ windowId: win.id });
    assert.strictEqual(current.length, tabs.length);
  });

  it('deletes pinned tab by !d', async () => {
    await browser.tabs.update(tabs[0].id, { active: true, pinned: true });
    const page = await Page.currentContext(webdriver);
    await page.sendKeys('!', 'd');

    const current = await browser.tabs.query({ windowId: win.id });
    assert.strictEqual(current.length, tabs.length - 1);
  });

  it('opens view-source by gf', async () => {
    await browser.tabs.update(tabs[0].id, { active: true });
    const page = await Page.currentContext(webdriver);
    await page.sendKeys('g', 'f');

    await eventually(async() => {
      const current = await browser.tabs.query({ windowId: win.id });
      assert.strictEqual(current.length, tabs.length + 1);
      assert.strictEqual(current[current.length - 1].url, `view-source:${server.url('/#0')}`);
    });
  });
});