mirror of
https://codeberg.org/forgejo/forgejo.git
synced 2025-01-20 16:50:28 -05:00
Fix mention and emoji expansion & Improve leaving list completion (#6597)
Reviewed-on: https://codeberg.org/forgejo/forgejo/pulls/6597 Reviewed-by: 0ko <0ko@noreply.codeberg.org>
This commit is contained in:
commit
0379739ac9
2 changed files with 71 additions and 30 deletions
|
@ -109,7 +109,7 @@ test('markdown indentation', async ({page}) => {
|
||||||
});
|
});
|
||||||
|
|
||||||
test('markdown list continuation', async ({page}) => {
|
test('markdown list continuation', async ({page}) => {
|
||||||
const initText = `* first\n* second\n* third\n* last`;
|
const initText = `* first\n* second`;
|
||||||
|
|
||||||
const response = await page.goto('/user2/repo1/issues/new');
|
const response = await page.goto('/user2/repo1/issues/new');
|
||||||
expect(response?.status()).toBe(200);
|
expect(response?.status()).toBe(200);
|
||||||
|
@ -119,25 +119,20 @@ test('markdown list continuation', async ({page}) => {
|
||||||
const indent = page.locator('button[data-md-action="indent"]');
|
const indent = page.locator('button[data-md-action="indent"]');
|
||||||
await textarea.fill(initText);
|
await textarea.fill(initText);
|
||||||
|
|
||||||
// Test continuation of '* ' prefix
|
// Test continuation of ' * ' prefix
|
||||||
await textarea.evaluate((it:HTMLTextAreaElement) => it.setSelectionRange(it.value.indexOf('cond'), it.value.indexOf('cond')));
|
await textarea.evaluate((it:HTMLTextAreaElement) => it.setSelectionRange(it.value.indexOf('rst'), it.value.indexOf('rst')));
|
||||||
|
await indent.click();
|
||||||
await textarea.press('End');
|
await textarea.press('End');
|
||||||
await textarea.press('Enter');
|
await textarea.press('Enter');
|
||||||
await textarea.pressSequentially('middle');
|
|
||||||
await expect(textarea).toHaveValue(`* first\n* second\n* middle\n* third\n* last`);
|
|
||||||
|
|
||||||
// Test continuation of ' * ' prefix
|
|
||||||
await indent.click();
|
|
||||||
await textarea.press('Enter');
|
|
||||||
await textarea.pressSequentially('muddle');
|
await textarea.pressSequentially('muddle');
|
||||||
await expect(textarea).toHaveValue(`* first\n* second\n${tab}* middle\n${tab}* muddle\n* third\n* last`);
|
await expect(textarea).toHaveValue(`${tab}* first\n${tab}* muddle\n* second`);
|
||||||
|
|
||||||
// Test breaking in the middle of a line
|
// Test breaking in the middle of a line
|
||||||
await textarea.evaluate((it:HTMLTextAreaElement) => it.setSelectionRange(it.value.lastIndexOf('ddle'), it.value.lastIndexOf('ddle')));
|
await textarea.evaluate((it:HTMLTextAreaElement) => it.setSelectionRange(it.value.lastIndexOf('ddle'), it.value.lastIndexOf('ddle')));
|
||||||
await textarea.pressSequentially('tate');
|
await textarea.pressSequentially('tate');
|
||||||
await textarea.press('Enter');
|
await textarea.press('Enter');
|
||||||
await textarea.pressSequentially('me');
|
await textarea.pressSequentially('me');
|
||||||
await expect(textarea).toHaveValue(`* first\n* second\n${tab}* middle\n${tab}* mutate\n${tab}* meddle\n* third\n* last`);
|
await expect(textarea).toHaveValue(`${tab}* first\n${tab}* mutate\n${tab}* meddle\n* second`);
|
||||||
|
|
||||||
// Test not triggering when Shift held
|
// Test not triggering when Shift held
|
||||||
await textarea.fill(initText);
|
await textarea.fill(initText);
|
||||||
|
@ -145,35 +140,36 @@ test('markdown list continuation', async ({page}) => {
|
||||||
await textarea.press('Shift+Enter');
|
await textarea.press('Shift+Enter');
|
||||||
await textarea.press('Enter');
|
await textarea.press('Enter');
|
||||||
await textarea.pressSequentially('...but not least');
|
await textarea.pressSequentially('...but not least');
|
||||||
await expect(textarea).toHaveValue(`* first\n* second\n* third\n* last\n\n...but not least`);
|
await expect(textarea).toHaveValue(`* first\n* second\n\n...but not least`);
|
||||||
|
|
||||||
// Test continuation of ordered list
|
// Test continuation of ordered list
|
||||||
await textarea.fill(`1. one\n2. two`);
|
await textarea.fill(`1. one`);
|
||||||
await textarea.evaluate((it:HTMLTextAreaElement) => it.setSelectionRange(it.value.length, it.value.length));
|
await textarea.evaluate((it:HTMLTextAreaElement) => it.setSelectionRange(it.value.length, it.value.length));
|
||||||
await textarea.press('Enter');
|
await textarea.press('Enter');
|
||||||
|
await textarea.pressSequentially(' ');
|
||||||
|
await textarea.press('Enter');
|
||||||
await textarea.pressSequentially('three');
|
await textarea.pressSequentially('three');
|
||||||
await expect(textarea).toHaveValue(`1. one\n2. two\n3. three`);
|
await textarea.press('Enter');
|
||||||
|
await textarea.press('Enter');
|
||||||
|
await expect(textarea).toHaveValue(`1. one\n2. \n3. three\n\n`);
|
||||||
|
|
||||||
// Test continuation of alternative ordered list syntax
|
// Test continuation of alternative ordered list syntax
|
||||||
await textarea.fill(`1) one\n2) two`);
|
await textarea.fill(`1) one`);
|
||||||
await textarea.evaluate((it:HTMLTextAreaElement) => it.setSelectionRange(it.value.length, it.value.length));
|
await textarea.evaluate((it:HTMLTextAreaElement) => it.setSelectionRange(it.value.length, it.value.length));
|
||||||
await textarea.press('Enter');
|
await textarea.press('Enter');
|
||||||
|
await textarea.pressSequentially(' ');
|
||||||
|
await textarea.press('Enter');
|
||||||
await textarea.pressSequentially('three');
|
await textarea.pressSequentially('three');
|
||||||
await expect(textarea).toHaveValue(`1) one\n2) two\n3) three`);
|
|
||||||
|
|
||||||
// Test continuation of blockquote
|
|
||||||
await textarea.fill(`> knowledge is power`);
|
|
||||||
await textarea.evaluate((it:HTMLTextAreaElement) => it.setSelectionRange(it.value.length, it.value.length));
|
|
||||||
await textarea.press('Enter');
|
await textarea.press('Enter');
|
||||||
await textarea.pressSequentially('france is bacon');
|
await textarea.press('Enter');
|
||||||
await expect(textarea).toHaveValue(`> knowledge is power\n> france is bacon`);
|
await expect(textarea).toHaveValue(`1) one\n2) \n3) three\n\n`);
|
||||||
|
|
||||||
// Test continuation of checklists
|
// Test continuation of checklists
|
||||||
await textarea.fill(`- [ ] have a problem\n- [x] create a solution`);
|
await textarea.fill(`- [ ]have a problem\n- [x]create a solution`);
|
||||||
await textarea.evaluate((it:HTMLTextAreaElement) => it.setSelectionRange(it.value.length, it.value.length));
|
await textarea.evaluate((it:HTMLTextAreaElement) => it.setSelectionRange(it.value.length, it.value.length));
|
||||||
await textarea.press('Enter');
|
await textarea.press('Enter');
|
||||||
await textarea.pressSequentially('write a test');
|
await textarea.pressSequentially('write a test');
|
||||||
await expect(textarea).toHaveValue(`- [ ] have a problem\n- [x] create a solution\n- [ ] write a test`);
|
await expect(textarea).toHaveValue(`- [ ]have a problem\n- [x]create a solution\n- [ ]write a test`);
|
||||||
|
|
||||||
// Test all conceivable syntax (except ordered lists)
|
// Test all conceivable syntax (except ordered lists)
|
||||||
const prefixes = [
|
const prefixes = [
|
||||||
|
@ -189,7 +185,6 @@ test('markdown list continuation', async ({page}) => {
|
||||||
'> ',
|
'> ',
|
||||||
'> > ',
|
'> > ',
|
||||||
'- [ ] ',
|
'- [ ] ',
|
||||||
'- [ ]', // This does seem to render, so allow.
|
|
||||||
'* [ ] ',
|
'* [ ] ',
|
||||||
'+ [ ] ',
|
'+ [ ] ',
|
||||||
];
|
];
|
||||||
|
@ -197,8 +192,12 @@ test('markdown list continuation', async ({page}) => {
|
||||||
await textarea.fill(`${prefix}one`);
|
await textarea.fill(`${prefix}one`);
|
||||||
await textarea.evaluate((it:HTMLTextAreaElement) => it.setSelectionRange(it.value.length, it.value.length));
|
await textarea.evaluate((it:HTMLTextAreaElement) => it.setSelectionRange(it.value.length, it.value.length));
|
||||||
await textarea.press('Enter');
|
await textarea.press('Enter');
|
||||||
|
await textarea.pressSequentially(' ');
|
||||||
|
await textarea.press('Enter');
|
||||||
await textarea.pressSequentially('two');
|
await textarea.pressSequentially('two');
|
||||||
await expect(textarea).toHaveValue(`${prefix}one\n${prefix}two`);
|
await textarea.press('Enter');
|
||||||
|
await textarea.press('Enter');
|
||||||
|
await expect(textarea).toHaveValue(`${prefix}one\n${prefix} \n${prefix}two\n\n`);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -224,3 +223,29 @@ test('markdown insert table', async ({page}) => {
|
||||||
await expect(textarea).toHaveValue('| Header | Header |\n|---------|---------|\n| Content | Content |\n| Content | Content |\n| Content | Content |\n');
|
await expect(textarea).toHaveValue('| Header | Header |\n|---------|---------|\n| Content | Content |\n| Content | Content |\n| Content | Content |\n');
|
||||||
await save_visual(page);
|
await save_visual(page);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('text expander has higher prio then prefix continuation', async ({page}) => {
|
||||||
|
const response = await page.goto('/user2/repo1/issues/new');
|
||||||
|
expect(response?.status()).toBe(200);
|
||||||
|
|
||||||
|
const textarea = page.locator('textarea[name=content]');
|
||||||
|
const initText = `* first`;
|
||||||
|
await textarea.fill(initText);
|
||||||
|
await textarea.evaluate((it:HTMLTextAreaElement) => it.setSelectionRange(it.value.indexOf('rst'), it.value.indexOf('rst')));
|
||||||
|
await textarea.press('End');
|
||||||
|
|
||||||
|
// Test emoji completion
|
||||||
|
await textarea.press('Enter');
|
||||||
|
await textarea.pressSequentially(':smile_c');
|
||||||
|
await textarea.press('Enter');
|
||||||
|
await expect(textarea).toHaveValue(`* first\n* 😸`);
|
||||||
|
|
||||||
|
// Test username completion
|
||||||
|
await textarea.press('Enter');
|
||||||
|
await textarea.pressSequentially('@user');
|
||||||
|
await textarea.press('Enter');
|
||||||
|
await expect(textarea).toHaveValue(`* first\n* 😸\n* @user2 `);
|
||||||
|
|
||||||
|
await textarea.press('Enter');
|
||||||
|
await expect(textarea).toHaveValue(`* first\n* 😸\n* @user2 \n* `);
|
||||||
|
});
|
||||||
|
|
|
@ -99,6 +99,8 @@ class ComboMarkdownEditor {
|
||||||
e.target._shiftDown = true;
|
e.target._shiftDown = true;
|
||||||
}
|
}
|
||||||
if (e.key === 'Enter' && !e.shiftKey && !e.ctrlKey && !e.altKey) {
|
if (e.key === 'Enter' && !e.shiftKey && !e.ctrlKey && !e.altKey) {
|
||||||
|
// Prevent special line break handling if currently a text expander popup is open
|
||||||
|
if (this.textarea.hasAttribute('aria-expanded')) return;
|
||||||
if (!this.breakLine()) return; // Nothing changed, let the default handler work.
|
if (!this.breakLine()) return; // Nothing changed, let the default handler work.
|
||||||
this.options?.onContentChanged?.(this, e);
|
this.options?.onContentChanged?.(this, e);
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
|
@ -407,13 +409,27 @@ class ComboMarkdownEditor {
|
||||||
// Find the beginning of the current line.
|
// Find the beginning of the current line.
|
||||||
const lineStart = Math.max(0, value.lastIndexOf('\n', start - 1) + 1);
|
const lineStart = Math.max(0, value.lastIndexOf('\n', start - 1) + 1);
|
||||||
// Find the end and extract the line.
|
// Find the end and extract the line.
|
||||||
const lineEnd = value.indexOf('\n', start);
|
const nextLF = value.indexOf('\n', start);
|
||||||
const line = value.slice(lineStart, lineEnd === -1 ? value.length : lineEnd);
|
const lineEnd = nextLF === -1 ? value.length : nextLF;
|
||||||
|
const line = value.slice(lineStart, lineEnd);
|
||||||
// Match any whitespace at the start + any repeatable prefix + exactly one space after.
|
// Match any whitespace at the start + any repeatable prefix + exactly one space after.
|
||||||
const prefix = line.match(/^\s*((\d+)[.)]\s|[-*+]\s+(\[[ x]\]\s?)?|(>\s+)+)?/);
|
const prefix = line.match(/^\s*((\d+)[.)]\s|[-*+]\s{1,4}\[[ x]\]\s?|[-*+]\s|(>\s?)+)?/);
|
||||||
|
|
||||||
// Defer to browser if we can't do anything more useful, or if the cursor is inside the prefix.
|
// Defer to browser if we can't do anything more useful, or if the cursor is inside the prefix.
|
||||||
if (!prefix || !prefix[0].length || lineStart + prefix[0].length > start) return false;
|
if (!prefix) return false;
|
||||||
|
const prefixLength = prefix[0].length;
|
||||||
|
if (!prefixLength || lineStart + prefixLength > start) return false;
|
||||||
|
// If the prefix is just indentation (which should always be an even number of spaces or tabs), check if a single whitespace is added to the end of the line.
|
||||||
|
// If this is the case do not leave the indentation and continue with the prefix.
|
||||||
|
if ((prefixLength % 2 === 1 && /^ +$/.test(prefix[0])) || /^\t+ $/.test(prefix[0])) {
|
||||||
|
prefix[0] = prefix[0].slice(0, prefixLength - 1);
|
||||||
|
} else if (prefixLength === lineEnd - lineStart) {
|
||||||
|
this.textarea.setSelectionRange(lineStart, lineEnd);
|
||||||
|
if (!document.execCommand('insertText', false, '\n')) {
|
||||||
|
this.textarea.setRangeText('\n');
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
// Insert newline + prefix.
|
// Insert newline + prefix.
|
||||||
let text = `\n${prefix[0]}`;
|
let text = `\n${prefix[0]}`;
|
||||||
|
|
Loading…
Add table
Reference in a new issue