Skip to content

Commit 621e3eb

Browse files
authored
[lexical-list][lexical-playground] Bug fix: support pasting google doc checklist (#6191)
1 parent 4816bf7 commit 621e3eb

File tree

4 files changed

+39
-8
lines changed

4 files changed

+39
-8
lines changed

packages/lexical-list/src/LexicalListItemNode.ts

Lines changed: 7 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,6 @@ import type {
2424

2525
import {
2626
addClassNamesToElement,
27-
isHTMLElement,
2827
removeClassNamesFromElement,
2928
} from '@lexical/utils';
3029
import {
@@ -493,9 +492,14 @@ function updateListItemChecked(
493492
}
494493
}
495494

496-
function $convertListItemElement(domNode: Node): DOMConversionOutput {
495+
function $convertListItemElement(domNode: HTMLElement): DOMConversionOutput {
496+
const ariaCheckedAttr = domNode.getAttribute('aria-checked');
497497
const checked =
498-
isHTMLElement(domNode) && domNode.getAttribute('aria-checked') === 'true';
498+
ariaCheckedAttr === 'true'
499+
? true
500+
: ariaCheckedAttr === 'false'
501+
? false
502+
: undefined;
499503
return {node: $createListItemNode(checked)};
500504
}
501505

packages/lexical-list/src/LexicalListNode.ts

Lines changed: 17 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -302,6 +302,22 @@ function $normalizeChildren(nodes: Array<LexicalNode>): Array<ListItemNode> {
302302
return normalizedListItems;
303303
}
304304

305+
function isDomChecklist(domNode: Node) {
306+
if (
307+
isHTMLElement(domNode) &&
308+
domNode.getAttribute('__lexicallisttype') === 'check'
309+
) {
310+
return true;
311+
}
312+
// if children are checklist items, the node is a checklist ul. Applicable for googledoc checklist pasting.
313+
for (const child of domNode.childNodes) {
314+
if (isHTMLElement(child) && child.hasAttribute('aria-checked')) {
315+
return true;
316+
}
317+
}
318+
return false;
319+
}
320+
305321
function $convertListNode(domNode: Node): DOMConversionOutput {
306322
const nodeName = domNode.nodeName.toLowerCase();
307323
let node = null;
@@ -310,10 +326,7 @@ function $convertListNode(domNode: Node): DOMConversionOutput {
310326
const start = domNode.start;
311327
node = $createListNode('number', start);
312328
} else if (nodeName === 'ul') {
313-
if (
314-
isHTMLElement(domNode) &&
315-
domNode.getAttribute('__lexicallisttype') === 'check'
316-
) {
329+
if (isDomChecklist(domNode)) {
317330
node = $createListNode('check');
318331
} else {
319332
node = $createListNode('bullet');

packages/lexical-playground/src/nodes/ImageNode.tsx

Lines changed: 10 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -37,9 +37,18 @@ export interface ImagePayload {
3737
captionsEnabled?: boolean;
3838
}
3939

40+
function isGoogleDocCheckboxImg(img: HTMLImageElement): boolean {
41+
return (
42+
img.parentElement != null &&
43+
img.parentElement.tagName === 'LI' &&
44+
img.previousSibling === null &&
45+
img.getAttribute('aria-roledescription') === 'checkbox'
46+
);
47+
}
48+
4049
function $convertImageElement(domNode: Node): null | DOMConversionOutput {
4150
const img = domNode as HTMLImageElement;
42-
if (img.src.startsWith('file:///')) {
51+
if (img.src.startsWith('file:///') || isGoogleDocCheckboxImg(img)) {
4352
return null;
4453
}
4554
const {alt: altText, src, width, height} = img;

packages/lexical/src/__tests__/unit/HTMLCopyAndPaste.test.ts

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -100,6 +100,11 @@ describe('HTMLCopyAndPaste tests', () => {
100100
name: 'nested div in a span',
101101
pastedHTML: ` <span>123<div>456</div></span>`,
102102
},
103+
{
104+
expectedHTML: `<ul><li role="checkbox" tabindex="-1" aria-checked="true" value="1" dir="ltr"><span data-lexical-text="true">done</span></li><li role="checkbox" tabindex="-1" aria-checked="false" value="2" dir="ltr"><span data-lexical-text="true">todo</span></li><li value="3"><ul><li role="checkbox" tabindex="-1" aria-checked="true" value="1" dir="ltr"><span data-lexical-text="true">done</span></li><li role="checkbox" tabindex="-1" aria-checked="false" value="2" dir="ltr"><span data-lexical-text="true">todo</span></li></ul></li><li role="checkbox" tabindex="-1" aria-checked="false" value="3" dir="ltr"><span data-lexical-text="true">todo</span></li></ul>`,
105+
name: 'google doc checklist',
106+
pastedHTML: `<meta charset='utf-8'><meta charset="utf-8"><b style="font-weight:normal;" id="docs-internal-guid-1980f960-7fff-f4df-4ba3-26c6e1508542"><ul style="margin-top:0;margin-bottom:0;padding-inline-start:28px;"><li dir="ltr" role="checkbox" aria-checked="true" style="list-style-type:none;font-size:11.5pt;font-family:'Optimistic Text',sans-serif;color:#000000;background-color:transparent;font-weight:400;font-style:normal;font-variant:normal;text-decoration:line-through;-webkit-text-decoration-skip:none;text-decoration-skip-ink:none;vertical-align:baseline;white-space:pre;" aria-level="1"><img src="" width="18.4px" height="18.4px" alt="checked" aria-roledescription="checkbox" style="margin-right:3px;" /><p dir="ltr" style="line-height:1.38;margin-top:0pt;margin-bottom:0pt;display:inline-block;vertical-align:top;margin-top:0;" role="presentation"><span style="font-size:11.5pt;font-family:'Optimistic Text',sans-serif;color:#000000;background-color:transparent;font-weight:400;font-style:normal;font-variant:normal;text-decoration:line-through;-webkit-text-decoration-skip:none;text-decoration-skip-ink:none;vertical-align:baseline;white-space:pre;white-space:pre-wrap;">done</span></p></li><li dir="ltr" role="checkbox" aria-checked="false" style="list-style-type:none;font-size:11.5pt;font-family:'Optimistic Text',sans-serif;color:#000000;background-color:transparent;font-weight:400;font-style:normal;font-variant:normal;text-decoration:none;vertical-align:baseline;white-space:pre;" aria-level="1"><img src="" width="18.4px" height="18.4px" alt="unchecked" aria-roledescription="checkbox" style="margin-right:3px;" /><p dir="ltr" style="line-height:1.38;margin-top:0pt;margin-bottom:0pt;display:inline-block;vertical-align:top;margin-top:0;" role="presentation"><span style="font-size:11.5pt;font-family:'Optimistic Text',sans-serif;color:#000000;background-color:transparent;font-weight:400;font-style:normal;font-variant:normal;text-decoration:none;vertical-align:baseline;white-space:pre;white-space:pre-wrap;">todo</span></p></li><ul style="margin-top:0;margin-bottom:0;padding-inline-start:28px;"><li dir="ltr" role="checkbox" aria-checked="true" style="list-style-type:none;font-size:11.5pt;font-family:'Optimistic Text',sans-serif;color:#000000;background-color:transparent;font-weight:400;font-style:normal;font-variant:normal;text-decoration:line-through;-webkit-text-decoration-skip:none;text-decoration-skip-ink:none;vertical-align:baseline;white-space:pre;" aria-level="2"><img src="" width="18.4px" height="18.4px" alt="checked" aria-roledescription="checkbox" style="margin-right:3px;" /><p dir="ltr" style="line-height:1.38;margin-top:0pt;margin-bottom:0pt;display:inline-block;vertical-align:top;margin-top:0;" role="presentation"><span style="font-size:11.5pt;font-family:'Optimistic Text',sans-serif;color:#000000;background-color:transparent;font-weight:400;font-style:normal;font-variant:normal;text-decoration:line-through;-webkit-text-decoration-skip:none;text-decoration-skip-ink:none;vertical-align:baseline;white-space:pre;white-space:pre-wrap;">done</span></p></li><li dir="ltr" role="checkbox" aria-checked="false" style="list-style-type:none;font-size:11.5pt;font-family:'Optimistic Text',sans-serif;color:#000000;background-color:transparent;font-weight:400;font-style:normal;font-variant:normal;text-decoration:none;vertical-align:baseline;white-space:pre;" aria-level="2"><img src="" width="18.4px" height="18.4px" alt="unchecked" aria-roledescription="checkbox" style="margin-right:3px;" /><p dir="ltr" style="line-height:1.38;margin-top:0pt;margin-bottom:0pt;display:inline-block;vertical-align:top;margin-top:0;" role="presentation"><span style="font-size:11.5pt;font-family:'Optimistic Text',sans-serif;color:#000000;background-color:transparent;font-weight:400;font-style:normal;font-variant:normal;text-decoration:none;vertical-align:baseline;white-space:pre;white-space:pre-wrap;">todo</span></p></li></ul><li dir="ltr" role="checkbox" aria-checked="false" style="list-style-type:none;font-size:11.5pt;font-family:'Optimistic Text',sans-serif;color:#000000;background-color:transparent;font-weight:400;font-style:normal;font-variant:normal;text-decoration:none;vertical-align:baseline;white-space:pre;" aria-level="1"><img src="" width="18.4px" height="18.4px" alt="unchecked" aria-roledescription="checkbox" style="margin-right:3px;" /><p dir="ltr" style="line-height:1.38;margin-top:0pt;margin-bottom:0pt;display:inline-block;vertical-align:top;margin-top:0;" role="presentation"><span style="font-size:11.5pt;font-family:'Optimistic Text',sans-serif;color:#000000;background-color:transparent;font-weight:400;font-style:normal;font-variant:normal;text-decoration:none;vertical-align:baseline;white-space:pre;white-space:pre-wrap;">todo</span></p></li></ul></b>`,
107+
},
103108
];
104109

105110
HTML_COPY_PASTING_TESTS.forEach((testCase, i) => {

0 commit comments

Comments
 (0)