Skip to content
This repository has been archived by the owner on Sep 11, 2024. It is now read-only.

Commands for plain text editor #10567

Merged
merged 76 commits into from
Apr 27, 2023
Merged
Show file tree
Hide file tree
Changes from 15 commits
Commits
Show all changes
76 commits
Select commit Hold shift + click to select a range
b305cc9
add the handlers for when autocomplete is open plus rough / handling
Apr 11, 2023
f092fb1
hack in using the wysiwyg autocomplete
Apr 11, 2023
86883f8
switch to using onSelect for the behaviour
Apr 11, 2023
d300050
expand comment
Apr 11, 2023
092fd71
add a handle command function to replace text
Apr 11, 2023
9f2e7f1
add event firing step
Apr 11, 2023
e29b2e4
Merge remote-tracking branch 'origin/develop' into alunturner/mention…
Apr 11, 2023
26b995d
Merge remote-tracking branch 'origin/develop' into alunturner/mention…
Apr 13, 2023
47d0d53
fix TS errors for RefObject
Apr 13, 2023
ac776b4
extract common functionality to new util
Apr 13, 2023
7de5f45
use util for plain text mode
Apr 13, 2023
5f89098
use util for rich text mode
Apr 13, 2023
ba8999c
remove unused imports
Apr 13, 2023
46f5309
make util able to handle either type of keyboard event
Apr 13, 2023
f857c14
fix TS error for mxClient
Apr 13, 2023
26baf97
lift all new code into main component prior to extracting to custom hook
Apr 13, 2023
380e265
shift logic into custom hook
Apr 13, 2023
e0fa7dd
rename ref to editorRef for clarity
Apr 13, 2023
76f46b4
remove comment
Apr 13, 2023
5d971b3
try to add cypress test for behaviour
Apr 14, 2023
14fd02b
remove unused imports
Apr 14, 2023
e08f450
fix various lint/TS errors for CI
Apr 14, 2023
6a40aa6
update cypress test
Apr 14, 2023
69a6ca4
add test for pressing escape to close autocomplete
Apr 14, 2023
c092b3e
expand cypress tests
Apr 14, 2023
262c7c0
add typing while autocomplete open test
Apr 14, 2023
3e845e4
Merge remote-tracking branch 'origin/develop' into alunturner/mention…
Apr 14, 2023
590650e
refactor to single piece of state and update comments
Apr 14, 2023
7ac2318
update comment
Apr 14, 2023
1d03122
extract functions for testing
Apr 14, 2023
cc357ae
add first tests
Apr 14, 2023
aac6630
improve tests
Apr 18, 2023
80108c7
Merge remote-tracking branch 'origin/develop' into alunturner/mention…
Apr 18, 2023
7e9a230
remove console log
Apr 18, 2023
2eb1203
call useSuggestion hook from different location
Apr 18, 2023
3ad7c45
update useSuggestion hook tests
Apr 18, 2023
9db2a45
improve cypress tests
Apr 18, 2023
b55ba4f
Merge remote-tracking branch 'origin/develop' into alunturner/mention…
Apr 18, 2023
f83d074
remove unused import
Apr 18, 2023
368ce13
fix selector in cypress test
Apr 18, 2023
449521a
add another set of util tests
Apr 18, 2023
c29d945
remove .only
Apr 18, 2023
db43084
remove .only
Apr 18, 2023
138b40e
Merge remote-tracking branch 'origin/develop' into alunturner/mention…
Apr 18, 2023
c7d653e
remove import
Apr 18, 2023
b52eeb2
improve cypress tests
Apr 18, 2023
3198acb
remove .only
Apr 18, 2023
6647bb5
add comment
Apr 18, 2023
750dea4
improve comments
Apr 18, 2023
a467a2b
tidy up tests
Apr 18, 2023
c934f9a
Merge remote-tracking branch 'origin/develop' into alunturner/mention…
Apr 18, 2023
8eb1d6a
Merge branch 'develop' into alunturner/mentions-for-plain-text-editor
artcodespace Apr 18, 2023
aeeca8c
Merge remote-tracking branch 'origin/develop' into alunturner/mention…
Apr 20, 2023
b3ca730
consolidate all cypress tests to one
Apr 20, 2023
cc17e0e
add early return
Apr 20, 2023
d883727
fix typo, add documentation
Apr 20, 2023
1a627ed
add early return, tidy up comments
Apr 20, 2023
bec06f9
change function expression to function declaration
Apr 20, 2023
6fdee58
add documentation
Apr 20, 2023
7948fa7
fix broken test
Apr 20, 2023
26663b5
Merge remote-tracking branch 'origin/develop' into alunturner/mention…
Apr 21, 2023
799f9a2
add check to cypress tests
Apr 21, 2023
eeb3c6c
update types
Apr 21, 2023
ab5abb4
update comment
Apr 21, 2023
9b8ee00
update comments
Apr 21, 2023
88840bf
shift ref declaration inside the hook
Apr 21, 2023
ed1b78c
remove unused import
Apr 21, 2023
d0d2f4d
Merge remote-tracking branch 'origin/develop' into alunturner/mention…
Apr 25, 2023
fe01f99
update cypress test and add comments
Apr 25, 2023
0d1852c
update usePlainTextListener comments
Apr 25, 2023
cbcb170
apply suggested changes to useSuggestion
Apr 25, 2023
6f3b7f0
update tests
Apr 25, 2023
66871b6
Merge remote-tracking branch 'origin/develop' into alunturner/mention…
Apr 25, 2023
da6ae51
Merge branch 'develop' into alunturner/mentions-for-plain-text-editor
Apr 25, 2023
43828d5
Merge branch 'develop' into alunturner/mentions-for-plain-text-editor
t3chguy Apr 26, 2023
3c374e0
Merge branch 'develop' into alunturner/mentions-for-plain-text-editor
artcodespace Apr 27, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
126 changes: 31 additions & 95 deletions cypress/e2e/composer/composer.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -117,128 +117,64 @@ describe("Composer", () => {
cy.viewRoomByName("Composing Room");
});

