import $ from 'jquery';
import TaskList from '~/task_list';
import axios from '~/lib/utils/axios_utils';

describe('TaskList', () => {
  let taskList;
  let currentTarget;
  const taskListOptions = {
    selector: '.task-list',
    dataType: 'issue',
    fieldName: 'description',
    lockVersion: 2,
  };
  const createTaskList = () => new TaskList(taskListOptions);

  beforeEach(() => {
    setFixtures(`
      <div class="task-list">
        <div class="js-task-list-container"></div>
      </div>
    `);

    currentTarget = $('<div></div>');
    taskList = createTaskList();
  });

  it('should call init when the class constructed', () => {
    spyOn(TaskList.prototype, 'init').and.callThrough();
    spyOn(TaskList.prototype, 'disable');
    spyOn($.prototype, 'taskList');
    spyOn($.prototype, 'on');

    taskList = createTaskList();
    const $taskListEl = $(taskList.taskListContainerSelector);

    expect(taskList.init).toHaveBeenCalled();
    expect(taskList.disable).toHaveBeenCalled();
    expect($taskListEl.taskList).toHaveBeenCalledWith('enable');
    expect($(document).on).toHaveBeenCalledWith(
      'tasklist:changed',
      taskList.taskListContainerSelector,
      taskList.updateHandler,
    );
  });

  describe('getTaskListTarget', () => {
    it('should return currentTarget from event object if exists', () => {
      const $target = taskList.getTaskListTarget({ currentTarget });

      expect($target).toEqual(currentTarget);
    });

    it('should return element of the taskListContainerSelector', () => {
      const $target = taskList.getTaskListTarget();

      expect($target).toEqual($(taskList.taskListContainerSelector));
    });
  });

  describe('disableTaskListItems', () => {
    it('should call taskList method with disable param', () => {
      spyOn($.prototype, 'taskList');

      taskList.disableTaskListItems({ currentTarget });

      expect(currentTarget.taskList).toHaveBeenCalledWith('disable');
    });
  });

  describe('enableTaskListItems', () => {
    it('should call taskList method with enable param', () => {
      spyOn($.prototype, 'taskList');

      taskList.enableTaskListItems({ currentTarget });

      expect(currentTarget.taskList).toHaveBeenCalledWith('enable');
    });
  });

  describe('disable', () => {
    it('should disable task list items and off document event', () => {
      spyOn(taskList, 'disableTaskListItems');
      spyOn($.prototype, 'off');

      taskList.disable();

      expect(taskList.disableTaskListItems).toHaveBeenCalled();
      expect($(document).off).toHaveBeenCalledWith(
        'tasklist:changed',
        taskList.taskListContainerSelector,
      );
    });
  });

  describe('update', () => {
    it('should disable task list items and make a patch request then enable them again', done => {
      const response = { data: { lock_version: 3 } };
      spyOn(taskList, 'enableTaskListItems');
      spyOn(taskList, 'disableTaskListItems');
      spyOn(taskList, 'onSuccess');
      spyOn(axios, 'patch').and.returnValue(Promise.resolve(response));

      const value = 'hello world';
      const endpoint = '/foo';
      const target = $(`<input data-update-url="${endpoint}" value="${value}" />`);
      const detail = {
        index: 2,
        checked: true,
        lineNumber: 8,
        lineSource: '- [ ] check item',
      };
      const event = { target, detail };
      const patchData = {
        [taskListOptions.dataType]: {
          [taskListOptions.fieldName]: value,
          lock_version: taskListOptions.lockVersion,
          update_task: {
            index: detail.index,
            checked: detail.checked,
            line_number: detail.lineNumber,
            line_source: detail.lineSource,
          },
        },
      };

      taskList
        .update(event)
        .then(() => {
          expect(taskList.disableTaskListItems).toHaveBeenCalledWith(event);
          expect(axios.patch).toHaveBeenCalledWith(endpoint, patchData);
          expect(taskList.enableTaskListItems).toHaveBeenCalledWith(event);
          expect(taskList.onSuccess).toHaveBeenCalledWith(response.data);
          expect(taskList.lockVersion).toEqual(response.data.lock_version);
        })
        .then(done)
        .catch(done.fail);
    });
  });

  it('should handle request error and enable task list items', done => {
    const response = { data: { error: 1 } };
    spyOn(taskList, 'enableTaskListItems');
    spyOn(taskList, 'onError');
    spyOn(axios, 'patch').and.returnValue(Promise.reject({ response })); // eslint-disable-line prefer-promise-reject-errors

    const event = { detail: {} };
    taskList
      .update(event)
      .then(() => {
        expect(taskList.enableTaskListItems).toHaveBeenCalledWith(event);
        expect(taskList.onError).toHaveBeenCalledWith(response.data);
      })
      .then(done)
      .catch(done.fail);
  });
});