From b517b94f1a5936715827e7776b9ff19aa531bc87 Mon Sep 17 00:00:00 2001 From: Ajay Bura <32841439+ajbura@users.noreply.github.com> Date: Thu, 19 Oct 2023 10:08:14 +0530 Subject: [PATCH] improve member search --- .../autocomplete/UserMentionAutocomplete.tsx | 32 ++++++++++--------- src/app/hooks/useAsyncSearch.ts | 5 +-- src/app/organisms/room/MembersDrawer.tsx | 15 ++++++--- src/app/organisms/room/RoomInput.tsx | 2 +- .../organisms/room/message/MessageEditor.tsx | 2 +- src/app/utils/room.ts | 10 ++++++ 6 files changed, 43 insertions(+), 23 deletions(-) diff --git a/src/app/components/editor/autocomplete/UserMentionAutocomplete.tsx b/src/app/components/editor/autocomplete/UserMentionAutocomplete.tsx index a99274a5d9..8c3d91bf15 100644 --- a/src/app/components/editor/autocomplete/UserMentionAutocomplete.tsx +++ b/src/app/components/editor/autocomplete/UserMentionAutocomplete.tsx @@ -1,7 +1,7 @@ import React, { useEffect, KeyboardEvent as ReactKeyboardEvent } from 'react'; import { Editor } from 'slate'; import { Avatar, AvatarFallback, AvatarImage, MenuItem, Text, color } from 'folds'; -import { MatrixClient, RoomMember } from 'matrix-js-sdk'; +import { MatrixClient, Room, RoomMember } from 'matrix-js-sdk'; import { AutocompleteQuery } from './autocompleteQuery'; import { AutocompleteMenu } from './AutocompleteMenu'; @@ -16,6 +16,7 @@ import { onTabPress } from '../../../utils/keyboard'; import { createMentionElement, moveCursor, replaceWithElement } from '../utils'; import { useKeyDown } from '../../../hooks/useKeyDown'; import { getMxIdLocalPart, getMxIdServer, validMxId } from '../../../utils/matrix'; +import { getMemberDisplayName, getMemberSearchStr } from '../../../utils/room'; type MentionAutoCompleteHandler = (userId: string, name: string) => void; @@ -64,7 +65,7 @@ function UnknownMentionItem({ } type UserMentionAutocompleteProps = { - roomId: string; + room: Room; editor: Editor; query: AutocompleteQuery; requestClose: () => void; @@ -77,21 +78,19 @@ const SEARCH_OPTIONS: UseAsyncSearchOptions = { }, }; -const getRoomMemberStr: SearchItemStrGetter = (roomMember) => [ - roomMember.name, - getMxIdLocalPart(roomMember.userId) ?? roomMember.userId, - roomMember.userId, -]; +const mxIdToName = (mxId: string) => getMxIdLocalPart(mxId) ?? mxId; +const getRoomMemberStr: SearchItemStrGetter = (m, query) => + getMemberSearchStr(m, query, mxIdToName); export function UserMentionAutocomplete({ - roomId, + room, editor, query, requestClose, }: UserMentionAutocompleteProps) { const mx = useMatrixClient(); - const room = mx.getRoom(roomId); - const roomAliasOrId = room?.getCanonicalAlias() || roomId; + const roomId: string = room.roomId!; + const roomAliasOrId = room.getCanonicalAlias() || roomId; const members = useRoomMembers(mx, roomId); const [result, search, resetSearch] = useAsyncSearch(members, getRoomMemberStr, SEARCH_OPTIONS); @@ -129,6 +128,9 @@ export function UserMentionAutocomplete({ }); }); + const getName = (member: RoomMember) => + getMemberDisplayName(room, member.userId) ?? getMxIdLocalPart(member.userId) ?? member.userId; + return ( Mentions} requestClose={requestClose}> {query.text === 'room' && ( @@ -155,9 +157,9 @@ export function UserMentionAutocomplete({ as="button" radii="300" onKeyDown={(evt: ReactKeyboardEvent) => - onTabPress(evt, () => handleAutocomplete(roomMember.userId, roomMember.name)) + onTabPress(evt, () => handleAutocomplete(roomMember.userId, getName(roomMember))) } - onClick={() => handleAutocomplete(roomMember.userId, roomMember.name)} + onClick={() => handleAutocomplete(roomMember.userId, getName(roomMember))} after={ {roomMember.userId} @@ -166,7 +168,7 @@ export function UserMentionAutocomplete({ before={ {avatarUrl ? ( - + ) : ( - {roomMember.name[0] || roomMember.userId[1]} + {getName(roomMember)[0]} )} } > - {roomMember.name} + {getName(roomMember)} ); diff --git a/src/app/hooks/useAsyncSearch.ts b/src/app/hooks/useAsyncSearch.ts index d0e73e7f8a..719e0d5618 100644 --- a/src/app/hooks/useAsyncSearch.ts +++ b/src/app/hooks/useAsyncSearch.ts @@ -17,7 +17,8 @@ export type UseAsyncSearchOptions = AsyncSearchOption & { }; export type SearchItemStrGetter = ( - searchItem: TSearchItem + searchItem: TSearchItem, + query: string ) => string | string[]; export type UseAsyncSearchResult = { @@ -38,7 +39,7 @@ export const useAsyncSearch = ( setResult(undefined); const handleMatch: MatchHandler = (item, query) => { - const itemStr = getItemStr(item); + const itemStr = getItemStr(item, query); if (Array.isArray(itemStr)) return !!itemStr.find((i) => matchQuery(normalize(i, options?.normalizeOptions), query, options?.matchOptions) diff --git a/src/app/organisms/room/MembersDrawer.tsx b/src/app/organisms/room/MembersDrawer.tsx index 7784afd399..365dc62d67 100644 --- a/src/app/organisms/room/MembersDrawer.tsx +++ b/src/app/organisms/room/MembersDrawer.tsx @@ -46,13 +46,17 @@ import { } from '../../hooks/useIntersectionObserver'; import { Membership } from '../../../types/matrix/room'; import { UseStateProvider } from '../../components/UseStateProvider'; -import { UseAsyncSearchOptions, useAsyncSearch } from '../../hooks/useAsyncSearch'; +import { + SearchItemStrGetter, + UseAsyncSearchOptions, + useAsyncSearch, +} from '../../hooks/useAsyncSearch'; import { useDebounce } from '../../hooks/useDebounce'; import colorMXID from '../../../util/colorMXID'; import { usePowerLevelTags, PowerLevelTag } from '../../hooks/usePowerLevelTags'; import { roomIdToTypingMembersAtom, selectRoomTypingMembersAtom } from '../../state/typingMembers'; import { TypingIndicator } from '../../components/typing-indicator'; -import { getMemberDisplayName } from '../../utils/room'; +import { getMemberDisplayName, getMemberSearchStr } from '../../utils/room'; import { getMxIdLocalPart } from '../../utils/matrix'; import { useSetting } from '../../state/hooks/settings'; import { settingsAtom } from '../../state/settings'; @@ -161,7 +165,10 @@ const SEARCH_OPTIONS: UseAsyncSearchOptions = { contain: true, }, }; -const getMemberItemStr = (m: RoomMember) => [m.name, m.userId]; + +const mxIdToName = (mxId: string) => getMxIdLocalPart(mxId) ?? mxId; +const getRoomMemberStr: SearchItemStrGetter = (m, query) => + getMemberSearchStr(m, query, mxIdToName); type MembersDrawerProps = { room: Room; @@ -200,7 +207,7 @@ export function MembersDrawer({ room }: MembersDrawerProps) { const [result, search, resetSearch] = useAsyncSearch( filteredMembers, - getMemberItemStr, + getRoomMemberStr, SEARCH_OPTIONS ); if (!result && searchInputRef.current?.value) search(searchInputRef.current.value); diff --git a/src/app/organisms/room/RoomInput.tsx b/src/app/organisms/room/RoomInput.tsx index 81c29b0304..ae12afb05f 100644 --- a/src/app/organisms/room/RoomInput.tsx +++ b/src/app/organisms/room/RoomInput.tsx @@ -443,7 +443,7 @@ export const RoomInput = forwardRef( )} {autocompleteQuery?.prefix === AutocompletePrefix.UserMention && ( ( )} {autocompleteQuery?.prefix === AutocompletePrefix.UserMention && ( string +): string[] => [ + member.rawDisplayName === member.userId ? mxIdToName(member.userId) : member.rawDisplayName, + query.startsWith('@') || query.indexOf(':') > -1 ? member.userId : mxIdToName(member.userId), +]; + export const getMemberAvatarMxc = (room: Room, userId: string): string | undefined => { const member = room.getMember(userId); return member?.getMxcAvatarUrl();