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
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
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
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('