asynctest(
'browser.tinymce.core.fmt.CaretFormatTest',
[
'ephox.agar.api.ApproxStructure',
'ephox.agar.api.Assertions',
'ephox.agar.api.GeneralSteps',
'ephox.agar.api.Logger',
'ephox.agar.api.Pipeline',
'ephox.agar.api.Step',
'ephox.mcagar.api.TinyApis',
'ephox.mcagar.api.TinyLoader',
'ephox.sugar.api.node.Element',
'tinymce.core.fmt.CaretFormat',
'tinymce.core.test.TypeText',
'tinymce.core.text.Zwsp',
'tinymce.themes.modern.Theme'
],
function (ApproxStructure, Assertions, GeneralSteps, Logger, Pipeline, Step, TinyApis, TinyLoader, Element, CaretFormat, TypeText, Zwsp, ModernTheme) {
var success = arguments[arguments.length - 2];
var failure = arguments[arguments.length - 1];
ModernTheme();
var sApplyCaretFormat = function (editor, name, vars) {
return Step.sync(function () {
CaretFormat.applyCaretFormat(editor, name, vars);
});
};
var sRemoveCaretFormat = function (editor, name, vars, similar) {
return Step.sync(function () {
CaretFormat.removeCaretFormat(editor, name, vars, similar);
});
};
var sSetRawContent = function (editor, html) {
return Step.sync(function () {
editor.getBody().innerHTML = html;
});
};
var sAssertNormalizedContentStructure = function (editor, expected) {
return Step.sync(function () {
var 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) {
var 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('<p>a</p>'),
tinyApis.sSetCursor([0, 0], 1),
sApplyCaretFormat(editor, 'bold', {}),
TypeText.sTypeContentAtSelection(Element.fromDom(editor.getDoc()), 'x'),
tinyApis.sAssertContent('<p>a<strong>x</strong></p>'),
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('<p>ab</p>'),
tinyApis.sSetCursor([0, 0], 1),
sApplyCaretFormat(editor, 'bold', {}),
tinyApis.sAssertContent('<p><strong>ab</strong></p>'),
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('<p><strong>a</strong></p>'),
tinyApis.sSetCursor([0, 0, 0], 1),
sRemoveCaretFormat(editor, 'bold', {}),
TypeText.sTypeContentAtSelection(Element.fromDom(editor.getDoc()), 'x'),
tinyApis.sAssertContent('<p><strong>a</strong>x</p>'),
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('<p><strong>ab</strong></p>'),
tinyApis.sSetCursor([0, 0, 0], 1),
sRemoveCaretFormat(editor, 'bold', {}),
tinyApis.sAssertContent('<p>ab</p>'),
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('<p>a</p>'),
tinyApis.sSetCursor([0, 0], 1),
sApplyCaretFormat(editor, 'bold', {}),
sRemoveCaretFormat(editor, 'bold', {}),
TypeText.sTypeContentAtSelection(Element.fromDom(editor.getDoc()), 'x'),
tinyApis.sAssertContent('<p>ax</p>'),
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('<p><strong>a</strong></p>'),
tinyApis.sSetCursor([0, 0], 1),
sRemoveCaretFormat(editor, 'bold', {}),
sApplyCaretFormat(editor, 'bold', {}),
TypeText.sTypeContentAtSelection(Element.fromDom(editor.getDoc()), 'x'),
tinyApis.sAssertContent('<p><strong>a</strong><strong>x</strong></p>'),
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([
sSetRawContent(editor, '<p>a<br></p>'),
tinyApis.sSetCursor([0, 0], 1),
sApplyCaretFormat(editor, 'bold', {}),
TypeText.sTypeContentAtSelection(Element.fromDom(editor.getDoc()), 'x'),
tinyApis.sAssertContent('<p>a<strong>x</strong></p>'),
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([
sSetRawContent(editor, '<p><strong>a<br></strong></p>'),
tinyApis.sSetCursor([0, 0, 0], 1),
sRemoveCaretFormat(editor, 'bold', {}),
TypeText.sTypeContentAtSelection(Element.fromDom(editor.getDoc()), 'x'),
tinyApis.sAssertContent('<p><strong>a</strong>x</p>'),
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([
sSetRawContent(editor, '<p>a</p><p><strong><br></strong></p>'),
tinyApis.sSetCursor([1, 0, 0], 0),
sRemoveCaretFormat(editor, 'bold', {}),
tinyApis.sAssertContent('<p>a</p>\n<p> </p>'),
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('<p>ax</p>\n<p> </p>'),
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, CaretFormat.isCaretNode(editor.dom.create('b')));
Assertions.assertEq('Should be false since it ia caret node', true, CaretFormat.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 () {
var body = Element.fromHtml('<div><span id="_mce_caret">a</span></div>');
var caret = Element.fromDom(body.dom().firstChild);
Assertions.assertDomEq('Should be caret element on child', caret, Element.fromDom(CaretFormat.getParentCaretContainer(body.dom(), caret.dom().firstChild)));
Assertions.assertDomEq('Should be caret element on self', caret, Element.fromDom(CaretFormat.getParentCaretContainer(body.dom(), caret.dom())));
Assertions.assertEq('Should not be caret element', null, CaretFormat.getParentCaretContainer(body, Element.fromTag('span').dom()));
Assertions.assertEq('Should not be caret element', null, CaretFormat.getParentCaretContainer(caret.dom(), caret.dom()));
})),
Logger.t('replaceWithCaretFormat', Step.sync(function () {
var body = Element.fromHtml('<div><br /></div>');
var formats = [
Element.fromTag('b').dom(),
Element.fromTag('i').dom()
];
var 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('<a href="#"></a>')));
Assertions.assertEq('Should not be format element', false, CaretFormat.isFormatElement(editor, Element.fromHtml('<span data-mce-bogus="1"></span>')));
Assertions.assertEq('Should not be format element', false, CaretFormat.isFormatElement(editor, Element.fromHtml('<span id="_mce_caret"></span>')));
}))
], onSuccess, onFailure);
}, {
plugins: '',
toolbar: '',
skin_url: '/project/src/skins/lightgray/dist/lightgray'
}, success, failure);
}
);
|