import {assert, driver, WebElement, WebElementPromise} from 'mocha-webdriver';
import * as gu from 'test/nbrowser/gristUtils';
import {setupTestSuite} from 'test/nbrowser/testUtils';


describe('ActionLog', function() {
  this.timeout(20000);
  const cleanup = setupTestSuite();

  afterEach(() => gu.checkForErrors());

  async function getActionUndoState(limit: number): Promise<string[]> {
    const state = await driver.findAll('.action_log .action_log_item', (el) => el.getAttribute('class'));
    return state.slice(0, limit).map((s) => s.replace(/action_log_item/, '').trim());
  }

  function getActionLogItems(): Promise<WebElement[]> {
    // Use a fancy negation of style selector to exclude hidden log entries.
    return driver.findAll(".action_log .action_log_item:not([style*='display: none'])");
  }

  function getActionLogItem(index: number): WebElementPromise {
    return new WebElementPromise(driver, getActionLogItems().then((elems) => elems[index]));
  }

  before(async function() {
    const session = await gu.session().user('user1').login();
    await session.tempDoc(cleanup, 'Hello.grist');
    await gu.dismissWelcomeTourIfNeeded();
  });

  after(async function() {
    // If were are debugging the browser won't be reloaded, so we need to close the right panel.
    if (process.env.NO_CLEANUP) {
      await driver.find(".test-right-tool-close").click();
    }
  });

  it("should cross out undone actions", async function() {
    // Open the action-log tab.
    await driver.findWait('.test-tools-log', 1000).click();
    await gu.waitToPass(() =>   // Click might not work while panel is sliding out to open.
      driver.findContentWait('.test-doc-history-tabs .test-select-button', 'Activity', 500).click());

    // Perform some actions and check that they all appear as default.
    await gu.enterGridRows({rowNum: 1, col: 0}, [['a'], ['b'], ['c'], ['d']]);

    assert.deepEqual(await getActionUndoState(4), ['default', 'default', 'default', 'default']);

    // Undo and check that the most recent action is crossed out.
    await gu.undo();
    assert.deepEqual(await getActionUndoState(4), ['undone', 'default', 'default', 'default']);

    await gu.undo(2);
    assert.deepEqual(await getActionUndoState(4), ['undone', 'undone', 'undone', 'default']);
    await gu.redo(2);
    assert.deepEqual(await getActionUndoState(4), ['undone', 'default', 'default', 'default']);
  });

  it("should indicate that actions that cannot be redone are buried", async function() {
    // Add an item after the undo actions and check that they get buried.
    await gu.getCell({rowNum: 1, col: 0}).click();
    await gu.enterCell('e');
    assert.deepEqual(await getActionUndoState(4), ['default', 'buried', 'default', 'default']);

    // Check that undos skip the buried actions.
    await gu.undo(2);
    assert.deepEqual(await getActionUndoState(4), ['undone', 'buried', 'undone', 'default']);

    // Check that burying around already buried actions works.
    await gu.enterCell('f');
    await gu.waitForServer();
    assert.deepEqual(await getActionUndoState(5), ['default', 'buried', 'buried', 'buried', 'default']);
  });

  it("should properly rebuild the action log on refresh", async function() {
    // Undo past buried actions to add complexity to the current state of the log
    // and refresh.
    await gu.undo(2);
    await driver.navigate().refresh();
    await gu.waitForDocToLoad();
    // refreshing browser will restore position on last cell
    // switch active cell to the first cell in the first row
    await gu.getCell(0, 1).click();
    await driver.findWait('.test-tools-log', 1000).click();
    await driver.findContentWait('.test-doc-history-tabs .test-select-button', 'Activity', 500).click();
    await gu.waitForServer();
    assert.deepEqual(await getActionUndoState(6), ['undone', 'buried', 'buried', 'buried', 'undone', 'default']);
  });

  it("should indicate to the user when they cannot undo or redo", async function() {
    assert.equal(await driver.find('.test-undo').matches('[class*=-disabled]'), false);
    assert.equal(await driver.find('.test-redo').matches('[class*=-disabled]'), false);

    // Undo and check that undo button gets disabled.
    await gu.undo();
    assert.equal(await driver.find('.test-undo').matches('[class*=-disabled]'), true);
    assert.equal(await driver.find('.test-redo').matches('[class*=-disabled]'), false);

    // Redo to the top of the log and check that redo button gets disabled.
    await gu.redo(3);
    assert.equal(await driver.find('.test-undo').matches('[class*=-disabled]'), false);
    assert.equal(await driver.find('.test-redo').matches('[class*=-disabled]'), true);
  });

  it("should show clickable tabular diffs", async function() {
    const item0 = await getActionLogItem(0);
    assert.equal(await item0.find('table caption').getText(), 'Table1');
    assert.equal(await item0.find('table th:nth-child(2)').getText(), 'A');
    assert.equal(await item0.find('table td:nth-child(2)').getText(), 'f');
    assert.equal(await gu.getActiveCell().getText(), 'a');
    await item0.find('table td:nth-child(2)').click();
    assert.equal(await gu.getActiveCell().getText(), 'f');
  });

  it("clickable tabular diffs should work across renames", async function() {
    // Add another table just to mix things up a bit.
    await gu.addNewTable();
    // Rename our old table.
    await gu.renameTable('Table1', 'Table1Renamed');
    await gu.getPageItem('Table1Renamed').click();
    await gu.renameColumn({col: 'A'}, 'ARenamed');

    // Check that it's still usable. (It doesn't reflect the new names in the content of prior
    // actions though -- e.g. the action below still mentions 'A' for column name -- and it's
    // unclear if it should.)
    const item2 = await getActionLogItem(2);
    assert.equal(await item2.find('table caption').getText(), 'Table1');
    assert.equal(await item2.find('table td:nth-child(2)').getText(), 'f');
    await gu.getCell({rowNum: 1, col: 0}).click();
    assert.notEqual(await gu.getActiveCell().getText(), 'f');
    await item2.find('table td:nth-child(2)').click();
    assert.equal(await gu.getActiveCell().getText(), 'f');

    // Delete Table1Renamed.
    await gu.removeTable('Table1Renamed');
    await driver.findContent('.action_log label', /All tables/).find('input').click();

    const item4 = await getActionLogItem(4);
    await gu.scrollIntoView(item4);
    await item4.find('table td:nth-child(2)').click();
    assert.include(await driver.findWait('.test-notifier-toast-wrapper', 1000).getText(),
      'Table1Renamed was subsequently removed');
    await driver.find('.test-notifier-toast-wrapper .test-notifier-toast-close').click();
    await driver.findContent('.action_log label', /All tables/).find('input').click();
  });

  it("should filter cell changes and renames by table", async function() {
    // Have Table2, now add some more
    // We are at Raw Data view now (since we deleted a table).
    assert.match(await driver.getCurrentUrl(), /p\/data$/);
    await gu.getPageItem('Table2').click();
    await gu.enterGridRows({rowNum: 1, col: 0}, [['2']]);
    await gu.addNewTable();  // Table1
    await gu.enterGridRows({rowNum: 1, col: 0}, [['1']]);
    await gu.addNewTable();  // Table3
    await gu.enterGridRows({rowNum: 1, col: 0}, [['3']]);
    await gu.getPageItem('Table1').click();

    assert.lengthOf(await getActionLogItems(), 2);

    assert.equal(await getActionLogItem(0).find("table:not([style*='display: none']) caption").getText(), 'Table1');
    assert.equal(await getActionLogItem(1).find('.action_log_rename').getText(), 'Add Table1');
    await gu.renameTable('Table1', 'Table1Renamed');
    assert.equal(await getActionLogItem(0).find('.action_log_rename').getText(),
      'Rename Table1 to Table1Renamed');

    await gu.renameColumn({col: 'A'}, 'ARenamed');
    assert.equal(await getActionLogItem(0).find('.action_log_rename').getText(),
      'Rename Table1Renamed.A to ARenamed');
    await gu.getPageItem('Table2').click();
    assert.equal(await getActionLogItem(0).find("table:not([style*='display: none']) caption").getText(), 'Table2');
    await gu.getPageItem('Table3').click();
    assert.equal(await getActionLogItem(0).find("table:not([style*='display: none']) caption").getText(), 'Table3');

      // Now show all tables and make sure the result is a longer (visible) log.
    const filteredCount = (await getActionLogItems()).length;
    await driver.findContent('.action_log label', /All tables/).find('input').click();
    const fullCount = (await getActionLogItems()).length;
    assert.isAbove(fullCount, filteredCount);
  });
});