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 827
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add room and user avatars to rte (#10497)
Co-authored-by: Michael Telatynski <7t3chguy@gmail.com>
- Loading branch information
1 parent
5c0e5eb
commit e03eac1
Showing
6 changed files
with
406 additions
and
82 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/utils/autocomplete.ts
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 { Attributes, MappedSuggestion } from "@matrix-org/matrix-wysiwyg"; | ||
import { MatrixClient, Room } from "matrix-js-sdk/src/matrix"; | ||
|
||
import { ICompletion } from "../../../../../autocomplete/Autocompleter"; | ||
import * as Avatar from "../../../../../Avatar"; | ||
|
||
/** | ||
* 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 | ||
*/ | ||
export 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}`; | ||
} | ||
|
||
/** | ||
* Find the room from the completion by looking it up using the client from the context | ||
* we are currently in | ||
* | ||
* @param completion - the completion from the autocomplete | ||
* @param client - the current client we are using | ||
* @returns a Room if one is found, null otherwise | ||
*/ | ||
export function getRoomFromCompletion(completion: ICompletion, client: MatrixClient): Room | null { | ||
const roomId = completion.completionId; | ||
const aliasFromCompletion = completion.completion; | ||
|
||
let roomToReturn: 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) { | ||
roomToReturn = client.getRoom(roomId); | ||
} else if (!aliasFromCompletion.startsWith("#")) { | ||
roomToReturn = client.getRoom(aliasFromCompletion); | ||
} else { | ||
roomToReturn = client.getRooms().find((r) => { | ||
return r.getCanonicalAlias() === aliasFromCompletion || r.getAltAliases().includes(aliasFromCompletion); | ||
}); | ||
} | ||
|
||
return roomToReturn ?? null; | ||
} | ||
|
||
/** | ||
* Given an autocomplete suggestion, determine the text to display in the pill | ||
* | ||
* @param completion - the item selected from the autocomplete | ||
* @param client - the MatrixClient is required for us to look up the correct room mention text | ||
* @returns the text to display in the mention | ||
*/ | ||
export function getMentionDisplayText(completion: ICompletion, client: MatrixClient): string { | ||
if (completion.type === "user") { | ||
return completion.completion; | ||
} else if (completion.type === "room") { | ||
// try and get the room and use it's name, if not available, fall back to | ||
// completion.completion | ||
return getRoomFromCompletion(completion, client)?.name || completion.completion; | ||
} | ||
return ""; | ||
} | ||
|
||
/** | ||
* For a given completion, the attributes will change depending on the completion type | ||
* | ||
* @param completion - the item selected from the autocomplete | ||
* @param client - the MatrixClient is required for us to look up the correct room mention text | ||
* @returns an object of attributes containing HTMLAnchor attributes or data-* attri | ||
*/ | ||
export function getMentionAttributes(completion: ICompletion, client: MatrixClient, room: Room): Attributes { | ||
// to ensure that we always have something set in the --avatar-letter CSS variable | ||
// as otherwise alignment varies depending on whether the content is empty or not | ||
const defaultLetterContent = "-"; | ||
|
||
if (completion.type === "user") { | ||
// logic as used in UserPillPart.setAvatar in parts.ts | ||
const mentionedMember = room.getMember(completion.completionId || ""); | ||
|
||
if (!mentionedMember) return {}; | ||
|
||
const name = mentionedMember.name || mentionedMember.userId; | ||
const defaultAvatarUrl = Avatar.defaultAvatarUrlForString(mentionedMember.userId); | ||
const avatarUrl = Avatar.avatarUrlForMember(mentionedMember, 16, 16, "crop"); | ||
let initialLetter = defaultLetterContent; | ||
if (avatarUrl === defaultAvatarUrl) { | ||
initialLetter = Avatar.getInitialLetter(name) ?? defaultLetterContent; | ||
} | ||
|
||
return { | ||
"data-mention-type": completion.type, | ||
"style": `--avatar-background: url(${avatarUrl}); --avatar-letter: '${initialLetter}'`, | ||
}; | ||
} else if (completion.type === "room") { | ||
// logic as used in RoomPillPart.setAvatar in parts.ts | ||
const mentionedRoom = getRoomFromCompletion(completion, client); | ||
const aliasFromCompletion = completion.completion; | ||
|
||
let initialLetter = defaultLetterContent; | ||
let avatarUrl = Avatar.avatarUrlForRoom(mentionedRoom ?? null, 16, 16, "crop"); | ||
if (!avatarUrl) { | ||
initialLetter = Avatar.getInitialLetter(mentionedRoom?.name || aliasFromCompletion) ?? defaultLetterContent; | ||
avatarUrl = Avatar.defaultAvatarUrlForString(mentionedRoom?.roomId ?? aliasFromCompletion); | ||
} | ||
|
||
return { | ||
"data-mention-type": completion.type, | ||
"style": `--avatar-background: url(${avatarUrl}); --avatar-letter: '${initialLetter}'`, | ||
}; | ||
} | ||
|
||
return {}; | ||
} |
Oops, something went wrong.