import { ApproxStructure, Assertions, GeneralSteps, Logger, Pipeline, Step } from '@ephox/agar'; import { TinyApis, TinyLoader } from '@ephox/mcagar'; import { Element } from '@ephox/sugar'; import * as CaretFormat from 'tinymce/core/fmt/CaretFormat'; import TypeText from '../../module/test/TypeText'; import Zwsp from 'tinymce/core/text/Zwsp'; import ModernTheme from 'tinymce/themes/modern/Theme'; import { UnitTest } from '@ephox/bedrock'; import { isCaretNode, getParentCaretContainer } from 'tinymce/core/fmt/FormatContainer'; UnitTest.asynctest('browser.tinymce.core.fmt.CaretFormatTest', function () { const success = arguments[arguments.length - 2]; const failure = arguments[arguments.length - 1]; ModernTheme(); const sApplyCaretFormat = function (editor, name, vars) { return Step.sync(function () { CaretFormat.applyCaretFormat(editor, name, vars); }); }; const sRemoveCaretFormat = function (editor, name, vars, similar?) { return Step.sync(function () { CaretFormat.removeCaretFormat(editor, name, vars, similar); }); }; const sAssertNormalizedContentStructure = function (editor, expected) { return Step.sync(function () { const rawBody = editor.getBody().cloneNode(true); rawBody.normalize(); Assertions.assertStructure( 'Asserting the normalized structure of tiny content.', expected, Element.fromDom(rawBody) ); }); }; TinyLoader.setup(function (editor, onSuccess, onFailure) { const tinyApis = TinyApis(editor); Pipeline.async({}, [ tinyApis.sFocus, Logger.t('Apply bold to caret and type bold text after the unformatted text', GeneralSteps.sequence([ tinyApis.sSetContent('

a

'), tinyApis.sSetCursor([0, 0], 1), sApplyCaretFormat(editor, 'bold', {}), TypeText.sTypeContentAtSelection(Element.fromDom(editor.getDoc()), 'x'), tinyApis.sAssertContent('

ax

'), tinyApis.sAssertContentStructure(ApproxStructure.build(function (s, str) { return s.element('body', { children: [ s.element('p', { children: [ s.text(str.is('a')), s.element('span', { attrs: { 'id': str.is('_mce_caret'), 'data-mce-bogus': str.is('1') }, children: [ s.element('strong', { children: [ s.text(str.is(Zwsp.ZWSP + 'x')) ] }) ] }) ] }) ] }); })), tinyApis.sAssertSelection([0, 1, 0, 0], 2, [0, 1, 0, 0], 2) ])), Logger.t('Apply bold to caret in middle of a word', GeneralSteps.sequence([ tinyApis.sSetContent('

ab

'), tinyApis.sSetCursor([0, 0], 1), sApplyCaretFormat(editor, 'bold', {}), tinyApis.sAssertContent('

ab

'), tinyApis.sAssertContentStructure(ApproxStructure.build(function (s, str) { return s.element('body', { children: [ s.element('p', { children: [ s.element('strong', { children: [ s.text(str.is('ab')) ] }) ] }) ] }); })), tinyApis.sAssertSelection([0, 0, 0], 1, [0, 0, 0], 1) ])), Logger.t('Remove bold from caret and type after the bold text', GeneralSteps.sequence([ tinyApis.sSetContent('

a

'), tinyApis.sSetCursor([0, 0, 0], 1), sRemoveCaretFormat(editor, 'bold', {}), TypeText.sTypeContentAtSelection(Element.fromDom(editor.getDoc()), 'x'), tinyApis.sAssertContent('

ax

'), tinyApis.sAssertContentStructure(ApproxStructure.build(function (s, str) { return s.element('body', { children: [ s.element('p', { children: [ s.element('strong', { children: [ s.text(str.is('a')) ] }), s.element('span', { attrs: { 'id': str.is('_mce_caret'), 'data-mce-bogus': str.is('1') }, children: [ s.text(str.is(Zwsp.ZWSP + 'x')) ] }) ] }) ] }); })), tinyApis.sAssertSelection([0, 1, 0], 2, [0, 1, 0], 2) ])), Logger.t('Remove bold from caret in the middle of a bold word', GeneralSteps.sequence([ tinyApis.sSetContent('

ab

'), tinyApis.sSetCursor([0, 0, 0], 1), sRemoveCaretFormat(editor, 'bold', {}), tinyApis.sAssertContent('

ab

'), tinyApis.sAssertContentStructure(ApproxStructure.build(function (s, str) { return s.element('body', { children: [ s.element('p', { children: [ s.text(str.is('ab')) ] }) ] }); })), tinyApis.sAssertSelection([0, 0], 1, [0, 0], 1) ])), Logger.t('Toggle bold format on and off and type after unformatted text', GeneralSteps.sequence([ tinyApis.sSetContent('

a

'), tinyApis.sSetCursor([0, 0], 1), sApplyCaretFormat(editor, 'bold', {}), sRemoveCaretFormat(editor, 'bold', {}), TypeText.sTypeContentAtSelection(Element.fromDom(editor.getDoc()), 'x'), tinyApis.sAssertContent('

ax

'), tinyApis.sAssertContentStructure(ApproxStructure.build(function (s, str) { return s.element('body', { children: [ s.element('p', { children: [ s.text(str.is('a')), s.element('span', { children: [ s.text(str.is(Zwsp.ZWSP + 'x')) ] }) ] }) ] }); })), tinyApis.sAssertSelection([0, 1, 0], 2, [0, 1, 0], 2) ])), Logger.t('Toggle bold format off and on and type after bold text', GeneralSteps.sequence([ tinyApis.sSetContent('

a

'), tinyApis.sSetCursor([0, 0], 1), sRemoveCaretFormat(editor, 'bold', {}), sApplyCaretFormat(editor, 'bold', {}), TypeText.sTypeContentAtSelection(Element.fromDom(editor.getDoc()), 'x'), tinyApis.sAssertContent('

ax

'), tinyApis.sAssertContentStructure(ApproxStructure.build(function (s, str) { return s.element('body', { children: [ s.element('p', { children: [ s.element('strong', { children: [ s.text(str.is('a')) ] }), s.element('span', { attrs: { 'id': str.is('_mce_caret'), 'data-mce-bogus': str.is('1') }, children: [ s.element('strong', { children: [ s.text(str.is(Zwsp.ZWSP + 'x')) ] }) ] }) ] }) ] }); })), tinyApis.sAssertSelection([0, 1, 0, 0], 2, [0, 1, 0, 0], 2) ])), Logger.t('Apply bold format to the end of text and with trailing br', GeneralSteps.sequence([ tinyApis.sSetRawContent('

a

'), tinyApis.sSetCursor([0, 0], 1), sApplyCaretFormat(editor, 'bold', {}), TypeText.sTypeContentAtSelection(Element.fromDom(editor.getDoc()), 'x'), tinyApis.sAssertContent('

ax

'), sAssertNormalizedContentStructure(editor, ApproxStructure.build(function (s, str) { return s.element('body', { children: [ s.element('p', { children: [ s.text(str.is('a')), s.element('span', { attrs: { 'id': str.is('_mce_caret'), 'data-mce-bogus': str.is('1') }, children: [ s.element('strong', { children: [ s.text(str.is(Zwsp.ZWSP + 'x')) ] }) ] }), s.element('br', {}) ] }) ] }); })), tinyApis.sAssertSelection([0, 1, 0, 0], 2, [0, 1, 0, 0], 2) ])), Logger.t('Remove bold format from word with trailing br', GeneralSteps.sequence([ tinyApis.sSetRawContent('

a

'), tinyApis.sSetCursor([0, 0, 0], 1), sRemoveCaretFormat(editor, 'bold', {}), TypeText.sTypeContentAtSelection(Element.fromDom(editor.getDoc()), 'x'), tinyApis.sAssertContent('

ax

'), tinyApis.sAssertContentStructure(ApproxStructure.build(function (s, str) { return s.element('body', { children: [ s.element('p', { children: [ s.element('strong', { children: [ s.text(str.is('a')) ] }), s.element('span', { attrs: { 'id': str.is('_mce_caret'), 'data-mce-bogus': str.is('1') }, children: [ s.text(str.is(Zwsp.ZWSP + 'x')) ] }) ] }) ] }); })), tinyApis.sAssertSelection([0, 1, 0], 2, [0, 1, 0], 2) ])), Logger.t('Remove bold format from empty paragraph and move selection', GeneralSteps.sequence([ tinyApis.sSetRawContent('

a


'), tinyApis.sSetCursor([1, 0, 0], 0), sRemoveCaretFormat(editor, 'bold', {}), tinyApis.sAssertContent('

a

\n

 

'), tinyApis.sAssertContentStructure(ApproxStructure.build(function (s, str) { return s.element('body', { children: [ s.element('p', { children: [s.text(str.is('a')) ] }), s.element('p', { children: [ s.element('span', { attrs: { 'id': str.is('_mce_caret'), 'data-mce-bogus': str.is('1') }, children: [ s.element('br', {}) ] }) ] }) ] }); })), tinyApis.sSetCursor([0, 0], 1), TypeText.sTypeContentAtSelection(Element.fromDom(editor.getDoc()), 'x'), tinyApis.sAssertContent('

ax

\n

 

'), tinyApis.sAssertContentStructure(ApproxStructure.build(function (s, str) { return s.element('body', { children: [ s.element('p', { children: [s.text(str.is('ax')) ] }), s.element('p', { children: [ s.element('br', {}) ] }) ] }); })) ])), Logger.t('isCaretNode', Step.sync(function () { Assertions.assertEq('Should be false since it is not a caret node', false, isCaretNode(editor.dom.create('b'))); Assertions.assertEq('Should be true since it is a caret node', true, isCaretNode(editor.dom.create('span', { id: '_mce_caret' }))); })), Logger.t('Apply some format to the empty editor and make sure that the content didn\'t mutate after serialization (TINY-1288)', GeneralSteps.sequence([ tinyApis.sSetContent(''), tinyApis.sSetCursor([0], 0), tinyApis.sExecCommand('fontname', 'Arial'), tinyApis.sAssertContent(''), sAssertNormalizedContentStructure(editor, ApproxStructure.build(function (s, str) { return s.element('body', { children: [ s.element('p', { children: [ s.element('span', { attrs: { 'id': str.is('_mce_caret'), 'data-mce-bogus': str.is('1') }, children: [ s.element('span', { attrs: { 'style': str.is('font-family: Arial;'), 'data-mce-bogus': str.none('1') // shouldn't be set }, children: [ s.text(str.is(Zwsp.ZWSP)) ] }) ] }), s.element('br', {}) ] }) ] }); })) ])), Logger.t('getParentCaretContainer', Step.sync(function () { const body = Element.fromHtml('
a
'); const caret = Element.fromDom(body.dom().firstChild); Assertions.assertDomEq('Should be caret element on child', caret, Element.fromDom(getParentCaretContainer(body.dom(), caret.dom().firstChild))); Assertions.assertDomEq('Should be caret element on self', caret, Element.fromDom(getParentCaretContainer(body.dom(), caret.dom()))); Assertions.assertEq('Should not be caret element', null, getParentCaretContainer(body, Element.fromTag('span').dom())); Assertions.assertEq('Should not be caret element', null, getParentCaretContainer(caret.dom(), caret.dom())); })), Logger.t('replaceWithCaretFormat', Step.sync(function () { const body = Element.fromHtml('

'); const formats = [ Element.fromTag('b').dom(), Element.fromTag('i').dom() ]; const pos = CaretFormat.replaceWithCaretFormat(body.dom().firstChild, formats); Assertions.assertEq('Should be at first offset', 0, pos.offset()); Assertions.assertEq('Should the zwsp text node', Zwsp.ZWSP, ( pos.container()).data); Assertions.assertStructure( 'Asserting the normalized structure of tiny content.', ApproxStructure.build(function (s, str) { return s.element('div', { children: [ s.element('span', { attrs: { 'id': str.is('_mce_caret'), 'data-mce-bogus': str.is('1') }, children: [ s.element('i', { children: [ s.element('b', { children: [ s.text(str.is(Zwsp.ZWSP)) ] }) ] }) ] }) ] }); }), body ); })), Logger.t('isFormatElement', Step.sync(function () { Assertions.assertEq('Should be format element', true, CaretFormat.isFormatElement(editor, Element.fromTag('b'))); Assertions.assertEq('Should be format element', true, CaretFormat.isFormatElement(editor, Element.fromTag('i'))); Assertions.assertEq('Should be format element', true, CaretFormat.isFormatElement(editor, Element.fromTag('u'))); Assertions.assertEq('Should be format element', true, CaretFormat.isFormatElement(editor, Element.fromTag('span'))); Assertions.assertEq('Should not be format element', false, CaretFormat.isFormatElement(editor, Element.fromTag('p'))); Assertions.assertEq('Should not be format element', false, CaretFormat.isFormatElement(editor, Element.fromTag('div'))); Assertions.assertEq('Should not be format element', false, CaretFormat.isFormatElement(editor, Element.fromHtml(''))); Assertions.assertEq('Should not be format element', false, CaretFormat.isFormatElement(editor, Element.fromHtml(''))); Assertions.assertEq('Should not be format element', false, CaretFormat.isFormatElement(editor, Element.fromHtml(''))); })) ], onSuccess, onFailure); }, { plugins: '', toolbar: '', skin_url: '/project/js/tinymce/skins/lightgray' }, success, failure); });