| asynctest(
  'browser.tinymce.core.dom.ScrollIntoViewTest',
  [
    'ephox.agar.api.Assertions',
    'ephox.agar.api.GeneralSteps',
    'ephox.agar.api.Logger',
    'ephox.agar.api.Pipeline',
    'ephox.agar.api.Step',
    'ephox.agar.api.Waiter',
    'ephox.katamari.api.Cell',
    'ephox.mcagar.api.TinyApis',
    'ephox.mcagar.api.TinyLoader',
    'ephox.sugar.api.node.Element',
    'global!window',
    'tinymce.core.dom.ScrollIntoView',
    'tinymce.themes.modern.Theme'
  ],
  function (Assertions, GeneralSteps, Logger, Pipeline, Step, Waiter, Cell, TinyApis, TinyLoader, Element, window, ScrollIntoView, Theme) {
    var success = arguments[arguments.length - 2];
    var failure = arguments[arguments.length - 1];
    Theme();
    var sScrollReset = function (editor) {
      return Step.sync(function () {
        editor.getWin().scrollTo(0, 0);
      });
    };
    var sSetContent = function (editor, tinyApis, html) {
      return GeneralSteps.sequence([
        tinyApis.sSetContent(html),
        Waiter.sTryUntil('Wait for scrollHeight to be updated', Step.sync(function () {
          Assertions.assertEq('Scroll body should be more than 100', true, editor.getBody().scrollHeight > 100);
        }), 100, 1000)
      ]);
    };
    var sScrollIntoView = function (editor, selector, alignToTop) {
      return Step.sync(function () {
        editor.selection.scrollIntoView(editor.dom.select(selector)[0], alignToTop);
      });
    };
    var sScrollIntoViewPrivateApi = function (editor, selector, alignToTop) {
      return Step.sync(function () {
        ScrollIntoView.scrollIntoView(editor, editor.dom.select(selector)[0], alignToTop);
      });
    };
    var sAssertScrollPosition = function (editor, x, y) {
      return Step.sync(function () {
        Assertions.assertEq('Scroll position X should be expected value', x, editor.dom.getViewPort(editor.getWin()).x);
        Assertions.assertEq('Scroll position Y should be expected value', y, editor.dom.getViewPort(editor.getWin()).y);
      });
    };
    var mBindScrollIntoViewEvent = function (editor) {
      return Step.stateful(function (value, next, die) {
        var state = Cell({});
        var handler = function (e) {
          e.preventDefault();
          state.set({
            elm: e.elm,
            alignToTop: e.alignToTop
          });
        };
        editor.on('ScrollIntoView', handler);
        next({
          handler: handler,
          state: state
        });
      });
    };
    var mAssertScrollIntoViewEventInfo = function (editor, expectedElementSelector, expectedAlignToTop) {
      return Step.stateful(function (value, next, die) {
        var expectedTarget = Element.fromDom(editor.dom.select(expectedElementSelector)[0]);
        var actualTarget = Element.fromDom(value.state.get().elm);
        Assertions.assertDomEq('Target should be expected element', expectedTarget, actualTarget);
        Assertions.assertEq('Align to top should be expected value', expectedAlignToTop, value.state.get().alignToTop);
        editor.off('ScrollIntoView', value.handler);
        next({});
      });
    };
    var steps = function (editor, tinyApis) {
      return [
        tinyApis.sFocus,
        Logger.t('Public Selection API', GeneralSteps.sequence([
          Logger.t('Scroll to element align to bottom', GeneralSteps.sequence([
            sScrollReset(editor),
            sSetContent(editor, tinyApis, '<div style="height: 1000px">a</div><div style="height: 50px">b</div><div style="height: 1000px">a</div>'),
            sScrollIntoView(editor, 'div:nth-child(2)', false),
            sAssertScrollPosition(editor, 0, 975)
          ])),
          Logger.t('Scroll to element align to top', GeneralSteps.sequence([
            sScrollReset(editor),
            sSetContent(editor, tinyApis, '<div style="height: 1000px">a</div><div style="height: 50px">b</div><div style="height: 1000px">a</div>'),
            sScrollIntoView(editor, 'div:nth-child(2)', true),
            sAssertScrollPosition(editor, 0, 925)
          ]))
        ])),
        Logger.t('Private ScrollIntoView API', GeneralSteps.sequence([
          Logger.t('Scroll to element align to bottom', GeneralSteps.sequence([
            sScrollReset(editor),
            sSetContent(editor, tinyApis, '<div style="height: 1000px">a</div><div style="height: 50px">b</div><div style="height: 1000px">a</div>'),
            sScrollIntoViewPrivateApi(editor, 'div:nth-child(2)', false),
            sAssertScrollPosition(editor, 0, 975)
          ])),
          Logger.t('Scroll to element align to top', GeneralSteps.sequence([
            sScrollReset(editor),
            sSetContent(editor, tinyApis, '<div style="height: 1000px">a</div><div style="height: 50px">b</div><div style="height: 1000px">a</div>'),
            sScrollIntoViewPrivateApi(editor, 'div:nth-child(2)', true),
            sAssertScrollPosition(editor, 0, 925)
          ]))
        ])),
        Logger.t('Override scrollIntoView event', GeneralSteps.sequence([
          Logger.t('Scroll to element align to bottom', GeneralSteps.sequence([
            sScrollReset(editor),
            sSetContent(editor, tinyApis, '<div style="height: 1000px">a</div><div style="height: 50px">b</div><div style="height: 1000px">a</div>'),
            mBindScrollIntoViewEvent(editor),
            sScrollIntoView(editor, 'div:nth-child(2)', false),
            mAssertScrollIntoViewEventInfo(editor, 'div:nth-child(2)', false),
            sAssertScrollPosition(editor, 0, 0)
          ])),
          Logger.t('Scroll to element align to top', GeneralSteps.sequence([
            sScrollReset(editor),
            sSetContent(editor, tinyApis, '<div style="height: 1000px">a</div><div style="height: 50px">b</div><div style="height: 1000px">a</div>'),
            mBindScrollIntoViewEvent(editor),
            sScrollIntoView(editor, 'div:nth-child(2)', true),
            mAssertScrollIntoViewEventInfo(editor, 'div:nth-child(2)', true),
            sAssertScrollPosition(editor, 0, 0)
          ])),
          Logger.t('Scroll to element align to bottom (private api)', GeneralSteps.sequence([
            sScrollReset(editor),
            sSetContent(editor, tinyApis, '<div style="height: 1000px">a</div><div style="height: 50px">b</div><div style="height: 1000px">a</div>'),
            mBindScrollIntoViewEvent(editor),
            sScrollIntoViewPrivateApi(editor, 'div:nth-child(2)', false),
            mAssertScrollIntoViewEventInfo(editor, 'div:nth-child(2)', false),
            sAssertScrollPosition(editor, 0, 0)
          ])),
          Logger.t('Scroll to element align to top (private api)', GeneralSteps.sequence([
            sScrollReset(editor),
            sSetContent(editor, tinyApis, '<div style="height: 1000px">a</div><div style="height: 50px">b</div><div style="height: 1000px">a</div>'),
            mBindScrollIntoViewEvent(editor),
            sScrollIntoViewPrivateApi(editor, 'div:nth-child(2)', true),
            mAssertScrollIntoViewEventInfo(editor, 'div:nth-child(2)', true),
            sAssertScrollPosition(editor, 0, 0)
          ]))
        ]))
      ];
    };
    var isPhantomJs = function () {
      return /PhantomJS/.test(window.navigator.userAgent);
    };
    TinyLoader.setup(function (editor, onSuccess, onFailure) {
      var tinyApis = TinyApis(editor);
      // Only run scrolling tests on real browsers doesn't seem to work on phantomjs for some reason
      Pipeline.async({}, isPhantomJs() ? [ ] : steps(editor, tinyApis), onSuccess, onFailure);
    }, {
      add_unload_trigger: false,
      skin_url: '/project/src/skins/lightgray/dist/lightgray',
      content_style: 'body.mce-content-body  { margin: 0 }'
    }, success, failure);
  }
);
 |