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

", stepFunction: keyPress("ArrowRight"), contentAfter: "

a

b

" + "

[]c

d

", }); }); test.tags("focus required"); test("should place cursor in the table above", async () => { await testEditor({ contentBefore: "

a

b

" + "

[]c

d

", stepFunction: keyPress("ArrowLeft"), contentAfter: "

a

b[]

" + "

c

d

", }); }); test("should place cursor in the paragraph below", async () => { await testEditor({ contentBefore: "

a

b[]

" + "


", stepFunction: keyPress("ArrowRight"), contentAfter: "

a

b

" + "

[]

", }); }); test("should place cursor in the paragraph above", async () => { await testEditor({ contentBefore: "


" + "

[]a

b

", stepFunction: keyPress("ArrowLeft"), contentAfter: "

[]

" + "

a

b

", }); }); test("should keep cursor at the same position (avoid reaching the editable root) (1)", async () => { await testEditor({ contentBefore: "

a

b[]

", stepFunction: keyPress("ArrowRight"), contentAfter: "

a

b[]

", }); }); test("should keep cursor at the same position (avoid reaching the editable root) (2)", async () => { await testEditor({ contentBefore: "

[]a

b

", stepFunction: keyPress("ArrowLeft"), contentAfter: "

[]a

b

", }); }); test("should place cursor after the second separator", async () => { await testEditor({ contentBefore: '

[]


' + '


', stepFunction: keyPress("ArrowRight"), contentAfter: "



" + "

[]

", }); }); test.tags("focus required"); test("should place cursor before the first separator", async () => { await testEditor({ contentBefore: '



' + '

[]

', stepFunction: keyPress("ArrowLeft"), contentAfter: "

[]


" + "


", }); }); }); describe.tags("focus required"); describe("Around invisible chars in RTL languages", () => { describe("ZWS", () => { const content = "

" + "الرجال" + '\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

`), }); }); });