This repository has been archived by the owner on Sep 11, 2024. It is now read-only.
-
-
Notifications
You must be signed in to change notification settings - Fork 828
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add mentions as links to rte (#10463)
* adds autocomplete behaviour for mentions in rich text editor * allows adding mentions as links * adds tests
- Loading branch information
1 parent
212977c
commit 6f791d2
Showing
11 changed files
with
585 additions
and
23 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
137 changes: 137 additions & 0 deletions
137
src/components/views/rooms/wysiwyg_composer/components/WysiwygAutocomplete.tsx
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,137 @@ | ||
/* | ||
Copyright 2023 The Matrix.org Foundation C.I.C. | ||
Licensed under the Apache License, Version 2.0 (the "License"); | ||
you may not use this file except in compliance with the License. | ||
You may obtain a copy of the License at | ||
http://www.apache.org/licenses/LICENSE-2.0 | ||
Unless required by applicable law or agreed to in writing, software | ||
distributed under the License is distributed on an "AS IS" BASIS, | ||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
See the License for the specific language governing permissions and | ||
limitations under the License. | ||
*/ | ||
|
||
import React, { ForwardedRef, forwardRef } from "react"; | ||
import { MatrixClient, Room } from "matrix-js-sdk/src/matrix"; | ||
import { FormattingFunctions, MappedSuggestion } from "@matrix-org/matrix-wysiwyg"; | ||
|
||
import { useRoomContext } from "../../../../../contexts/RoomContext"; | ||
import Autocomplete from "../../Autocomplete"; | ||
import { ICompletion } from "../../../../../autocomplete/Autocompleter"; | ||
import { useMatrixClientContext } from "../../../../../contexts/MatrixClientContext"; | ||
|
||
interface WysiwygAutocompleteProps { | ||
/** | ||
* The suggestion output from the rust model is used to build the query that is | ||
* passed to the `<Autocomplete />` component | ||
*/ | ||
suggestion: MappedSuggestion | null; | ||
|
||
/** | ||
* This handler will be called with the href and display text for a mention on clicking | ||
* a mention in the autocomplete list or pressing enter on a selected item | ||
*/ | ||
handleMention: FormattingFunctions["mention"]; | ||
} | ||
|
||
/** | ||
* Builds the query for the `<Autocomplete />` component from the rust suggestion. This | ||
* will change as we implement handling / commands. | ||
* | ||
* @param suggestion - represents if the rust model is tracking a potential mention | ||
* @returns an empty string if we can not generate a query, otherwise a query beginning | ||
* with @ for a user query, # for a room or space query | ||
*/ | ||
function buildQuery(suggestion: MappedSuggestion | null): string { | ||
if (!suggestion || !suggestion.keyChar || suggestion.type === "command") { | ||
// if we have an empty key character, we do not build a query | ||
// TODO implement the command functionality | ||
return ""; | ||
} | ||
|
||
return `${suggestion.keyChar}${suggestion.text}`; | ||
} | ||
|
||
/** | ||
* Given a room type mention, determine the text that should be displayed in the mention | ||
* TODO expand this function to more generally handle outputting the display text from a | ||
* given completion | ||
* | ||
* @param completion - the item selected from the autocomplete, currently treated as a room completion | ||
* @param client - the MatrixClient is required for us to look up the correct room mention text | ||
* @returns the text to display in the mention | ||
*/ | ||
function getRoomMentionText(completion: ICompletion, client: MatrixClient): string { | ||
const roomId = completion.completionId; | ||
const alias = completion.completion; | ||
|
||
let roomForAutocomplete: Room | null | undefined; | ||
|
||
// Not quite sure if the logic here makes sense - specifically calling .getRoom with an alias | ||
// that doesn't start with #, but keeping the logic the same as in PartCreator.roomPill for now | ||
if (roomId) { | ||
roomForAutocomplete = client.getRoom(roomId); | ||
} else if (!alias.startsWith("#")) { | ||
roomForAutocomplete = client.getRoom(alias); | ||
} else { | ||
roomForAutocomplete = client.getRooms().find((r) => { | ||
return r.getCanonicalAlias() === alias || r.getAltAliases().includes(alias); | ||
}); | ||
} | ||
|
||
// if we haven't managed to find the room, use the alias as a fallback | ||
return roomForAutocomplete?.name || alias; | ||
} | ||
|
||
/** | ||
* Given the current suggestion from the rust model and a handler function, this component | ||
* will display the legacy `<Autocomplete />` component (as used in the BasicMessageComposer) | ||
* and call the handler function with the required arguments when a mention is selected | ||
* | ||
* @param props.ref - the ref will be attached to the rendered `<Autocomplete />` component | ||
*/ | ||
const WysiwygAutocomplete = forwardRef( | ||
({ suggestion, handleMention }: WysiwygAutocompleteProps, ref: ForwardedRef<Autocomplete>): JSX.Element | null => { | ||
const { room } = useRoomContext(); | ||
const client = useMatrixClientContext(); | ||
|
||
function handleConfirm(completion: ICompletion): void { | ||
if (!completion.href || !client) return; | ||
|
||
switch (completion.type) { | ||
case "user": | ||
handleMention(completion.href, completion.completion); | ||
break; | ||
case "room": { | ||
handleMention(completion.href, getRoomMentionText(completion, client)); | ||
break; | ||
} | ||
// TODO implement the command functionality | ||
// case "command": | ||
// console.log("/command functionality not yet in place"); | ||
// break; | ||
default: | ||
break; | ||
} | ||
} | ||
|
||
return room ? ( | ||
<div className="mx_SendWysiwygComposer_AutoCompleteWrapper" data-testid="autocomplete-wrapper"> | ||
<Autocomplete | ||
ref={ref} | ||
query={buildQuery(suggestion)} | ||
onConfirm={handleConfirm} | ||
selection={{ start: 0, end: 0 }} | ||
room={room} | ||
/> | ||
</div> | ||
) : null; | ||
}, | ||
); | ||
|
||
WysiwygAutocomplete.displayName = "WysiwygAutocomplete"; | ||
|
||
export { WysiwygAutocomplete }; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.