import {
  editor as monacoEditor,
  languages as monacoLanguages,
  Range,
  Selection,
} from 'monaco-editor';
import Editor from '~/ide/lib/editor';
import { createStore } from '~/ide/stores';
import { defaultEditorOptions } from '~/ide/lib/editor_options';
import { file } from '../helpers';

describe('Multi-file editor library', () => {
  let instance;
  let el;
  let holder;
  let store;

  const setNodeOffsetWidth = val => {
    Object.defineProperty(instance.instance.getDomNode(), 'offsetWidth', {
      get() {
        return val;
      },
    });
  };

  beforeEach(() => {
    store = createStore();
    el = document.createElement('div');
    holder = document.createElement('div');
    el.appendChild(holder);

    document.body.appendChild(el);

    instance = Editor.create(store);
  });

  afterEach(() => {
    instance.modelManager.dispose();
    instance.dispose();
    Editor.editorInstance = null;

    el.remove();
  });

  it('creates instance of editor', () => {
    expect(Editor.editorInstance).not.toBeNull();
  });

  it('creates instance returns cached instance', () => {
    expect(Editor.create(store)).toEqual(instance);
  });

  describe('createInstance', () => {
    it('creates editor instance', () => {
      jest.spyOn(monacoEditor, 'create');

      instance.createInstance(holder);

      expect(monacoEditor.create).toHaveBeenCalled();
    });

    it('creates dirty diff controller', () => {
      instance.createInstance(holder);

      expect(instance.dirtyDiffController).not.toBeNull();
    });

    it('creates model manager', () => {
      instance.createInstance(holder);

      expect(instance.modelManager).not.toBeNull();
    });
  });

  describe('createDiffInstance', () => {
    it('creates editor instance', () => {
      jest.spyOn(monacoEditor, 'createDiffEditor');

      instance.createDiffInstance(holder);

      expect(monacoEditor.createDiffEditor).toHaveBeenCalledWith(holder, {
        ...defaultEditorOptions,
        ignoreTrimWhitespace: false,
        quickSuggestions: false,
        occurrencesHighlight: false,
        renderSideBySide: false,
        readOnly: false,
        renderLineHighlight: 'none',
        hideCursorInOverviewRuler: true,
      });
    });
  });

  describe('createModel', () => {
    it('calls model manager addModel', () => {
      jest.spyOn(instance.modelManager, 'addModel').mockImplementation(() => {});

      instance.createModel('FILE');

      expect(instance.modelManager.addModel).toHaveBeenCalledWith('FILE', null);
    });
  });

  describe('attachModel', () => {
    let model;

    beforeEach(() => {
      instance.createInstance(document.createElement('div'));

      model = instance.createModel(file());
    });

    it('sets the current model on the instance', () => {
      instance.attachModel(model);

      expect(instance.currentModel).toBe(model);
    });

    it('attaches the model to the current instance', () => {
      jest.spyOn(instance.instance, 'setModel').mockImplementation(() => {});

      instance.attachModel(model);

      expect(instance.instance.setModel).toHaveBeenCalledWith(model.getModel());
    });

    it('sets original & modified when diff editor', () => {
      jest.spyOn(instance.instance, 'getEditorType').mockReturnValue('vs.editor.IDiffEditor');
      jest.spyOn(instance.instance, 'setModel').mockImplementation(() => {});

      instance.attachModel(model);

      expect(instance.instance.setModel).toHaveBeenCalledWith({
        original: model.getOriginalModel(),
        modified: model.getModel(),
      });
    });

    it('attaches the model to the dirty diff controller', () => {
      jest.spyOn(instance.dirtyDiffController, 'attachModel').mockImplementation(() => {});

      instance.attachModel(model);

      expect(instance.dirtyDiffController.attachModel).toHaveBeenCalledWith(model);
    });

    it('re-decorates with the dirty diff controller', () => {
      jest.spyOn(instance.dirtyDiffController, 'reDecorate').mockImplementation(() => {});

      instance.attachModel(model);

      expect(instance.dirtyDiffController.reDecorate).toHaveBeenCalledWith(model);
    });
  });

  describe('attachMergeRequestModel', () => {
    let model;

    beforeEach(() => {
      instance.createDiffInstance(document.createElement('div'));

      const f = file();
      f.mrChanges = { diff: 'ABC' };
      f.baseRaw = 'testing';

      model = instance.createModel(f);
    });

    it('sets original & modified', () => {
      jest.spyOn(instance.instance, 'setModel').mockImplementation(() => {});

      instance.attachMergeRequestModel(model);

      expect(instance.instance.setModel).toHaveBeenCalledWith({
        original: model.getBaseModel(),
        modified: model.getModel(),
      });
    });
  });

  describe('clearEditor', () => {
    it('resets the editor model', () => {
      instance.createInstance(document.createElement('div'));

      jest.spyOn(instance.instance, 'setModel').mockImplementation(() => {});

      instance.clearEditor();

      expect(instance.instance.setModel).toHaveBeenCalledWith(null);
    });
  });

  describe('languages', () => {
    it('registers custom languages defined with Monaco', () => {
      expect(monacoLanguages.getLanguages()).toEqual(
        expect.arrayContaining([
          expect.objectContaining({
            id: 'vue',
          }),
        ]),
      );
    });
  });

  describe('replaceSelectedText', () => {
    let model;
    let editor;

    beforeEach(() => {
      instance.createInstance(holder);

      model = instance.createModel({
        ...file(),
        key: 'index.md',
        path: 'index.md',
      });

      instance.attachModel(model);

      editor = instance.instance;
      editor.getModel().setValue('foo bar baz');
      editor.setSelection(new Range(1, 5, 1, 8));

      instance.replaceSelectedText('hello');
    });

    it('replaces the text selected in editor with the one provided', () => {
      expect(editor.getModel().getValue()).toBe('foo hello baz');
    });

    it('sets cursor to end of the replaced string', () => {
      const selection = editor.getSelection();
      expect(selection).toEqual(new Selection(1, 10, 1, 10));
    });
  });

  describe('dispose', () => {
    it('calls disposble dispose method', () => {
      jest.spyOn(instance.disposable, 'dispose');

      instance.dispose();

      expect(instance.disposable.dispose).toHaveBeenCalled();
    });

    it('resets instance', () => {
      instance.createInstance(document.createElement('div'));

      expect(instance.instance).not.toBeNull();

      instance.dispose();

      expect(instance.instance).toBeNull();
    });

    it('does not dispose modelManager', () => {
      jest.spyOn(instance.modelManager, 'dispose').mockImplementation(() => {});

      instance.dispose();

      expect(instance.modelManager.dispose).not.toHaveBeenCalled();
    });

    it('does not dispose decorationsController', () => {
      jest.spyOn(instance.decorationsController, 'dispose').mockImplementation(() => {});

      instance.dispose();

      expect(instance.decorationsController.dispose).not.toHaveBeenCalled();
    });
  });

  describe('updateDiffView', () => {
    describe('edit mode', () => {
      it('does not update options', () => {
        instance.createInstance(holder);

        jest.spyOn(instance.instance, 'updateOptions').mockImplementation(() => {});

        instance.updateDiffView();

        expect(instance.instance.updateOptions).not.toHaveBeenCalled();
      });
    });

    describe('diff mode', () => {
      beforeEach(() => {
        instance.createDiffInstance(holder);

        jest.spyOn(instance.instance, 'updateOptions');
      });

      it('sets renderSideBySide to false if el is less than 700 pixels', () => {
        setNodeOffsetWidth(600);

        expect(instance.instance.updateOptions).not.toHaveBeenCalledWith({
          renderSideBySide: false,
        });
      });

      it('sets renderSideBySide to false if el is more than 700 pixels', () => {
        setNodeOffsetWidth(800);

        expect(instance.instance.updateOptions).not.toHaveBeenCalledWith({
          renderSideBySide: true,
        });
      });
    });
  });

  describe('isDiffEditorType', () => {
    it('returns true when diff editor', () => {
      instance.createDiffInstance(holder);

      expect(instance.isDiffEditorType).toBe(true);
    });

    it('returns false when not diff editor', () => {
      instance.createInstance(holder);

      expect(instance.isDiffEditorType).toBe(false);
    });
  });

  it('sets quickSuggestions to false when language is markdown', () => {
    instance.createInstance(holder);

    jest.spyOn(instance.instance, 'updateOptions');

    const model = instance.createModel({
      ...file(),
      key: 'index.md',
      path: 'index.md',
    });

    instance.attachModel(model);

    expect(instance.instance.updateOptions).toHaveBeenCalledWith({
      readOnly: false,
      quickSuggestions: false,
    });
  });
});