2022-08-08 17:00:14 +02:00
|
|
|
/**
|
|
|
|
* @jest-environment jsdom
|
|
|
|
*/
|
|
|
|
|
2022-12-30 19:25:31 +02:00
|
|
|
import { EditorSelection } from '@codemirror/state';
|
2022-08-08 17:00:14 +02:00
|
|
|
import {
|
|
|
|
toggleBolded, toggleCode, toggleHeaderLevel, toggleItalicized, toggleMath, updateLink,
|
|
|
|
} from './markdownCommands';
|
2022-12-30 19:25:31 +02:00
|
|
|
import createEditor from './testUtil/createEditor';
|
|
|
|
import { blockMathTagName } from './markdownMathParser';
|
2022-08-08 17:00:14 +02:00
|
|
|
|
|
|
|
describe('markdownCommands', () => {
|
2022-12-30 19:25:31 +02:00
|
|
|
it('should bold/italicize everything selected', async () => {
|
2022-08-08 17:00:14 +02:00
|
|
|
const initialDocText = 'Testing...';
|
2022-12-30 19:25:31 +02:00
|
|
|
const editor = await createEditor(
|
|
|
|
initialDocText, EditorSelection.range(0, initialDocText.length), []
|
2022-08-08 17:00:14 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
toggleBolded(editor);
|
|
|
|
|
|
|
|
let mainSel = editor.state.selection.main;
|
|
|
|
const boldedText = '**Testing...**';
|
|
|
|
expect(editor.state.doc.toString()).toBe(boldedText);
|
|
|
|
expect(mainSel.from).toBe(0);
|
|
|
|
expect(mainSel.to).toBe(boldedText.length);
|
|
|
|
|
|
|
|
toggleBolded(editor);
|
|
|
|
mainSel = editor.state.selection.main;
|
|
|
|
expect(editor.state.doc.toString()).toBe(initialDocText);
|
|
|
|
expect(mainSel.from).toBe(0);
|
|
|
|
expect(mainSel.to).toBe(initialDocText.length);
|
|
|
|
|
|
|
|
toggleItalicized(editor);
|
|
|
|
expect(editor.state.doc.toString()).toBe('*Testing...*');
|
|
|
|
|
|
|
|
toggleItalicized(editor);
|
|
|
|
expect(editor.state.doc.toString()).toBe('Testing...');
|
|
|
|
});
|
|
|
|
|
2022-12-30 19:25:31 +02:00
|
|
|
it('for a cursor, bolding, then italicizing, should produce a bold-italic region', async () => {
|
2022-09-05 13:50:32 +02:00
|
|
|
const initialDocText = '';
|
2022-12-30 19:25:31 +02:00
|
|
|
const editor = await createEditor(
|
|
|
|
initialDocText, EditorSelection.cursor(0), []
|
2022-09-05 13:50:32 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
toggleBolded(editor);
|
|
|
|
toggleItalicized(editor);
|
|
|
|
expect(editor.state.doc.toString()).toBe('******');
|
|
|
|
|
|
|
|
editor.dispatch(editor.state.replaceSelection('Test'));
|
|
|
|
expect(editor.state.doc.toString()).toBe('***Test***');
|
|
|
|
|
|
|
|
toggleItalicized(editor);
|
|
|
|
editor.dispatch(editor.state.replaceSelection(' Test'));
|
|
|
|
expect(editor.state.doc.toString()).toBe('***Test*** Test');
|
|
|
|
});
|
|
|
|
|
2022-12-30 19:25:31 +02:00
|
|
|
it('toggling math should both create and navigate out of math regions', async () => {
|
2022-08-08 17:00:14 +02:00
|
|
|
const initialDocText = 'Testing... ';
|
2022-12-30 19:25:31 +02:00
|
|
|
const editor = await createEditor(initialDocText, EditorSelection.cursor(initialDocText.length), []);
|
2022-08-08 17:00:14 +02:00
|
|
|
|
|
|
|
toggleMath(editor);
|
|
|
|
expect(editor.state.doc.toString()).toBe('Testing... $$');
|
|
|
|
expect(editor.state.selection.main.empty).toBe(true);
|
|
|
|
|
|
|
|
editor.dispatch(editor.state.replaceSelection('3 + 3 \\neq 5'));
|
|
|
|
expect(editor.state.doc.toString()).toBe('Testing... $3 + 3 \\neq 5$');
|
|
|
|
|
|
|
|
toggleMath(editor);
|
|
|
|
editor.dispatch(editor.state.replaceSelection('...'));
|
|
|
|
expect(editor.state.doc.toString()).toBe('Testing... $3 + 3 \\neq 5$...');
|
|
|
|
});
|
|
|
|
|
2022-12-30 19:25:31 +02:00
|
|
|
it('toggling inline code should both create and navigate out of an inline code region', async () => {
|
2022-08-08 17:00:14 +02:00
|
|
|
const initialDocText = 'Testing...\n\n';
|
2022-12-30 19:25:31 +02:00
|
|
|
const editor = await createEditor(initialDocText, EditorSelection.cursor(initialDocText.length), []);
|
2022-08-08 17:00:14 +02:00
|
|
|
|
|
|
|
toggleCode(editor);
|
|
|
|
editor.dispatch(editor.state.replaceSelection('f(x) = ...'));
|
|
|
|
toggleCode(editor);
|
|
|
|
|
|
|
|
editor.dispatch(editor.state.replaceSelection(' is a function.'));
|
|
|
|
expect(editor.state.doc.toString()).toBe('Testing...\n\n`f(x) = ...` is a function.');
|
|
|
|
});
|
|
|
|
|
2022-12-30 19:25:31 +02:00
|
|
|
it('should set headers to the proper levels (when toggling)', async () => {
|
2022-08-08 17:00:14 +02:00
|
|
|
const initialDocText = 'Testing...\nThis is a test.';
|
2022-12-30 19:25:31 +02:00
|
|
|
const editor = await createEditor(initialDocText, EditorSelection.cursor(3), []);
|
2022-08-08 17:00:14 +02:00
|
|
|
|
|
|
|
toggleHeaderLevel(1)(editor);
|
|
|
|
|
|
|
|
let mainSel = editor.state.selection.main;
|
|
|
|
expect(editor.state.doc.toString()).toBe('# Testing...\nThis is a test.');
|
|
|
|
expect(mainSel.empty).toBe(true);
|
|
|
|
expect(mainSel.from).toBe('# Testing...'.length);
|
|
|
|
|
|
|
|
toggleHeaderLevel(2)(editor);
|
|
|
|
|
|
|
|
mainSel = editor.state.selection.main;
|
|
|
|
expect(editor.state.doc.toString()).toBe('## Testing...\nThis is a test.');
|
|
|
|
expect(mainSel.empty).toBe(true);
|
|
|
|
expect(mainSel.from).toBe('## Testing...'.length);
|
|
|
|
|
|
|
|
toggleHeaderLevel(2)(editor);
|
|
|
|
|
|
|
|
mainSel = editor.state.selection.main;
|
|
|
|
expect(editor.state.doc.toString()).toEqual(initialDocText);
|
|
|
|
expect(mainSel.empty).toBe(true);
|
|
|
|
expect(mainSel.from).toBe('Testing...'.length);
|
|
|
|
});
|
|
|
|
|
2022-12-30 19:25:31 +02:00
|
|
|
it('headers should toggle properly within block quotes', async () => {
|
2022-08-08 17:00:14 +02:00
|
|
|
const initialDocText = 'Testing...\n\n> This is a test.\n> ...a test';
|
2022-12-30 19:25:31 +02:00
|
|
|
const editor = await createEditor(
|
2022-08-08 17:00:14 +02:00
|
|
|
initialDocText,
|
2022-12-30 19:25:31 +02:00
|
|
|
EditorSelection.cursor('Testing...\n\n> This'.length),
|
|
|
|
['Blockquote']
|
2022-08-08 17:00:14 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
toggleHeaderLevel(1)(editor);
|
|
|
|
|
|
|
|
const mainSel = editor.state.selection.main;
|
|
|
|
expect(editor.state.doc.toString()).toBe(
|
|
|
|
'Testing...\n\n> # This is a test.\n> ...a test'
|
|
|
|
);
|
|
|
|
expect(mainSel.empty).toBe(true);
|
|
|
|
expect(mainSel.from).toBe('Testing...\n\n> # This is a test.'.length);
|
|
|
|
|
|
|
|
toggleHeaderLevel(3)(editor);
|
|
|
|
|
|
|
|
expect(editor.state.doc.toString()).toBe(
|
|
|
|
'Testing...\n\n> ### This is a test.\n> ...a test'
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2022-12-30 19:25:31 +02:00
|
|
|
it('block math should be created correctly within block quotes', async () => {
|
|
|
|
const initialDocText = 'Testing...\n\n> This is a test.\n> y = mx + b\n> ...a test';
|
|
|
|
const editor = await createEditor(
|
|
|
|
initialDocText,
|
|
|
|
EditorSelection.range(
|
|
|
|
'Testing...\n\n> This'.length,
|
|
|
|
'Testing...\n\n> This is a test.\n> y = mx + b'.length
|
|
|
|
),
|
|
|
|
['Blockquote']
|
|
|
|
);
|
|
|
|
|
|
|
|
toggleMath(editor);
|
|
|
|
|
|
|
|
// Toggling math should surround the content in '$$'s
|
|
|
|
const mainSel = editor.state.selection.main;
|
|
|
|
expect(editor.state.doc.toString()).toEqual(
|
|
|
|
'Testing...\n\n> $$\n> This is a test.\n> y = mx + b\n> $$\n> ...a test'
|
|
|
|
);
|
|
|
|
expect(mainSel.from).toBe('Testing...\n\n'.length);
|
|
|
|
expect(mainSel.to).toBe('Testing...\n\n> $$\n> This is a test.\n> y = mx + b\n> $$'.length);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('block math should be correctly removed within block quotes', async () => {
|
|
|
|
const initialDocText = 'Testing...\n\n> $$\n> This is a test.\n> y = mx + b\n> $$\n> ...a test';
|
|
|
|
|
|
|
|
const editor = await createEditor(
|
|
|
|
initialDocText,
|
|
|
|
EditorSelection.cursor('Testing...\n\n> $$\n> This is'.length),
|
|
|
|
['Blockquote', blockMathTagName]
|
|
|
|
);
|
|
|
|
|
|
|
|
// Toggling math should remove the '$$'s
|
|
|
|
toggleMath(editor);
|
|
|
|
const mainSel = editor.state.selection.main;
|
|
|
|
expect(editor.state.doc.toString()).toEqual('Testing...\n\n> This is a test.\n> y = mx + b\n> ...a test');
|
|
|
|
expect(mainSel.from).toBe('Testing...\n\n'.length);
|
|
|
|
expect(mainSel.to).toBe('Testing...\n\n> This is a test.\n> y = mx + b'.length);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('updateLink should replace link titles and isolate URLs if no title is given', async () => {
|
2022-08-08 17:00:14 +02:00
|
|
|
const initialDocText = '[foo](http://example.com/)';
|
2022-12-30 19:25:31 +02:00
|
|
|
const editor = await createEditor(initialDocText, EditorSelection.cursor('[f'.length), ['Link']);
|
2022-08-08 17:00:14 +02:00
|
|
|
|
|
|
|
updateLink('bar', 'https://example.com/')(editor);
|
|
|
|
expect(editor.state.doc.toString()).toBe(
|
|
|
|
'[bar](https://example.com/)'
|
|
|
|
);
|
|
|
|
|
|
|
|
updateLink('', 'https://example.com/')(editor);
|
|
|
|
expect(editor.state.doc.toString()).toBe(
|
|
|
|
'https://example.com/'
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2022-12-30 19:25:31 +02:00
|
|
|
it('toggling math twice, starting on a line with content, should a math block', async () => {
|
2022-08-08 17:00:14 +02:00
|
|
|
const initialDocText = 'Testing... ';
|
2022-12-30 19:25:31 +02:00
|
|
|
const editor = await createEditor(initialDocText, EditorSelection.cursor(initialDocText.length), []);
|
2022-08-08 17:00:14 +02:00
|
|
|
|
|
|
|
toggleMath(editor);
|
|
|
|
toggleMath(editor);
|
|
|
|
editor.dispatch(editor.state.replaceSelection('f(x) = ...'));
|
|
|
|
expect(editor.state.doc.toString()).toBe('Testing... \n$$\nf(x) = ...\n$$');
|
|
|
|
});
|
|
|
|
|
2022-12-30 19:25:31 +02:00
|
|
|
it('toggling math twice on an empty line should create an empty math block', async () => {
|
2022-08-08 17:00:14 +02:00
|
|
|
const initialDocText = 'Testing...\n\n';
|
2022-12-30 19:25:31 +02:00
|
|
|
const editor = await createEditor(initialDocText, EditorSelection.cursor(initialDocText.length), []);
|
2022-08-08 17:00:14 +02:00
|
|
|
|
|
|
|
toggleMath(editor);
|
|
|
|
toggleMath(editor);
|
|
|
|
editor.dispatch(editor.state.replaceSelection('f(x) = ...'));
|
|
|
|
expect(editor.state.doc.toString()).toBe('Testing...\n\n$$\nf(x) = ...\n$$');
|
|
|
|
});
|
|
|
|
|
2022-12-30 19:25:31 +02:00
|
|
|
it('toggling code twice on an empty line should create an empty code block', async () => {
|
2022-08-08 17:00:14 +02:00
|
|
|
const initialDocText = 'Testing...\n\n';
|
2022-12-30 19:25:31 +02:00
|
|
|
const editor = await createEditor(initialDocText, EditorSelection.cursor(initialDocText.length), []);
|
2022-08-08 17:00:14 +02:00
|
|
|
|
|
|
|
// Toggling code twice should create a block code region
|
|
|
|
toggleCode(editor);
|
|
|
|
toggleCode(editor);
|
|
|
|
editor.dispatch(editor.state.replaceSelection('f(x) = ...'));
|
|
|
|
expect(editor.state.doc.toString()).toBe('Testing...\n\n```\nf(x) = ...\n```');
|
|
|
|
|
|
|
|
toggleCode(editor);
|
|
|
|
expect(editor.state.doc.toString()).toBe('Testing...\n\nf(x) = ...\n');
|
|
|
|
});
|
|
|
|
|
2022-12-30 19:25:31 +02:00
|
|
|
it('toggling math twice inside a block quote should produce an empty math block', async () => {
|
2022-08-08 17:00:14 +02:00
|
|
|
const initialDocText = '> Testing...> \n> ';
|
2022-12-30 19:25:31 +02:00
|
|
|
const editor = await createEditor(initialDocText, EditorSelection.cursor(initialDocText.length), ['Blockquote']);
|
2022-08-08 17:00:14 +02:00
|
|
|
|
|
|
|
toggleMath(editor);
|
|
|
|
toggleMath(editor);
|
|
|
|
editor.dispatch(editor.state.replaceSelection('f(x) = ...'));
|
|
|
|
expect(editor.state.doc.toString()).toBe(
|
|
|
|
'> Testing...> \n> \n> $$\n> f(x) = ...\n> $$'
|
|
|
|
);
|
|
|
|
|
|
|
|
// If we toggle math again, everything from the start of the line with the first
|
|
|
|
// $$ to the end of the document should be selected.
|
|
|
|
toggleMath(editor);
|
|
|
|
const sel = editor.state.selection.main;
|
|
|
|
expect(sel.from).toBe('> Testing...> \n> \n'.length);
|
|
|
|
expect(sel.to).toBe(editor.state.doc.length);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|