import { describe, expect, test } from "@odoo/hoot"; import { setupEditor, testEditor } from "../_helpers/editor"; import { tick } from "@odoo/hoot-mock"; import { press } from "@odoo/hoot-dom"; import { simulateArrowKeyPress } from "../_helpers/user_actions"; import { getContent, setSelection } from "../_helpers/selection"; import { unformat } from "../_helpers/format"; const keyPress = (keys) => async (editor) => { await simulateArrowKeyPress(editor, keys); // Allow onselectionchange handler to run. await tick(); }; describe("Around ZWS", () => { test("should move past a zws (collapsed - ArrowRight)", async () => { await testEditor({ contentBefore: '
ab[]\u200Bcd
', stepFunction: keyPress("ArrowRight"), contentAfter: 'ab\u200Bc[]d
', }); await testEditor({ contentBefore: 'ab[]\u200Bcd
', stepFunction: keyPress("ArrowRight"), contentAfter: 'ab\u200Bc[]d
', }); }); test.tags("focus required"); test("should move past a zws (collapsed - ArrowLeft)", async () => { await testEditor({ contentBefore: 'ab\u200B[]cd
', stepFunction: keyPress("ArrowLeft"), contentAfter: 'a[]b\u200Bcd
', }); await testEditor({ contentBefore: 'ab\u200B[]cd
', stepFunction: keyPress("ArrowLeft"), contentAfter: 'a[]b\u200Bcd
', }); await testEditor({ contentBefore: '\u200B
[]\u200Bab
', stepFunction: keyPress("ArrowLeft"), contentAfter: '\u200B[]
\u200Bab
', }); await testEditor({ contentBefore: '\u200B
\u200B[]
', stepFunction: keyPress("ArrowLeft"), contentAfter: '\u200B[]
\u200B
', }); }); test("should move past a zws (collapsed at the end of a block)", async () => { await testEditor({ contentBefore: 'ab[]\u200B
cd
', stepFunction: keyPress("ArrowRight"), contentAfter: 'ab\u200B
[]cd
', }); await testEditor({ contentBefore: 'ab[]\u200B
cd
', stepFunction: keyPress("ArrowRight"), contentAfter: 'ab\u200B
[]cd
', }); await testEditor({ contentBefore: 'ab\u200B[]
\u200B
', stepFunction: keyPress("ArrowRight"), contentAfter: 'ab\u200B
[]\u200B
', }); await testEditor({ contentBefore: 'ab[]\u200B
\u200B
', stepFunction: keyPress("ArrowRight"), contentAfter: 'ab\u200B
[]\u200B
', }); }); test("should select a zws", async () => { await testEditor({ contentBefore: '[ab]\u200Bcd
', stepFunction: keyPress(["Shift", "ArrowRight"]), contentAfter: '[ab\u200Bc]d
', }); await testEditor({ contentBefore: '[ab]\u200Bcd
', stepFunction: keyPress(["Shift", "ArrowRight"]), contentAfter: '[ab\u200Bc]d
', }); }); test("should select a zws (2)", async () => { await testEditor({ contentBefore: 'a[b]\u200Bcd
', stepFunction: keyPress(["Shift", "ArrowRight"]), contentAfter: 'a[b\u200Bc]d
', }); await testEditor({ contentBefore: 'a[b]\u200Bcd
', stepFunction: keyPress(["Shift", "ArrowRight"]), contentAfter: 'a[b\u200Bc]d
', }); await testEditor({ contentBefore: 'a[b]\u200B
\u200B
', stepFunction: keyPress(["Shift", "ArrowRight"]), contentAfter: 'a[b\u200B
]\u200B
', }); }); test("should select a zws (3)", async () => { await testEditor({ contentBefore: 'ab[]\u200Bcd
', stepFunction: keyPress(["Shift", "ArrowRight"]), contentAfter: 'ab[\u200Bc]d
', }); await testEditor({ contentBefore: 'ab[]\u200Bcd
', stepFunction: keyPress(["Shift", "ArrowRight"]), contentAfter: 'ab[\u200Bc]d
', }); }); test("should select a zws backwards (ArrowLeft)", async () => { await testEditor({ contentBefore: 'ab\u200B[]cd
', stepFunction: keyPress(["Shift", "ArrowLeft"]), contentAfter: 'a]b\u200B[cd
', }); await testEditor({ contentBefore: 'ab\u200B[]cd
', stepFunction: keyPress(["Shift", "ArrowLeft"]), contentAfter: 'a]b\u200B[cd
', }); }); test("should select a zws backwards (ArrowLeft - 2)", async () => { await testEditor({ contentBefore: 'ab\u200B]cd[
', stepFunction: keyPress(["Shift", "ArrowLeft"]), contentAfter: 'a]b\u200Bcd[
', }); await testEditor({ contentBefore: 'ab\u200B]cd[
', stepFunction: keyPress(["Shift", "ArrowLeft"]), contentAfter: 'a]b\u200Bcd[
', }); }); test("should select a zws backwards (ArrowLeft - 3)", async () => { await testEditor({ contentBefore: 'ab\u200B]c[d
', stepFunction: keyPress(["Shift", "ArrowLeft"]), contentAfter: 'a]b\u200Bc[d
', }); await testEditor({ contentBefore: 'ab\u200B]c[d
', stepFunction: keyPress(["Shift", "ArrowLeft"]), contentAfter: 'a]b\u200Bc[d
', }); }); test("should select a zws backwards (ArrowRight)", async () => { await testEditor({ contentBefore: 'ab]\u200B[cd
', stepFunction: keyPress(["Shift", "ArrowRight"]), contentAfter: 'ab\u200B[c]d
', }); await testEditor({ contentBefore: 'ab]\u200B[cd
', stepFunction: keyPress(["Shift", "ArrowRight"]), contentAfter: 'ab\u200B[c]d
', }); await testEditor({ contentBefore: 'ab]\u200B[cd
', stepFunction: keyPress(["Shift", "ArrowRight"]), contentAfter: 'ab\u200B[c]d
', }); await testEditor({ contentBefore: 'ab]\u200B[cd
', stepFunction: keyPress(["Shift", "ArrowRight"]), contentAfter: 'ab\u200B[c]d
', }); }); test("should select a zws backwards (ArrowRight - 2)", async () => { await testEditor({ contentBefore: 'ab]\u200Bc[d
', stepFunction: keyPress(["Shift", "ArrowRight"]), contentAfter: 'ab\u200Bc[]d
', }); await testEditor({ contentBefore: 'ab]\u200Bc[d
', stepFunction: keyPress(["Shift", "ArrowRight"]), contentAfter: 'ab\u200Bc[]d
', }); }); test("should deselect a zws", async () => { await testEditor({ contentBefore: 'ab[\u200B]cd
', stepFunction: keyPress(["Shift", "ArrowLeft"]), contentAfter: 'a]b[\u200Bcd
', // Normalized by the browser }); await testEditor({ contentBefore: 'ab[\u200B]cd
', stepFunction: keyPress(["Shift", "ArrowLeft"]), contentAfter: 'a]b[\u200Bcd
', // Normalized by the browser }); await testEditor({ contentBefore: 'ab[\u200B]cd
', stepFunction: keyPress(["Shift", "ArrowLeft"]), contentAfter: 'a]b[\u200Bcd
', // Normalized by the browser }); await testEditor({ contentBefore: 'ab[\u200B]cd
', stepFunction: keyPress(["Shift", "ArrowLeft"]), contentAfter: 'a]b[\u200Bcd
', // Normalized by the browser }); }); test("should deselect a zws (2)", async () => { await testEditor({ contentBefore: 'a[b\u200B]cd
', stepFunction: keyPress(["Shift", "ArrowLeft"]), contentAfter: 'a[]b\u200Bcd
', }); await testEditor({ contentBefore: 'a[b\u200B]cd
', stepFunction: keyPress(["Shift", "ArrowLeft"]), contentAfter: 'a[]b\u200Bcd
', }); }); }); describe("Around links", () => { test("should move into a link (ArrowRight)", async () => { await testEditor({ contentBefore: 'ab[]cdef
', contentBeforeEdit: "ab[]" + "\ufeff" + // before zwnbsp '' + "\ufeff" + // start zwnbsp "cd" + // content "\ufeff" + // end zwnbsp "" + "\ufeff" + // after zwnbsp "ef
", stepFunction: keyPress("ArrowRight"), contentAfterEdit: "ab" + "\ufeff" + // before zwnbsp '' + "\ufeff" + // start zwnbsp "[]cd" + // content "\ufeff" + // end zwnbsp "" + "\ufeff" + // after zwnbsp "ef
", contentAfter: 'ab[]cdef
', }); }); test("should move into a link (ArrowLeft)", async () => { await testEditor({ contentBefore: 'abcd[]ef
', contentBeforeEdit: "ab" + "\ufeff" + // before zwnbsp '' + "\ufeff" + // start zwnbsp "cd" + // content "\ufeff" + // end zwnbsp "" + "\ufeff" + // after zwnbsp "[]ef
", stepFunction: keyPress("ArrowLeft"), contentAfterEdit: "ab" + "\ufeff" + // before zwnbsp '' + "\ufeff" + // start zwnbsp "cd[]" + // content "\ufeff" + // end zwnbsp "" + "\ufeff" + // after zwnbsp "ef
", contentAfter: 'abcd[]ef
', }); }); test("should move out of a link (ArrowRight)", async () => { await testEditor({ contentBefore: 'abcd[]ef
', contentBeforeEdit: "ab" + "\ufeff" + // before zwnbsp '' + "\ufeff" + // start zwnbsp "cd[]" + // content "\ufeff" + // end zwnbsp "" + "\ufeff" + // after zwnbsp "ef
", stepFunction: keyPress("ArrowRight"), contentAfterEdit: "ab" + "\ufeff" + // before zwnbsp '' + "\ufeff" + // start zwnbsp "cd" + // content "\ufeff" + // end zwnbsp "" + "\ufeff" + // after zwnbsp "[]ef
", contentAfter: 'abcd[]ef
', }); }); test("should move out of a link (ArrowLeft)", async () => { await testEditor({ contentBefore: 'ab[]cdef
', contentBeforeEdit: "ab" + "\ufeff" + // before zwnbsp '' + "\ufeff" + // start zwnbsp "[]cd" + // content "\ufeff" + // end zwnbsp "" + "\ufeff" + // after zwnbsp "ef
", stepFunction: keyPress("ArrowLeft"), contentAfterEdit: "ab[]" + "\ufeff" + // before zwnbsp '' + "\ufeff" + // start zwnbsp "cd" + // content "\ufeff" + // end zwnbsp "" + "\ufeff" + // after zwnbsp "ef
", contentAfter: 'ab[]cdef
', }); }); }); describe("Around icons", () => { test("should move past the icon (ArrowRight)", async () => { await testEditor({ contentBefore: `abc[]def
`, contentBeforeEdit: `abc[]\u200bdef
`, stepFunction: keyPress("ArrowRight"), contentAfterEdit: `abc\u200b[]def
`, contentAfter: `abc[]def
`, }); }); test("should move past the icon (ArrowLeft)", async () => { await testEditor({ contentBefore: `abc[]def
`, contentBeforeEdit: `abc\u200b[]def
`, stepFunction: keyPress("ArrowLeft"), contentAfterEdit: `abc[]\u200bdef
`, contentAfter: `abc[]def
`, }); }); }); describe("Selection correction when it lands at the editable root", () => { test("should place cursor in the table below", async () => { await testEditor({ contentBefore: "a b[] |
c d |
a b |
[]c d |
a b |
[]c d |
a b[] |
c d |
a b[] |
a b |
[]
[]a b |
[]
a b |
a b[] |
a b[] |
[]a b |
[]a b |
[]
[]
[]
[]
" + "الرجال" + '\u200B' + "هؤلاء" + "
"; // Displayed as " هؤلاء\u200Bالرجال" in the editor: // third + span + first test("should move past the zws (ArrowLeft)", async () => { const { editor, el } = await setupEditor(content, { config: { direction: "rtl" } }); const pFirstChild = el.firstChild.firstChild; // "الرجال" const pThirdChild = el.firstChild.childNodes[2]; // "هؤلاء" // Place cursor at the end of first child (next to the span) // Displayed as هؤلاء\u200B[]الرجال setSelection({ anchorNode: pFirstChild, anchorOffset: pFirstChild.length }); await keyPress("ArrowLeft")(editor); const selection = editor.document.getSelection(); expect(selection.anchorNode).toBe(pThirdChild); expect(selection.anchorOffset).toBe(1); // Displayed as ه[]ؤلاء\u200Bالرجال expect(getContent(el)).toBe('الرجال\u200Bه[]ؤلاء
'); }); test("should move past the zws (ArrowRight)", async () => { const { editor, el } = await setupEditor(content, { config: { direction: "rtl" } }); const pFirstChild = el.firstChild.firstChild; // "الرجال" const pThirdChild = el.firstChild.childNodes[2]; // "هؤلاء" // Place cursor at the beginning of third child (next to the span) // Displayed as هؤلاء[]\u200Bالرجال setSelection({ anchorNode: pThirdChild, anchorOffset: 0 }); await keyPress("ArrowRight")(editor); const selection = editor.document.getSelection(); expect(selection.anchorNode).toBe(pFirstChild); expect(selection.anchorOffset).toBe(pFirstChild.length - 1); // Displayed as هؤلاء\u200Bالرجا[]ل expect(getContent(el)).toBe('الرجا[]ل\u200Bهؤلاء
'); }); }); describe("ZWNBSP", () => { const content = "" + "الرجال" + 'اءيتجنب' + "هؤلاء" + "
"; // Displayed as "هؤلاءاءيتجنبالرجال" in the editor: // third + link + first test("should move into a link (ArrowLeft)", async () => { const { editor, el } = await setupEditor(content, { config: { direction: "rtl" } }); const pFirstChild = el.firstChild.firstChild; // "الرجال" // childNodes[1] and childNodes[3] are the ZWNBSP text nodes const link = el.firstChild.childNodes[2]; // Place cursor at the end of first child (before the FEFF char) // Displayed as هؤلاء\uFEFF\uFEFFاءيتجنب\uFEFF\uFEFF[]الرجال setSelection({ anchorNode: pFirstChild, anchorOffset: pFirstChild.length }); await keyPress("ArrowLeft")(editor); const selection = editor.document.getSelection(); expect(selection.anchorNode).toBe(link.firstChild); // FEFF node expect(selection.anchorOffset).toBe(1); // Displayed as هؤلاء\uFEFF\uFEFFاءيتجنب[]\uFEFF\uFEFFالرجال expect(getContent(el)).toBe( 'الرجال\uFEFF\uFEFF[]اءيتجنب\uFEFF\uFEFFهؤلاء
' ); }); test("should move into a link (ArrowRight)", async () => { const { editor, el } = await setupEditor(content, { config: { direction: "rtl" } }); // childNodes[1] and childNodes[3] are the ZWNBSP text nodes const link = el.firstChild.childNodes[2]; const pFifthChild = el.firstChild.childNodes[4]; // "هؤلاء" const link2ndChild = link.childNodes[1]; // اءيتجنب // Place cursor at the beginning of fifth child (after the FEFF char) // Displayed as هؤلاء[]\uFEFF\uFEFFاءيتجنب\uFEFF\uFEFFالرجال setSelection({ anchorNode: pFifthChild, anchorOffset: 0 }); await keyPress("ArrowRight")(editor); const selection = editor.document.getSelection(); expect(selection.anchorNode).toBe(link2ndChild); expect(selection.anchorOffset).toBe(link2ndChild.length); // Displayed as هؤلاء\uFEFF\uFEFF[]اءيتجنب\uFEFF\uFEFFالرجال expect(getContent(el)).toBe( 'الرجال\uFEFF\uFEFFاءيتجنب[]\uFEFF\uFEFFهؤلاء
' ); }); test("should move out of a link (ArrowLeft)", async () => { const { editor, el } = await setupEditor(content, { config: { direction: "rtl" } }); // childNodes[1] and childNodes[3] are the ZWNBSP text nodes const link = el.firstChild.childNodes[2]; const link2ndChild = link.childNodes[1]; // text content inside link: اءيتجنب // Place cursor at the end of link's content (before the FEFF char) // Displayed as هؤلاء\uFEFF\uFEFF[]اءيتجنب\uFEFF\uFEFFالرجال setSelection({ anchorNode: link2ndChild, anchorOffset: link2ndChild.length }); await keyPress("ArrowLeft")(editor); const selection = editor.document.getSelection(); expect(selection.anchorNode).toBe(el.firstChild.childNodes[3]); // FEFF node outside link expect(selection.anchorOffset).toBe(1); // Displayed as هؤلاء[]\uFEFF\uFEFFاءيتجنب\uFEFF\uFEFFالرجال expect(getContent(el)).toBe( 'الرجال\uFEFF\uFEFFاءيتجنب\uFEFF\uFEFF[]هؤلاء
' ); }); test("should move out of a link (ArrowRight)", async () => { const { editor, el } = await setupEditor(content, { config: { direction: "rtl" } }); // childNodes[1] and childNodes[3] are the ZWNBSP text nodes const pFirstChild = el.firstChild.firstChild; // "الرجال" const link = el.firstChild.childNodes[2]; const link2ndChild = link.childNodes[1]; // text content inside link: اءيتجنب // Place cursor at the beginning of link's content (after the FEFF char) // Displayed as هؤلاء\uFEFF\uFEFFاءيتجنب[]\uFEFF\uFEFFالرجال setSelection({ anchorNode: link2ndChild, anchorOffset: 0 }); await keyPress("ArrowRight")(editor); const selection = editor.document.getSelection(); expect(selection.anchorNode).toBe(pFirstChild); expect(selection.anchorOffset).toBe(pFirstChild.length); // Displayed as هؤلاء\uFEFF\uFEFFاءيتجنب\uFEFF\uFEFF[]الرجال expect(getContent(el)).toBe( 'الرجال[]\uFEFF\uFEFFاءيتجنب\uFEFF\uFEFFهؤلاء
' ); }); }); }); describe("Around contenteditable false elements containing contenteditable true elements", () => { test("should select contenteditable false element (ArrowRight)", async () => { await testEditor({ contentBefore: unformat(`abc
de[f]
ghi
jkl
mno
`), stepFunction: () => press(["shift", "arrowright"]), contentAfterEdit: unformat(`abc
de[f
ghi
jkl
]mno
`), }); }); test("should select contenteditable false element (ArrowLeft)", async () => { await testEditor({ contentBefore: unformat(`abc
def
ghi
jkl
]m[no
`), stepFunction: () => press(["shift", "arrowleft"]), contentAfter: unformat(`abc
def]
ghi
jkl
m[no
`), }); }); test("should select contenteditable false element (ArrowUp)", async () => { await testEditor({ contentBefore: unformat(`abc
def
ghi
jkl
]mno[
`), stepFunction: () => press(["shift", "arrowup"]), contentAfter: unformat(`abc
def]
ghi
jkl
mno[
`), }); }); test("should select contenteditable false element (ArrowDown)", async () => { await testEditor({ contentBefore: unformat(`abc
[def]
ghi
jkl
mno
`), stepFunction: () => press(["shift", "arrowdown"]), contentAfter: unformat(`abc
[def
ghi
jkl
]mno
`), }); }); });