describe("commands", () => {
describe("Commands", () => {
// TODO add tests for rich text mode

describe("plain text mode", () => {
it("autocomplete opens when / is pressed and contains autocomplete items", () => {
describe("Plain text mode", () => {
it("autocomplete behaviour tests", () => {
// Select plain text mode after composer is ready
cy.get("div[contenteditable=true]").should("exist");
cy.findByRole("button", { name: "Hide formatting" }).click();

// Type a /
// Typing a single / displays the autocomplete menu and contents
cy.findByRole("textbox").type("/");

// Check that the autocomplete options are visible and there are more than 0
cy.findByTestId("autocomplete-wrapper").within(() => {
cy.findAllByRole("presentation").should("have.length.above", 0);
});
});

it("autocomplete can be used to enter a command", () => {
// Select plain text mode after composer is ready
cy.get("div[contenteditable=true]").should("exist");
cy.findByRole("button", { name: "Hide formatting" }).click();
// Check that the autocomplete options are visible and there are more than 0 items
cy.findByTestId("autocomplete-wrapper").should("not.be.empty");

// Type a message
cy.findByRole("textbox").type("/spo");
// Entering `//` or `/ ` hides the autocomplete contents
cy.findByRole("textbox").type(" ");
cy.findByTestId("autocomplete-wrapper").should("be.empty");
cy.findByRole("textbox").type("{Backspace}");
cy.findByTestId("autocomplete-wrapper").should("not.be.empty");
cy.findByRole("textbox").type("{Backspace}/");
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

should we assert that the autocomplete options reappear after backspace?

cy.findByTestId("autocomplete-wrapper").should("be.empty");

// Check that the autocomplete /spoiler option is visible and click it
// Typing a command that takes no arguments (/devtools) and selecting by click works
cy.findByRole("textbox").type("{Backspace}dev");
cy.findByTestId("autocomplete-wrapper").within(() => {
cy.findByText("/spoiler").click();
cy.findByText("/devtools").click();
});

// Check the autocomplete is closed and the composer contains the completion
// Check it has closed the autocomplete and put the text into the composer
cy.findByTestId("autocomplete-wrapper").should("not.be.visible");
cy.findByRole("textbox").within(() => {
cy.findByText("/spoiler").should("exist");
cy.findByText("/devtools").should("exist");
});
});

it("autocomplete can be used to write and send a command that takes arguments", () => {
// Select plain text mode after composer is ready
cy.get("div[contenteditable=true]").should("exist");
cy.findByRole("button", { name: "Hide formatting" }).click();

// Type a message
cy.findByRole("textbox").type("/spo");
// Send the message and check the devtools dialog appeared, then close it
cy.findByRole("button", { name: "Send message" }).click();
cy.findByRole("dialog").within(() => {
cy.findByText("Developer Tools").should("exist");
});
cy.findByRole("button", { name: "Close dialog" }).click();

// Check that the autocomplete /spoiler option is visible and click it
// Typing a command that takes arguments (/spoiler) and selecting with enter works
cy.findByRole("textbox").type("/spoil");
cy.findByTestId("autocomplete-wrapper").within(() => {
cy.findByText("/spoiler").click();
cy.findByText("/spoiler").should("exist");
});

// Check the autocomplete is closed and the composer contains the completion
cy.findByRole("textbox").type("{Enter}");
// Check it has closed the autocomplete and put the text into the composer
cy.findByTestId("autocomplete-wrapper").should("not.be.visible");
cy.findByRole("textbox").within(() => {
cy.findByText("/spoiler").should("exist");
});

// Type some more text then send the message
const argumentText = "this is the spoiler text";
cy.findByRole("textbox").type(argumentText);
// Enter some more text, then send the message
cy.findByRole("textbox").type("this is the spoiler text ");
cy.findByRole("button", { name: "Send message" }).click();

// Check that a spoiler item has appeared in the timeline and contains the spoiler command text
cy.get("span.mx_EventTile_spoiler").should("exist");
cy.findByText("this is the spoiler text").should("exist");
});

it("autocomplete can be used to write and send a command that takes no arguments", () => {
// Select plain text mode after composer is ready
cy.get("div[contenteditable=true]").should("exist");
cy.findByRole("button", { name: "Hide formatting" }).click();

// Type a message
cy.findByRole("textbox").type("/dev");

// Check that the autocomplete /spoiler option is visible and click it
cy.findByTestId("autocomplete-wrapper").within(() => {
cy.findByText("/devtools").click();
});

// Check the autocomplete is closed and the composer contains the completion
cy.findByTestId("autocomplete-wrapper").should("not.be.visible");
cy.findByRole("textbox").within(() => {
cy.findByText("/devtools").should("exist");
});

// Click the send message button
cy.findByRole("button", { name: "Send message" }).click();

// Check that the devtools dialog menu has appeared
cy.findByRole("dialog").within(() => {
cy.findByText("Developer Tools").should("exist");
});
});

it("autocomplete is not displayed for a message starting with //", () => {
// Select plain text mode after composer is ready
cy.get("div[contenteditable=true]").should("exist");
cy.findByRole("button", { name: "Hide formatting" }).click();

// Type a message
cy.findByRole("textbox").type("//anyText");

// Check that the autocomplete options are not visible
cy.findByTestId("autocomplete-wrapper").within(() => {
cy.findAllByRole("presentation").should("have.length", 0);
});
});

it("autocomplete is not displayed when user inserts whitespace after command", () => {
// Select plain text mode after composer is ready
cy.get("div[contenteditable=true]").should("exist");
cy.findByRole("button", { name: "Hide formatting" }).click();

// Type a message
cy.findByRole("textbox").type("/spoiler followed by a space");

// Check that the autocomplete options are not visible
cy.findByTestId("autocomplete-wrapper").within(() => {
cy.findAllByRole("presentation").should("have.length", 0);
});
});
});
});

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ limitations under the License.
*/

import classNames from "classnames";
import React, { MutableRefObject, ReactNode, useRef } from "react";
import React, { MutableRefObject, ReactNode } from "react";

import { useComposerFunctions } from "../hooks/useComposerFunctions";
import { useIsFocused } from "../hooks/useIsFocused";
Expand All @@ -24,7 +24,6 @@ import { usePlainTextListeners } from "../hooks/usePlainTextListeners";
import { useSetCursorPosition } from "../hooks/useSetCursorPosition";
import { ComposerFunctions } from "../types";
import { Editor } from "./Editor";
import Autocomplete from "../../Autocomplete";
import { WysiwygAutocomplete } from "./WysiwygAutocomplete";

interface PlainTextComposerProps {
Expand All @@ -50,10 +49,9 @@ export function PlainTextComposer({
leftComponent,
rightComponent,
}: PlainTextComposerProps): JSX.Element {
const autocompleteRef = useRef<Autocomplete | null>(null);

const {
ref: editorRef,
autocompleteRef,
onInput,
onPaste,
onKeyDown,
Expand All @@ -63,7 +61,7 @@ export function PlainTextComposer({
onSelect,
handleCommand,
handleMention,
} = usePlainTextListeners(autocompleteRef, initialContent, onChange, onSend);
} = usePlainTextListeners(initialContent, onChange, onSend);

const composerFunctions = useComposerFunctions(editorRef, setContent);
usePlainTextInitialization(initialContent, editorRef);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -37,13 +37,29 @@ function amendInnerHtml(text: string): string {
.replace(/<\/div>/g, "");
}

/**
* Generates all of the listeners and the ref to be attached to the editor. Also returns
* pieces of state and utility functions that are required for use in other hooks and by
* the autocomplete component.
richvdh marked this conversation as resolved.
Show resolved Hide resolved
*
* @param initialContent - can set the content of the editor on mount
richvdh marked this conversation as resolved.
Show resolved Hide resolved
* @param onChange - called whenever there is change in the editor content
* @param onSend - called whenever the user sends the message
* @returns
* - `ref`: a ref object which the caller must attach to the HTML `div` node for the editor
* * `autocompleteRef`: a ref object which the caller must attach to the autocomplete component
* - `content`: state representing the editor's current text content
* - `setContent`: the setter for the content state
richvdh marked this conversation as resolved.
Show resolved Hide resolved
* - `onInput`, `onPaste`, `onKeyDown`: handlers for input, paste and keyDown events
* - the output from the {@link useSuggestion} hook
*/
export function usePlainTextListeners(
artcodespace marked this conversation as resolved.
Show resolved Hide resolved
autocompleteRef: React.RefObject<Autocomplete>,
initialContent?: string,
onChange?: (content: string) => void,
onSend?: () => void,
): {
ref: RefObject<HTMLDivElement>;
autocompleteRef: React.RefObject<Autocomplete>;
content?: string;
onInput(event: SyntheticEvent<HTMLDivElement, InputEvent | ClipboardEvent>): void;
onPaste(event: SyntheticEvent<HTMLDivElement, InputEvent | ClipboardEvent>): void;
Expand All @@ -55,6 +71,7 @@ export function usePlainTextListeners(
suggestion: MappedSuggestion | null;
} {
const ref = useRef<HTMLDivElement | null>(null);
const autocompleteRef = useRef<Autocomplete | null>(null);
const [content, setContent] = useState<string | undefined>(initialContent);

const send = useCallback(() => {
Expand Down Expand Up @@ -122,6 +139,7 @@ export function usePlainTextListeners(

return {
ref,
autocompleteRef,
onInput,
onPaste: onInput,
onKeyDown,
Expand Down
Loading