Skip to content

Commit

Permalink
Preserve selection in tables with open typeahead menu (#5820)
Browse files Browse the repository at this point in the history
  • Loading branch information
serey-roth authored Apr 22, 2024
1 parent bba84b9 commit da06d8f
Show file tree
Hide file tree
Showing 2 changed files with 183 additions and 42 deletions.
205 changes: 163 additions & 42 deletions packages/lexical-playground/__tests__/e2e/Tables.spec.mjs
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import {
moveLeft,
moveRight,
moveToEditorBeginning,
moveUp,
pressBackspace,
selectAll,
} from '../keyboardShortcuts/index.mjs';
Expand Down Expand Up @@ -44,6 +45,7 @@ import {
test,
toggleColumnHeader,
unmergeTableCell,
waitForSelector,
} from '../utils/index.mjs';

async function fillTablePartiallyWithText(page) {
Expand Down Expand Up @@ -486,52 +488,171 @@ test.describe('Tables', () => {
});
});

test(`Can navigate table with keyboard`, async ({
page,
isPlainText,
isCollab,
}) => {
await initialize({isCollab, page});
test.skip(isPlainText);
test.describe(`Can navigate table with keyboard`, () => {
test(`Can navigate cells horizontally`, async ({
page,
isPlainText,
isCollab,
}) => {
await initialize({isCollab, page});
test.skip(isPlainText);

await focusEditor(page);
await insertTable(page, 2, 3);
await focusEditor(page);
await insertTable(page, 2, 2);

await fillTablePartiallyWithText(page);
await assertHTML(
page,
html`
<p><br /></p>
<table>
<tr>
<th>
<p><br /></p>
</th>
<th>
<p><br /></p>
</th>
</tr>
<tr>
<th>
<p><br /></p>
</th>
<td>
<p><br /></p>
</td>
</tr>
</table>
<p><br /></p>
`,
undefined,
{ignoreClasses: true},
);

await assertHTML(
await assertSelection(page, {
anchorOffset: 0,
anchorPath: [1, 0, 0, 0],
focusOffset: 0,
focusPath: [1, 0, 0, 0],
});

await moveRight(page, 1);
await assertSelection(page, {
anchorOffset: 0,
anchorPath: [1, 0, 1, 0],
focusOffset: 0,
focusPath: [1, 0, 1, 0],
});

await moveRight(page, 1);
await assertSelection(page, {
anchorOffset: 0,
anchorPath: [1, 1, 0, 0],
focusOffset: 0,
focusPath: [1, 1, 0, 0],
});

await moveRight(page, 1);
await assertSelection(page, {
anchorOffset: 0,
anchorPath: [1, 1, 1, 0],
focusOffset: 0,
focusPath: [1, 1, 1, 0],
});

await moveLeft(page, 1);
await assertSelection(page, {
anchorOffset: 0,
anchorPath: [1, 1, 0, 0],
focusOffset: 0,
focusPath: [1, 1, 0, 0],
});

await moveLeft(page, 1);
await assertSelection(page, {
anchorOffset: 0,
anchorPath: [1, 0, 1, 0],
focusOffset: 0,
focusPath: [1, 0, 1, 0],
});

await moveLeft(page, 1);
await assertSelection(page, {
anchorOffset: 0,
anchorPath: [1, 0, 0, 0],
focusOffset: 0,
focusPath: [1, 0, 0, 0],
});
});

test(`Can navigate cells vertically`, async ({
page,
html`
<p><br /></p>
<table>
<tr>
<th>
<p dir="ltr"><span data-lexical-text="true">a</span></p>
</th>
<th>
<p dir="ltr"><span data-lexical-text="true">bb</span></p>
</th>
<th>
<p dir="ltr"><span data-lexical-text="true">cc</span></p>
</th>
</tr>
<tr>
<th>
<p dir="ltr"><span data-lexical-text="true">d</span></p>
</th>
<td>
<p dir="ltr"><span data-lexical-text="true">e</span></p>
</td>
<td>
<p dir="ltr"><span data-lexical-text="true">f</span></p>
</td>
</tr>
</table>
<p><br /></p>
`,
undefined,
{ignoreClasses: true},
);
isPlainText,
isCollab,
}) => {
await initialize({isCollab, page});
test.skip(isPlainText);

await focusEditor(page);
await insertTable(page, 2, 2);

await assertSelection(page, {
anchorOffset: 0,
anchorPath: [1, 0, 0, 0],
focusOffset: 0,
focusPath: [1, 0, 0, 0],
});

await moveDown(page, 1);
await assertSelection(page, {
anchorOffset: 0,
anchorPath: [1, 1, 0, 0],
focusOffset: 0,
focusPath: [1, 1, 0, 0],
});

await moveUp(page, 1);
await assertSelection(page, {
anchorOffset: 0,
anchorPath: [1, 0, 0, 0],
focusOffset: 0,
focusPath: [1, 0, 0, 0],
});
});

test('Should not navigate cells when typeahead menu is open and focused', async ({
page,
isCollab,
isPlainText,
}) => {
await initialize({isCollab, page});
test.skip(isPlainText);

await focusEditor(page);
await insertTable(page, 2, 2);

await page.keyboard.type('@A');
await assertSelection(page, {
anchorOffset: 2,
anchorPath: [1, 0, 0, 0, 0, 0],
focusOffset: 2,
focusPath: [1, 0, 0, 0, 0, 0],
});

await waitForSelector(page, `#typeahead-menu ul li:first-child.selected`);

await moveDown(page, 1);
await assertSelection(page, {
anchorOffset: 2,
anchorPath: [1, 0, 0, 0, 0, 0],
focusOffset: 2,
focusPath: [1, 0, 0, 0, 0, 0],
});

await waitForSelector(
page,
'#typeahead-menu ul li:nth-child(2).selected',
);
});
});

test(`Can select cells using Table selection`, async ({
Expand Down
20 changes: 20 additions & 0 deletions packages/lexical-table/src/LexicalTableSelectionHelpers.ts
Original file line number Diff line number Diff line change
Expand Up @@ -1263,6 +1263,13 @@ function $handleArrowKey(
tableNode: TableNode,
tableObserver: TableObserver,
): boolean {
if (
(direction === 'up' || direction === 'down') &&
isTypeaheadMenuInView(editor)
) {
return false;
}

const selection = $getSelection();

if (!$isSelectionInTable(selection, tableNode)) {
Expand Down Expand Up @@ -1479,6 +1486,19 @@ function stopEvent(event: Event) {
event.stopPropagation();
}

function isTypeaheadMenuInView(editor: LexicalEditor) {
// There is no inbuilt way to check if the component picker is in view
// but we can check if the root DOM element has the aria-controls attribute "typeahead-menu".
const root = editor.getRootElement();
if (!root) {
return false;
}
return (
root.hasAttribute('aria-controls') &&
root.getAttribute('aria-controls') === 'typeahead-menu'
);
}

function isExitingTableAnchor(
type: string,
offset: number,
Expand Down

0 comments on commit da06d8f

Please sign in to comment.