| asynctest('browser.tinymce.core.CaretUtilTest',
  [
    'ephox.mcagar.api.LegacyUnit',
    'ephox.agar.api.Pipeline',
    'tinymce.core.dom.DOMUtils',
    'tinymce.core.Env',
    'tinymce.core.caret.CaretUtils',
    'tinymce.core.caret.CaretPosition',
    'tinymce.core.text.Zwsp',
    'tinymce.core.dom.DomQuery',
    'tinymce.core.test.CaretAsserts',
    'tinymce.core.test.ViewBlock',
    'global!document'
  ],
  function (LegacyUnit, Pipeline, DOMUtils, Env, CaretUtils, CaretPosition, Zwsp, $, CaretAsserts, ViewBlock, document) {
    var success = arguments[arguments.length - 2];
    var failure = arguments[arguments.length - 1];
    var suite = LegacyUnit.createSuite();
    var assertRange = CaretAsserts.assertRange;
    var createRange = CaretAsserts.createRange;
    var viewBlock = new ViewBlock();
    if (!Env.ceFalse) {
      return;
    }
    var ZWSP = Zwsp.ZWSP;
    var getRoot = function () {
      return viewBlock.get();
    };
    var setupHtml = function (html) {
      viewBlock.update(html);
      // IE messes zwsp up on innerHTML so we need to first set markers then replace then using dom operations
      viewBlock.get().innerHTML = html.replace(new RegExp(ZWSP, 'g'), '__ZWSP__');
      replaceWithZwsp(viewBlock.get());
    };
    var replaceWithZwsp = function (node) {
      for (var i = 0; i < node.childNodes.length; i++) {
        var childNode = node.childNodes[i];
        if (childNode.nodeType === 3) {
          childNode.nodeValue = childNode.nodeValue.replace(/__ZWSP__/, ZWSP);
        } else {
          replaceWithZwsp(childNode);
        }
      }
    };
    var findElm = function (selector) {
      return $(selector, getRoot())[0];
    };
    suite.test('isForwards', function () {
      LegacyUnit.equal(CaretUtils.isForwards(1), true);
      LegacyUnit.equal(CaretUtils.isForwards(10), true);
      LegacyUnit.equal(CaretUtils.isForwards(0), false);
      LegacyUnit.equal(CaretUtils.isForwards(-1), false);
      LegacyUnit.equal(CaretUtils.isForwards(-10), false);
    });
    suite.test('isBackwards', function () {
      LegacyUnit.equal(CaretUtils.isBackwards(1), false);
      LegacyUnit.equal(CaretUtils.isBackwards(10), false);
      LegacyUnit.equal(CaretUtils.isBackwards(0), false);
      LegacyUnit.equal(CaretUtils.isBackwards(-1), true);
      LegacyUnit.equal(CaretUtils.isBackwards(-10), true);
    });
    suite.test('findNode', function () {
      setupHtml('<b>abc</b><b><i>123</i></b>def');
      var isBold = function (node) {
        return node.nodeName === 'B';
      };
      var isText = function (node) {
        return node.nodeType === 3;
      };
      LegacyUnit.equalDom(CaretUtils.findNode(getRoot(), 1, isBold, getRoot()), getRoot().firstChild);
      LegacyUnit.equalDom(CaretUtils.findNode(getRoot(), 1, isText, getRoot()), getRoot().firstChild.firstChild);
      LegacyUnit.equal(CaretUtils.findNode(getRoot().childNodes[1], 1, isBold, getRoot().childNodes[1]) === null, true);
      LegacyUnit.equal(CaretUtils.findNode(getRoot().childNodes[1], 1, isText, getRoot().childNodes[1]).nodeName, '#text');
      LegacyUnit.equalDom(CaretUtils.findNode(getRoot(), -1, isBold, getRoot()), getRoot().childNodes[1]);
      LegacyUnit.equalDom(CaretUtils.findNode(getRoot(), -1, isText, getRoot()), getRoot().lastChild);
    });
    suite.test('getEditingHost', function () {
      setupHtml('<span contentEditable="true"><span contentEditable="false"></span></span>');
      LegacyUnit.equalDom(CaretUtils.getEditingHost(getRoot(), getRoot()), getRoot());
      LegacyUnit.equalDom(CaretUtils.getEditingHost(getRoot().firstChild, getRoot()), getRoot());
      LegacyUnit.equalDom(CaretUtils.getEditingHost(getRoot().firstChild.firstChild, getRoot()), getRoot().firstChild);
    });
    suite.test('getParentBlock', function () {
      setupHtml('<p>abc</p><div><p><table><tr><td>X</td></tr></p></div>');
      LegacyUnit.equalDom(CaretUtils.getParentBlock(findElm('p:first')), findElm('p:first'));
      LegacyUnit.equalDom(CaretUtils.getParentBlock(findElm('td:first').firstChild), findElm('td:first'));
      LegacyUnit.equalDom(CaretUtils.getParentBlock(findElm('td:first')), findElm('td:first'));
      LegacyUnit.equalDom(CaretUtils.getParentBlock(findElm('table')), findElm('table'));
    });
    suite.test('isInSameBlock', function () {
      setupHtml('<p>abc</p><p>def<b>ghj</b></p>');
      LegacyUnit.strictEqual(CaretUtils.isInSameBlock(
        CaretPosition(findElm('p:first').firstChild, 0),
        CaretPosition(findElm('p:last').firstChild, 0)
      ), false);
      LegacyUnit.strictEqual(CaretUtils.isInSameBlock(
        CaretPosition(findElm('p:first').firstChild, 0),
        CaretPosition(findElm('p:first').firstChild, 0)
      ), true);
      LegacyUnit.strictEqual(CaretUtils.isInSameBlock(
        CaretPosition(findElm('p:last').firstChild, 0),
        CaretPosition(findElm('b').firstChild, 0)
      ), true);
    });
    suite.test('isInSameEditingHost', function () {
      setupHtml(
        '<p>abc</p>' +
        'def' +
        '<span contentEditable="false">' +
        '<span contentEditable="true">ghi</span>' +
        '<span contentEditable="true">jkl</span>' +
        '</span>'
      );
      LegacyUnit.strictEqual(CaretUtils.isInSameEditingHost(
        CaretPosition(findElm('p:first').firstChild, 0),
        CaretPosition(findElm('p:first').firstChild, 1)
      ), true);
      LegacyUnit.strictEqual(CaretUtils.isInSameEditingHost(
        CaretPosition(findElm('p:first').firstChild, 0),
        CaretPosition(getRoot().childNodes[1], 1)
      ), true);
      LegacyUnit.strictEqual(CaretUtils.isInSameEditingHost(
        CaretPosition(findElm('span span:first').firstChild, 0),
        CaretPosition(findElm('span span:first').firstChild, 1)
      ), true);
      LegacyUnit.strictEqual(CaretUtils.isInSameEditingHost(
        CaretPosition(findElm('p:first').firstChild, 0),
        CaretPosition(findElm('span span:first').firstChild, 1)
      ), false);
      LegacyUnit.strictEqual(CaretUtils.isInSameEditingHost(
        CaretPosition(findElm('span span:first').firstChild, 0),
        CaretPosition(findElm('span span:last').firstChild, 1)
      ), false);
    });
    suite.test('isBeforeContentEditableFalse', function () {
      setupHtml(
        '<span contentEditable="false"></span>' +
        '<span contentEditable="false"></span>a'
      );
      LegacyUnit.strictEqual(CaretUtils.isBeforeContentEditableFalse(CaretPosition(getRoot(), 0)), true);
      LegacyUnit.strictEqual(CaretUtils.isBeforeContentEditableFalse(CaretPosition(getRoot(), 1)), true);
      LegacyUnit.strictEqual(CaretUtils.isBeforeContentEditableFalse(CaretPosition(getRoot(), 2)), false);
      LegacyUnit.strictEqual(CaretUtils.isBeforeContentEditableFalse(CaretPosition(getRoot(), 3)), false);
    });
    suite.test('isAfterContentEditableFalse', function () {
      setupHtml(
        '<span contentEditable="false"></span>' +
        '<span contentEditable="false"></span>a'
      );
      LegacyUnit.strictEqual(CaretUtils.isAfterContentEditableFalse(CaretPosition(getRoot(), 0)), false);
      LegacyUnit.strictEqual(CaretUtils.isAfterContentEditableFalse(CaretPosition(getRoot(), 1)), true);
      LegacyUnit.strictEqual(CaretUtils.isAfterContentEditableFalse(CaretPosition(getRoot(), 2)), true);
      LegacyUnit.strictEqual(CaretUtils.isAfterContentEditableFalse(CaretPosition(getRoot(), 3)), false);
    });
    suite.test('normalizeRange', function () {
      setupHtml(
        'abc<span contentEditable="false">1</span>def'
      );
      assertRange(CaretUtils.normalizeRange(1, getRoot(), createRange(getRoot().firstChild, 2)), createRange(getRoot().firstChild, 2));
      assertRange(CaretUtils.normalizeRange(1, getRoot(), createRange(getRoot().firstChild, 3)), createRange(getRoot(), 1));
      assertRange(CaretUtils.normalizeRange(1, getRoot(), createRange(getRoot().lastChild, 2)), createRange(getRoot().lastChild, 2));
      assertRange(CaretUtils.normalizeRange(1, getRoot(), createRange(getRoot().lastChild, 0)), createRange(getRoot(), 2));
    });
    suite.test('normalizeRange deep', function () {
      setupHtml(
        '<i><b>abc</b></i><span contentEditable="false">1</span><i><b>def</b></i>'
      );
      assertRange(
        CaretUtils.normalizeRange(1, getRoot(), createRange(findElm('b').firstChild, 2)),
        createRange(findElm('b').firstChild, 2)
      );
      assertRange(CaretUtils.normalizeRange(1, getRoot(), createRange(findElm('b').firstChild, 3)), createRange(getRoot(), 1));
      assertRange(
        CaretUtils.normalizeRange(-1, getRoot(), createRange(findElm('b:last').firstChild, 1)),
        createRange(findElm('b:last').firstChild, 1)
      );
      assertRange(CaretUtils.normalizeRange(-1, getRoot(), createRange(findElm('b:last').firstChild, 0)), createRange(getRoot(), 2));
    });
    suite.test('normalizeRange break at candidate', function () {
      setupHtml(
        '<p><b>abc</b><input></p><p contentEditable="false">1</p><p><input><b>abc</b></p>'
      );
      assertRange(
        CaretUtils.normalizeRange(1, getRoot(), createRange(findElm('b').firstChild, 3)),
        createRange(findElm('b').firstChild, 3)
      );
      assertRange(
        CaretUtils.normalizeRange(1, getRoot(), createRange(findElm('b:last').lastChild, 0)),
        createRange(findElm('b:last').lastChild, 0)
      );
    });
    suite.test('normalizeRange at block caret container', function () {
      setupHtml(
        '<p data-mce-caret="before">\u00a0</p><p contentEditable="false">1</p><p data-mce-caret="after">\u00a0</p>'
      );
      assertRange(CaretUtils.normalizeRange(1, getRoot(), createRange(findElm('p:first').firstChild, 0)), createRange(getRoot(), 1));
      assertRange(CaretUtils.normalizeRange(1, getRoot(), createRange(findElm('p:first').firstChild, 1)), createRange(getRoot(), 1));
      assertRange(CaretUtils.normalizeRange(-1, getRoot(), createRange(findElm('p:last').firstChild, 0)), createRange(getRoot(), 2));
      assertRange(CaretUtils.normalizeRange(-1, getRoot(), createRange(findElm('p:last').firstChild, 1)), createRange(getRoot(), 2));
    });
    suite.test('normalizeRange at inline caret container', function () {
      setupHtml(
        'abc<span contentEditable="false">1</span>def'
      );
      getRoot().insertBefore(document.createTextNode(ZWSP), getRoot().childNodes[1]);
      getRoot().insertBefore(document.createTextNode(ZWSP), getRoot().childNodes[3]);
      assertRange(CaretUtils.normalizeRange(1, getRoot(), createRange(getRoot().firstChild, 3)), createRange(getRoot(), 2));
      assertRange(CaretUtils.normalizeRange(1, getRoot(), createRange(getRoot().childNodes[1], 0)), createRange(getRoot(), 2));
      assertRange(CaretUtils.normalizeRange(1, getRoot(), createRange(getRoot().childNodes[1], 1)), createRange(getRoot(), 2));
      assertRange(CaretUtils.normalizeRange(1, getRoot(), createRange(getRoot().lastChild, 0)), createRange(getRoot(), 3));
      assertRange(CaretUtils.normalizeRange(1, getRoot(), createRange(getRoot().childNodes[3], 0)), createRange(getRoot(), 3));
      assertRange(CaretUtils.normalizeRange(1, getRoot(), createRange(getRoot().childNodes[3], 1)), createRange(getRoot(), 3));
      assertRange(CaretUtils.normalizeRange(-1, getRoot(), createRange(getRoot().firstChild, 3)), createRange(getRoot(), 2));
      assertRange(CaretUtils.normalizeRange(-1, getRoot(), createRange(getRoot().childNodes[1], 0)), createRange(getRoot(), 2));
      assertRange(CaretUtils.normalizeRange(-1, getRoot(), createRange(getRoot().childNodes[1], 1)), createRange(getRoot(), 2));
      assertRange(CaretUtils.normalizeRange(-1, getRoot(), createRange(getRoot().lastChild, 0)), createRange(getRoot(), 3));
      assertRange(CaretUtils.normalizeRange(-1, getRoot(), createRange(getRoot().childNodes[3], 0)), createRange(getRoot(), 3));
      assertRange(CaretUtils.normalizeRange(-1, getRoot(), createRange(getRoot().childNodes[3], 1)), createRange(getRoot(), 3));
    });
    suite.test('normalizeRange at inline caret container (combined)', function () {
      setupHtml(
        'abc' + ZWSP + '<span contentEditable="false">1</span>' + ZWSP + 'def'
      );
      assertRange(CaretUtils.normalizeRange(1, getRoot(), createRange(getRoot().firstChild, 3)), createRange(getRoot(), 1));
      assertRange(CaretUtils.normalizeRange(1, getRoot(), createRange(getRoot().firstChild, 4)), createRange(getRoot(), 1));
      assertRange(CaretUtils.normalizeRange(-1, getRoot(), createRange(getRoot().lastChild, 0)), createRange(getRoot(), 2));
      assertRange(CaretUtils.normalizeRange(-1, getRoot(), createRange(getRoot().lastChild, 1)), createRange(getRoot(), 2));
    });
    suite.test('normalizeRange at inline caret container after block', function () {
      setupHtml(
        '<p><span contentEditable="false">1</span></p>' + ZWSP + 'abc'
      );
      assertRange(CaretUtils.normalizeRange(1, getRoot(), createRange(getRoot().lastChild, 0)), createRange(getRoot().lastChild, 0));
    });
    viewBlock.attach();
    Pipeline.async({}, suite.toSteps({}), function () {
      viewBlock.detach();
      success();
    }, failure);
  }
);
 |