import { GeneralSteps, Pipeline, Logger, Assertions, ApproxStructure } from '@ephox/agar'; import { UnitTest } from '@ephox/bedrock'; import { TinyApis, TinyLoader } from '@ephox/mcagar'; import { Editor } from 'tinymce/core/api/Editor'; import ModernTheme from 'tinymce/themes/modern/Theme'; import { sAnnotate, sAssertHtmlContent } from '../../module/test/AnnotationAsserts'; import { Element } from '@ephox/sugar'; UnitTest.asynctest('browser.tinymce.core.annotate.AnnotateTest', (success, failure) => { ModernTheme(); TinyLoader.setup(function (editor: Editor, onSuccess, onFailure) { const tinyApis = TinyApis(editor); // TODO: Consider testing collapse sections. const sTestWordGrabIfCollapsed = Logger.t( 'Should word grab with a collapsed selection', GeneralSteps.sequence([ // '
This |is| the first paragraph
This is the second.
' tinyApis.sSetContent('This is the first paragraph here
This is the second.
'), tinyApis.sSetSelection([ 0, 0 ], 'This is the first p'.length, [ 0, 0 ], 'This is the first p'.length), sAnnotate(editor, 'test-annotation', 'test-uid', { anything: 'one-paragraph' }), sAssertHtmlContent(tinyApis, [ `This is the first paragraph here
`, 'This is the second.
FirstThird
'), tinyApis.sSetSelection([ 1 ], 0, [ 1 ], 0), sAnnotate(editor, 'test-annotation', 'test-uid', { anything: 'empty-paragraph' }), sAssertHtmlContent(tinyApis, [ 'First
', 'Third
' ]), ]) ); const sTestCanAnnotateBeforeTwoNonBreakingSpaces = Logger.t( 'Should annotate when the cursor is collapsed before two nbsps', GeneralSteps.sequence([ tinyApis.sSetContent('Annotation here , please
'), tinyApis.sSetCursor([ 0, 0 ], 'Annotation here '.length), sAnnotate(editor, 'test-annotation', 'test-uid', { anything: 'nbsp-paragraph' }), Assertions.sAssertStructure( 'Checking body element', ApproxStructure.build((s, str, arr) => { return s.element('body', { children: [ s.element('p', { children: [ s.text( str.is('Annotation here ') ), s.element('span', { classes: [ arr.has('mce-annotation') ], html: str.is(' ') }), s.text( str.is('\u00A0\u00A0, please')) ] }) ] }); }), Element.fromDom(editor.getBody()) ) ]) ); const sTestCanAnnotateWithinTwoNonBreakingSpaces = Logger.t( 'Should annotate when the cursor is collapsed between two nbsps', GeneralSteps.sequence([ tinyApis.sSetContent('Annotation here , please
'), tinyApis.sSetCursor([ 0, 0 ], 'Annotation here '.length), sAnnotate(editor, 'test-annotation', 'test-uid', { anything: 'nbsp-paragraph' }), Assertions.sAssertStructure( 'Checking body element', ApproxStructure.build((s, str, arr) => { return s.element('body', { children: [ s.element('p', { children: [ s.text( str.is('Annotation here \u00A0') ), s.element('span', { classes: [ arr.has('mce-annotation') ], html: str.is(' ') }), s.text( str.is('\u00A0, please')) ] }) ] }); }), Element.fromDom(editor.getBody()) ) ]) ); const sTestCanAnnotateAfterTwoNonBreakingSpaces = Logger.t( 'Should annotate when the cursor is collapsed after two nbsps', GeneralSteps.sequence([ tinyApis.sSetContent('Annotation here , please
'), tinyApis.sSetCursor([ 0, 0 ], 'Annotation here '.length), sAnnotate(editor, 'test-annotation', 'test-uid', { anything: 'nbsp-paragraph' }), Assertions.sAssertStructure( 'Checking body element', ApproxStructure.build((s, str, arr) => { return s.element('body', { children: [ s.element('p', { children: [ s.text( str.is('Annotation here \u00A0\u00A0') ), s.element('span', { classes: [ arr.has('mce-annotation') ], html: str.is(',') }), s.text( str.is(' please')) ] }) ] }); }), Element.fromDom(editor.getBody()) ) ]) ); const sTestDoesNotWordGrabIfNotCollapsed = Logger.t( 'Should not word grab if the selection is not collapsed', GeneralSteps.sequence([ // 'This |is| the first paragraph
This is the second.
' tinyApis.sSetContent('This is the first paragraph
This is the second.
'), tinyApis.sSetSelection([ 0, 0 ], 'This is the first p'.length, [ 0, 0 ], 'This is the first par'.length), sAnnotate(editor, 'test-annotation', 'test-uid', { anything: 'one-paragraph' }), sAssertHtmlContent(tinyApis, [ `This is the first paragraph
`, 'This is the second.
This |is| the first paragraphThis is the second.
' tinyApis.sSetContent('This is the first paragraph
This is the second.
'), tinyApis.sSetSelection([ 0, 0 ], 'This '.length, [ 0, 0 ], 'This is'.length), sAnnotate(editor, 'test-annotation', 'test-uid', { anything: 'one-paragraph' }), sAssertHtmlContent(tinyApis, [ `This is the first paragraph
`, 'This is the second.
This |is the first paragraphThis is| the second.
' tinyApis.sSetContent('This is the first paragraph
This is the second.
'), tinyApis.sSetSelection([ 0, 0 ], 'This '.length, [ 1, 0 ], 'This is'.length), sAnnotate(editor, 'test-annotation', 'test-uid', { anything: 'two-paragraphs' }), sAssertHtmlContent(tinyApis, [ `This is the first paragraph
`, `This is the second.
` ]), tinyApis.sAssertSelection([ 0 ], 1, [ 1 ], 1) ]) ); const sTestInThreeParagraphs = Logger.t( 'Testing over three paragraphs', GeneralSteps.sequence([ // 'This |is the first paragraph
This is the second.
This is| the third.
' tinyApis.sSetContent('This is the first paragraph
This is the second.
This is the third.
'), tinyApis.sSetSelection([ 0, 0 ], 'This '.length, [ 2, 0 ], 'This is'.length), sAnnotate(editor, 'test-annotation', 'test-uid', { anything: 'three-paragraphs' }), sAssertHtmlContent(tinyApis, [ `This is the first paragraph
`, `This is the second.
`, `This is the third.
` ]), tinyApis.sAssertSelection([ 0 ], 1, [ 2 ], 1) ]) ); Pipeline.async({}, [ tinyApis.sFocus, sTestWordGrabIfCollapsed, sTestDoesNotWordGrabIfNotCollapsed, sTestCanAnnotateDirectParentOfRoot, sTestCanAnnotateBeforeTwoNonBreakingSpaces, sTestCanAnnotateWithinTwoNonBreakingSpaces, sTestCanAnnotateAfterTwoNonBreakingSpaces, sTestInOneParagraph, sTestInTwoParagraphs, sTestInThreeParagraphs ], onSuccess, onFailure); }, { skin_url: '/project/js/tinymce/skins/lightgray', setup: (ed: Editor) => { ed.on('init', () => { ed.annotator.register('test-annotation', { decorate: (uid, data) => { return { attributes: { 'data-test-anything': data.anything }, classes: [ ] }; } }); }); } }, success, failure); });