debian-mirror-gitlab/spec/frontend/lib/utils/text_markdown_spec.js

477 lines
15 KiB
JavaScript
Raw Normal View History

2020-11-24 15:15:51 +05:30
import { insertMarkdownText, keypressNoteText } from '~/lib/utils/text_markdown';
2018-03-17 18:26:18 +05:30
describe('init markdown', () => {
let textArea;
beforeAll(() => {
textArea = document.createElement('textarea');
document.querySelector('body').appendChild(textArea);
textArea.focus();
});
afterAll(() => {
textArea.parentNode.removeChild(textArea);
});
2021-01-03 14:25:43 +05:30
describe('insertMarkdownText', () => {
it('will not error if selected text is a number', () => {
const selected = 2;
insertMarkdownText({
textArea,
text: '',
tag: '',
blockTag: null,
selected,
wrap: false,
});
expect(textArea.value).toBe(selected.toString());
});
});
2019-02-15 15:39:39 +05:30
describe('textArea', () => {
describe('without selection', () => {
it('inserts the tag on an empty line', () => {
const initialValue = '';
2018-03-17 18:26:18 +05:30
2019-02-15 15:39:39 +05:30
textArea.value = initialValue;
textArea.selectionStart = 0;
textArea.selectionEnd = 0;
2018-03-17 18:26:18 +05:30
2019-02-15 15:39:39 +05:30
insertMarkdownText({
textArea,
text: textArea.value,
2020-06-23 00:09:42 +05:30
tag: '- ',
2019-02-15 15:39:39 +05:30
blockTag: null,
selected: '',
wrap: false,
});
2020-06-23 00:09:42 +05:30
expect(textArea.value).toEqual(`${initialValue}- `);
2018-12-13 13:39:08 +05:30
});
2018-03-17 18:26:18 +05:30
2021-06-08 01:23:25 +05:30
it('inserts dollar signs correctly', () => {
const initialValue = '';
textArea.value = initialValue;
textArea.selectionStart = 0;
textArea.selectionEnd = 0;
insertMarkdownText({
textArea,
text: textArea.value,
tag: '```suggestion:-0+0\n{text}\n```',
blockTag: true,
selected: '# Does not parse the `$` currently.',
wrap: false,
});
expect(textArea.value).toContain('# Does not parse the `$` currently.');
});
2019-02-15 15:39:39 +05:30
it('inserts the tag on a new line if the current one is not empty', () => {
const initialValue = 'some text';
2018-03-17 18:26:18 +05:30
2019-02-15 15:39:39 +05:30
textArea.value = initialValue;
textArea.setSelectionRange(initialValue.length, initialValue.length);
2018-03-17 18:26:18 +05:30
2019-02-15 15:39:39 +05:30
insertMarkdownText({
textArea,
text: textArea.value,
2020-06-23 00:09:42 +05:30
tag: '- ',
2019-02-15 15:39:39 +05:30
blockTag: null,
selected: '',
wrap: false,
});
2018-03-17 18:26:18 +05:30
2020-06-23 00:09:42 +05:30
expect(textArea.value).toEqual(`${initialValue}\n- `);
2018-12-13 13:39:08 +05:30
});
2018-03-17 18:26:18 +05:30
2021-11-11 11:23:49 +05:30
it('unescapes new line characters', () => {
const initialValue = '';
textArea.value = initialValue;
textArea.selectionStart = 0;
textArea.selectionEnd = 0;
insertMarkdownText({
textArea,
text: textArea.value,
tag: '```suggestion:-0+0\n{text}\n```',
blockTag: true,
2021-12-11 22:18:48 +05:30
selected: '# Does not %br parse the %br currently.',
2021-11-11 11:23:49 +05:30
wrap: false,
});
2021-12-11 22:18:48 +05:30
expect(textArea.value).toContain('# Does not \\n parse the \\n currently.');
2021-11-11 11:23:49 +05:30
});
2019-02-15 15:39:39 +05:30
it('inserts the tag on the same line if the current line only contains spaces', () => {
const initialValue = ' ';
2018-03-17 18:26:18 +05:30
2019-02-15 15:39:39 +05:30
textArea.value = initialValue;
textArea.setSelectionRange(initialValue.length, initialValue.length);
2018-03-17 18:26:18 +05:30
2019-02-15 15:39:39 +05:30
insertMarkdownText({
textArea,
text: textArea.value,
2020-06-23 00:09:42 +05:30
tag: '- ',
2019-02-15 15:39:39 +05:30
blockTag: null,
selected: '',
wrap: false,
});
2018-03-17 18:26:18 +05:30
2020-06-23 00:09:42 +05:30
expect(textArea.value).toEqual(`${initialValue}- `);
2018-12-13 13:39:08 +05:30
});
2018-03-17 18:26:18 +05:30
2019-02-15 15:39:39 +05:30
it('inserts the tag on the same line if the current line only contains tabs', () => {
const initialValue = '\t\t\t';
2018-03-17 18:26:18 +05:30
2019-02-15 15:39:39 +05:30
textArea.value = initialValue;
textArea.setSelectionRange(initialValue.length, initialValue.length);
2018-03-17 18:26:18 +05:30
2019-02-15 15:39:39 +05:30
insertMarkdownText({
textArea,
text: textArea.value,
2020-06-23 00:09:42 +05:30
tag: '- ',
2019-02-15 15:39:39 +05:30
blockTag: null,
selected: '',
wrap: false,
});
2018-03-17 18:26:18 +05:30
2020-06-23 00:09:42 +05:30
expect(textArea.value).toEqual(`${initialValue}- `);
2018-12-13 13:39:08 +05:30
});
2018-03-17 18:26:18 +05:30
2019-02-15 15:39:39 +05:30
it('places the cursor inside the tags', () => {
const start = 'lorem ';
const end = ' ipsum';
const tag = '*';
2018-12-05 23:21:45 +05:30
2019-02-15 15:39:39 +05:30
textArea.value = `${start}${end}`;
textArea.setSelectionRange(start.length, start.length);
2018-12-05 23:21:45 +05:30
2019-02-15 15:39:39 +05:30
insertMarkdownText({
textArea,
text: textArea.value,
tag,
blockTag: null,
selected: '',
wrap: true,
});
2018-12-05 23:21:45 +05:30
2019-02-15 15:39:39 +05:30
expect(textArea.value).toEqual(`${start}**${end}`);
2018-12-05 23:21:45 +05:30
2019-02-15 15:39:39 +05:30
// cursor placement should be between tags
expect(textArea.selectionStart).toBe(start.length + tag.length);
2018-12-13 13:39:08 +05:30
});
2022-04-04 11:22:00 +05:30
describe('Continuing markdown lists', () => {
const enterEvent = new KeyboardEvent('keydown', { key: 'Enter' });
beforeEach(() => {
gon.features = { markdownContinueLists: true };
});
it.each`
text | expected
${'- item'} | ${'- item\n- '}
${'- [ ] item'} | ${'- [ ] item\n- [ ] '}
${'- [x] item'} | ${'- [x] item\n- [x] '}
${'- item\n - second'} | ${'- item\n - second\n - '}
${'1. item'} | ${'1. item\n1. '}
${'1. [ ] item'} | ${'1. [ ] item\n1. [ ] '}
${'1. [x] item'} | ${'1. [x] item\n1. [x] '}
${'108. item'} | ${'108. item\n108. '}
${'108. item\n - second'} | ${'108. item\n - second\n - '}
${'108. item\n 1. second'} | ${'108. item\n 1. second\n 1. '}
`('adds correct list continuation characters', ({ text, expected }) => {
textArea.value = text;
textArea.setSelectionRange(text.length, text.length);
textArea.addEventListener('keydown', keypressNoteText);
textArea.dispatchEvent(enterEvent);
expect(textArea.value).toEqual(expected);
expect(textArea.selectionStart).toBe(expected.length);
});
// test that when pressing Enter on an empty list item, the empty
// list item text is selected, so that when the Enter propagates,
// it's removed
it.each`
text | expected
${'- item\n- '} | ${'- item\n'}
${'- [ ] item\n- [ ] '} | ${'- [ ] item\n'}
${'- [x] item\n- [x] '} | ${'- [x] item\n'}
${'- item\n - second\n - '} | ${'- item\n - second\n'}
${'1. item\n1. '} | ${'1. item\n'}
${'1. [ ] item\n1. [ ] '} | ${'1. [ ] item\n'}
${'1. [x] item\n1. [x] '} | ${'1. [x] item\n'}
${'108. item\n108. '} | ${'108. item\n'}
${'108. item\n - second\n - '} | ${'108. item\n - second\n'}
${'108. item\n 1. second\n 1. '} | ${'108. item\n 1. second\n'}
`('adds correct list continuation characters', ({ text, expected }) => {
textArea.value = text;
textArea.setSelectionRange(text.length, text.length);
textArea.addEventListener('keydown', keypressNoteText);
textArea.dispatchEvent(enterEvent);
expect(textArea.value.substr(0, textArea.selectionStart)).toEqual(expected);
expect(textArea.selectionStart).toBe(expected.length);
expect(textArea.selectionEnd).toBe(text.length);
});
it('does nothing if feature flag disabled', () => {
gon.features = { markdownContinueLists: false };
const text = '- item';
const expected = '- item';
textArea.value = text;
textArea.setSelectionRange(text.length, text.length);
textArea.addEventListener('keydown', keypressNoteText);
textArea.dispatchEvent(enterEvent);
expect(textArea.value).toEqual(expected);
expect(textArea.selectionStart).toBe(expected.length);
});
});
2018-12-05 23:21:45 +05:30
});
2019-02-15 15:39:39 +05:30
describe('with selection', () => {
const text = 'initial selected value';
const selected = 'selected';
2020-11-24 15:15:51 +05:30
let selectedIndex;
2019-02-15 15:39:39 +05:30
beforeEach(() => {
textArea.value = text;
2020-11-24 15:15:51 +05:30
selectedIndex = text.indexOf(selected);
2019-02-15 15:39:39 +05:30
textArea.setSelectionRange(selectedIndex, selectedIndex + selected.length);
});
it('applies the tag to the selected value', () => {
const tag = '*';
2018-12-05 23:21:45 +05:30
2018-12-13 13:39:08 +05:30
insertMarkdownText({
textArea,
2018-12-05 23:21:45 +05:30
text: textArea.value,
tag,
blockTag: null,
selected,
2019-02-15 15:39:39 +05:30
wrap: true,
2018-12-13 13:39:08 +05:30
});
2018-12-05 23:21:45 +05:30
2019-02-15 15:39:39 +05:30
expect(textArea.value).toEqual(text.replace(selected, `*${selected}*`));
2018-12-13 13:39:08 +05:30
2019-02-15 15:39:39 +05:30
// cursor placement should be after selection + 2 tag lengths
expect(textArea.selectionStart).toBe(selectedIndex + selected.length + 2 * tag.length);
2018-12-05 23:21:45 +05:30
});
2019-02-15 15:39:39 +05:30
it('replaces the placeholder in the tag', () => {
2018-12-13 13:39:08 +05:30
insertMarkdownText({
textArea,
2018-12-05 23:21:45 +05:30
text: textArea.value,
2019-02-15 15:39:39 +05:30
tag: '[{text}](url)',
2018-12-05 23:21:45 +05:30
blockTag: null,
selected,
wrap: false,
2018-12-13 13:39:08 +05:30
});
2018-12-05 23:21:45 +05:30
2019-02-15 15:39:39 +05:30
expect(textArea.value).toEqual(text.replace(selected, `[${selected}](url)`));
2018-12-05 23:21:45 +05:30
});
2018-12-13 13:39:08 +05:30
2021-04-17 20:07:23 +05:30
describe('surrounds selected text with matching character', () => {
it.each`
key | expected
${'['} | ${`[${selected}]`}
${'*'} | ${`**${selected}**`}
${"'"} | ${`'${selected}'`}
${'_'} | ${`_${selected}_`}
${'`'} | ${`\`${selected}\``}
${'"'} | ${`"${selected}"`}
${'{'} | ${`{${selected}}`}
${'('} | ${`(${selected})`}
${'<'} | ${`<${selected}>`}
`('generates $expected when $key is pressed', ({ key, expected }) => {
const event = new KeyboardEvent('keydown', { key });
gon.markdown_surround_selection = true;
textArea.addEventListener('keydown', keypressNoteText);
textArea.dispatchEvent(event);
expect(textArea.value).toEqual(text.replace(selected, expected));
// cursor placement should be after selection + 2 tag lengths
expect(textArea.selectionStart).toBe(selectedIndex + expected.length);
});
2020-11-24 15:15:51 +05:30
2021-04-17 20:07:23 +05:30
it('does nothing if user preference disabled', () => {
const event = new KeyboardEvent('keydown', { key: '[' });
gon.markdown_surround_selection = false;
textArea.addEventListener('keydown', keypressNoteText);
textArea.dispatchEvent(event);
expect(textArea.value).toEqual(text);
});
2020-11-24 15:15:51 +05:30
});
2019-02-15 15:39:39 +05:30
describe('and text to be selected', () => {
const tag = '[{text}](url)';
const select = 'url';
it('selects the text', () => {
insertMarkdownText({
textArea,
text: textArea.value,
tag,
blockTag: null,
selected,
wrap: false,
select,
});
const expectedText = text.replace(selected, `[${selected}](url)`);
expect(textArea.value).toEqual(expectedText);
expect(textArea.selectionStart).toEqual(expectedText.indexOf(select));
expect(textArea.selectionEnd).toEqual(expectedText.indexOf(select) + select.length);
});
2018-12-13 13:39:08 +05:30
2019-02-15 15:39:39 +05:30
it('selects the right text when multiple tags are present', () => {
const initialValue = `${tag} ${tag} ${selected}`;
textArea.value = initialValue;
2020-11-24 15:15:51 +05:30
selectedIndex = initialValue.indexOf(selected);
2019-02-15 15:39:39 +05:30
textArea.setSelectionRange(selectedIndex, selectedIndex + selected.length);
insertMarkdownText({
textArea,
text: textArea.value,
tag,
blockTag: null,
selected,
wrap: false,
select,
});
const expectedText = initialValue.replace(selected, `[${selected}](url)`);
expect(textArea.value).toEqual(expectedText);
expect(textArea.selectionStart).toEqual(expectedText.lastIndexOf(select));
expect(textArea.selectionEnd).toEqual(expectedText.lastIndexOf(select) + select.length);
});
2018-12-13 13:39:08 +05:30
2019-02-15 15:39:39 +05:30
it('should support selected urls', () => {
const expectedUrl = 'http://www.gitlab.com';
const expectedSelectionText = 'text';
const expectedText = `text [${expectedSelectionText}](${expectedUrl}) text`;
const initialValue = `text ${expectedUrl} text`;
textArea.value = initialValue;
2020-11-24 15:15:51 +05:30
selectedIndex = initialValue.indexOf(expectedUrl);
2019-02-15 15:39:39 +05:30
textArea.setSelectionRange(selectedIndex, selectedIndex + expectedUrl.length);
insertMarkdownText({
textArea,
text: textArea.value,
tag,
blockTag: null,
selected: expectedUrl,
wrap: false,
select,
});
expect(textArea.value).toEqual(expectedText);
expect(textArea.selectionStart).toEqual(expectedText.indexOf(expectedSelectionText, 1));
expect(textArea.selectionEnd).toEqual(
expectedText.indexOf(expectedSelectionText, 1) + expectedSelectionText.length,
);
2018-12-13 13:39:08 +05:30
});
2019-02-15 15:39:39 +05:30
});
});
});
2021-09-30 23:02:18 +05:30
describe('Source Editor', () => {
2020-07-28 23:09:34 +05:30
let editor;
beforeEach(() => {
editor = {
getSelection: jest.fn().mockReturnValue({
startLineNumber: 1,
startColumn: 1,
endLineNumber: 2,
endColumn: 2,
}),
getValue: jest.fn().mockReturnValue('this is text \n in two lines'),
selectWithinSelection: jest.fn(),
replaceSelectedText: jest.fn(),
moveCursor: jest.fn(),
};
});
it('replaces selected text', () => {
insertMarkdownText({
text: editor.getValue,
tag: '*',
blockTag: null,
selected: '',
wrap: false,
editor,
});
expect(editor.replaceSelectedText).toHaveBeenCalled();
});
it('adds block tags on line above and below selection', () => {
const selected = 'this text \n is multiple \n lines';
const text = `before \n ${selected} \n after`;
insertMarkdownText({
text,
tag: '',
blockTag: '***',
selected,
wrap: true,
editor,
});
expect(editor.replaceSelectedText).toHaveBeenCalledWith(`***\n${selected}\n***\n`, undefined);
});
it('uses ace editor to navigate back tag length when nothing is selected', () => {
editor.getSelection = jest.fn().mockReturnValue({
startLineNumber: 1,
startColumn: 1,
endLineNumber: 1,
endColumn: 1,
});
insertMarkdownText({
text: editor.getValue,
tag: '*',
blockTag: null,
selected: '',
wrap: true,
editor,
});
expect(editor.moveCursor).toHaveBeenCalledWith(-1);
});
it('ace editor does not navigate back when there is selected text', () => {
insertMarkdownText({
text: editor.getValue,
tag: '*',
blockTag: null,
selected: 'foobar',
wrap: true,
editor,
});
expect(editor.selectWithinSelection).not.toHaveBeenCalled();
});
});
2018-03-17 18:26:18 +05:30
});