diff --git a/lib/components/chat-mention-provider.react.js b/lib/components/chat-mention-provider.react.js
new file mode 100644
--- /dev/null
+++ b/lib/components/chat-mention-provider.react.js
@@ -0,0 +1,227 @@
+// @flow
+
+import * as React from 'react';
+
+import genesis from '../facts/genesis.js';
+import { threadInfoSelector } from '../selectors/thread-selectors.js';
+import SentencePrefixSearchIndex from '../shared/sentence-prefix-search-index.js';
+import { threadTypes } from '../types/thread-types-enum.js';
+import type {
+ ChatMentionCandidates,
+ ChatMentionCandidatesObj,
+ ResolvedThreadInfo,
+ ThreadInfo,
+} from '../types/thread-types.js';
+import { useResolvedThreadInfosObj } from '../utils/entity-helpers.js';
+import { useSelector } from '../utils/redux-utils.js';
+
+type Props = {
+ +children: React.Node,
+};
+export type ChatMentionContextType = {
+ +getChatMentionSearchIndex: (
+ threadInfo: ThreadInfo,
+ ) => SentencePrefixSearchIndex,
+ +communityThreadIDForGenesisThreads: { +[id: string]: string },
+ +chatMentionCandidatesObj: ChatMentionCandidatesObj,
+};
+
+const emptySearchIndex = new SentencePrefixSearchIndex();
+const ChatMentionContext: React.Context =
+ React.createContext({
+ getChatMentionSearchIndex: () => emptySearchIndex,
+ communityThreadIDForGenesisThreads: {},
+ chatMentionCandidatesObj: {},
+ });
+
+function ChatMentionContextProvider(props: Props): React.Node {
+ const { children } = props;
+
+ const { communityThreadIDForGenesisThreads, chatMentionCandidatesObj } =
+ useChatMentionCandidatesObjAndUtils();
+ const searchIndices = useChatMentionSearchIndex(chatMentionCandidatesObj);
+
+ const getChatMentionSearchIndex = React.useCallback(
+ (threadInfo: ThreadInfo) => {
+ if (threadInfo.community === genesis.id) {
+ return searchIndices[communityThreadIDForGenesisThreads[threadInfo.id]];
+ }
+ return searchIndices[threadInfo.community ?? threadInfo.id];
+ },
+ [communityThreadIDForGenesisThreads, searchIndices],
+ );
+
+ const value = React.useMemo(
+ () => ({
+ getChatMentionSearchIndex,
+ communityThreadIDForGenesisThreads,
+ chatMentionCandidatesObj,
+ }),
+ [
+ getChatMentionSearchIndex,
+ communityThreadIDForGenesisThreads,
+ chatMentionCandidatesObj,
+ ],
+ );
+
+ return (
+
+ {children}
+
+ );
+}
+
+function getChatMentionCandidates(threadInfos: {
+ +[id: string]: ResolvedThreadInfo,
+}): {
+ chatMentionCandidatesObj: ChatMentionCandidatesObj,
+ communityThreadIDForGenesisThreads: { +[id: string]: string },
+} {
+ const result = {};
+ const visitedGenesisThreads = new Set();
+ const communityThreadIDForGenesisThreads = {};
+ for (const currentThreadID in threadInfos) {
+ const currentThreadInfo = threadInfos[currentThreadID];
+ const { community: currentThreadCommunity } = currentThreadInfo;
+ if (!currentThreadCommunity) {
+ if (!result[currentThreadID]) {
+ result[currentThreadID] = { [currentThreadID]: currentThreadInfo };
+ }
+ continue;
+ }
+ if (!result[currentThreadCommunity]) {
+ result[currentThreadCommunity] = {
+ [currentThreadCommunity]: threadInfos[currentThreadCommunity],
+ };
+ }
+ // Handle GENESIS community case: mentioning inside GENESIS should only
+ // show chats and threads inside the top level that is below GENESIS.
+ if (threadInfos[currentThreadCommunity].type === threadTypes.GENESIS) {
+ if (visitedGenesisThreads.has(currentThreadID)) {
+ continue;
+ }
+ const threadTraversePath = [currentThreadInfo];
+ visitedGenesisThreads.add(currentThreadID);
+ let currentlySelectedThreadID = currentThreadInfo.parentThreadID;
+ while (currentlySelectedThreadID) {
+ const currentlySelectedThreadInfo =
+ threadInfos[currentlySelectedThreadID];
+ if (
+ visitedGenesisThreads.has(currentlySelectedThreadID) ||
+ !currentlySelectedThreadInfo ||
+ currentlySelectedThreadInfo.type === threadTypes.GENESIS
+ ) {
+ break;
+ }
+ threadTraversePath.push(currentlySelectedThreadInfo);
+ visitedGenesisThreads.add(currentlySelectedThreadID);
+ currentlySelectedThreadID = currentlySelectedThreadInfo.parentThreadID;
+ }
+ const lastThreadInTraversePath =
+ threadTraversePath[threadTraversePath.length - 1];
+ let lastThreadInTraversePathParentID;
+ if (lastThreadInTraversePath.parentThreadID) {
+ lastThreadInTraversePathParentID = threadInfos[
+ lastThreadInTraversePath.parentThreadID
+ ]
+ ? lastThreadInTraversePath.parentThreadID
+ : lastThreadInTraversePath.id;
+ } else {
+ lastThreadInTraversePathParentID = lastThreadInTraversePath.id;
+ }
+ if (
+ threadInfos[lastThreadInTraversePathParentID].type ===
+ threadTypes.GENESIS
+ ) {
+ if (!result[lastThreadInTraversePath.id]) {
+ result[lastThreadInTraversePath.id] = {};
+ }
+ for (const threadInfo of threadTraversePath) {
+ result[lastThreadInTraversePath.id][threadInfo.id] = threadInfo;
+ communityThreadIDForGenesisThreads[threadInfo.id] =
+ lastThreadInTraversePath.id;
+ }
+ if (
+ lastThreadInTraversePath.type !== threadTypes.PERSONAL &&
+ lastThreadInTraversePath.type !== threadTypes.PRIVATE
+ ) {
+ result[genesis.id][lastThreadInTraversePath.id] =
+ lastThreadInTraversePath;
+ }
+ } else {
+ if (
+ !communityThreadIDForGenesisThreads[lastThreadInTraversePathParentID]
+ ) {
+ result[lastThreadInTraversePathParentID] = {};
+ communityThreadIDForGenesisThreads[lastThreadInTraversePathParentID] =
+ lastThreadInTraversePathParentID;
+ }
+ const lastThreadInTraversePathParentCommunityThreadID =
+ communityThreadIDForGenesisThreads[lastThreadInTraversePathParentID];
+ for (const threadInfo of threadTraversePath) {
+ result[lastThreadInTraversePathParentCommunityThreadID][
+ threadInfo.id
+ ] = threadInfo;
+ communityThreadIDForGenesisThreads[threadInfo.id] =
+ lastThreadInTraversePathParentCommunityThreadID;
+ }
+ }
+ continue;
+ }
+ result[currentThreadCommunity][currentThreadID] = currentThreadInfo;
+ }
+ return {
+ chatMentionCandidatesObj: result,
+ communityThreadIDForGenesisThreads,
+ };
+}
+
+function useChatMentionCandidatesObjAndUtils(): {
+ chatMentionCandidatesObj: ChatMentionCandidatesObj,
+ resolvedThreadInfos: ChatMentionCandidates,
+ communityThreadIDForGenesisThreads: { +[id: string]: string },
+} {
+ const threadInfos = useSelector(threadInfoSelector);
+ const resolvedThreadInfos = useResolvedThreadInfosObj(threadInfos);
+ const { chatMentionCandidatesObj, communityThreadIDForGenesisThreads } =
+ React.useMemo(
+ () => getChatMentionCandidates(resolvedThreadInfos),
+ [resolvedThreadInfos],
+ );
+ return {
+ chatMentionCandidatesObj,
+ resolvedThreadInfos,
+ communityThreadIDForGenesisThreads,
+ };
+}
+
+function useChatMentionSearchIndex(
+ chatMentionCandidatesObj: ChatMentionCandidatesObj,
+): {
+ +[id: string]: SentencePrefixSearchIndex,
+} {
+ return React.useMemo(() => {
+ const result = {};
+ for (const communityThreadID in chatMentionCandidatesObj) {
+ const searchIndex = new SentencePrefixSearchIndex();
+ const searchIndexEntries = [];
+ for (const threadID in chatMentionCandidatesObj[communityThreadID]) {
+ searchIndexEntries.push({
+ id: threadID,
+ uiName: chatMentionCandidatesObj[communityThreadID][threadID].uiName,
+ });
+ }
+ // Sort the keys so that the order of the search result is consistent
+ searchIndexEntries.sort(({ uiName: uiNameA }, { uiName: uiNameB }) =>
+ uiNameA.localeCompare(uiNameB),
+ );
+ for (const { id, uiName } of searchIndexEntries) {
+ searchIndex.addEntry(id, uiName);
+ }
+ result[communityThreadID] = searchIndex;
+ }
+ return result;
+ }, [chatMentionCandidatesObj]);
+}
+
+export { ChatMentionContextProvider, ChatMentionContext };
diff --git a/lib/hooks/chat-mention-hooks.js b/lib/hooks/chat-mention-hooks.js
new file mode 100644
--- /dev/null
+++ b/lib/hooks/chat-mention-hooks.js
@@ -0,0 +1,49 @@
+// @flow
+
+import invariant from 'invariant';
+import * as React from 'react';
+
+import {
+ ChatMentionContext,
+ type ChatMentionContextType,
+} from '../components/chat-mention-provider.react.js';
+import genesis from '../facts/genesis.js';
+import type {
+ ChatMentionCandidates,
+ ThreadInfo,
+} from '../types/thread-types.js';
+
+function useChatMentionContext(): ChatMentionContextType {
+ const context = React.useContext(ChatMentionContext);
+ invariant(context, 'ChatMentionContext not found');
+
+ return context;
+}
+
+function useThreadChatMentionCandidates(
+ threadInfo: ThreadInfo,
+): ChatMentionCandidates {
+ const { communityThreadIDForGenesisThreads, chatMentionCandidatesObj } =
+ useChatMentionContext();
+ return React.useMemo(() => {
+ let communityID,
+ result = {};
+ if (threadInfo.community === genesis.id) {
+ communityID = communityThreadIDForGenesisThreads[threadInfo.id];
+ } else {
+ communityID = threadInfo.community ?? threadInfo.id;
+ }
+ if (chatMentionCandidatesObj[communityID]) {
+ result = { ...chatMentionCandidatesObj[communityID] };
+ }
+ delete result[threadInfo.id];
+ return result;
+ }, [
+ chatMentionCandidatesObj,
+ communityThreadIDForGenesisThreads,
+ threadInfo.community,
+ threadInfo.id,
+ ]);
+}
+
+export { useThreadChatMentionCandidates, useChatMentionContext };
diff --git a/lib/shared/thread-utils.js b/lib/shared/thread-utils.js
--- a/lib/shared/thread-utils.js
+++ b/lib/shared/thread-utils.js
@@ -45,7 +45,6 @@
usersWithPersonalThreadSelector,
} from '../selectors/user-selectors.js';
import { getUserAvatarForThread } from '../shared/avatar-utils.js';
-import SentencePrefixSearchIndex from '../shared/sentence-prefix-search-index.js';
import type { CalendarQuery } from '../types/entry-types.js';
import { messageTypes } from '../types/message-types-enum.js';
import {
@@ -79,9 +78,6 @@
type ClientNewThreadRequest,
type NewThreadResult,
type ChangeThreadSettingsPayload,
- type ResolvedThreadInfo,
- type ChatMentionCandidatesObj,
- type ChatMentionCandidates,
type UserProfileThreadInfo,
} from '../types/thread-types.js';
import { updateTypes } from '../types/update-types-enum.js';
@@ -99,7 +95,6 @@
} from '../utils/action-utils.js';
import type { DispatchActionPromise } from '../utils/action-utils.js';
import type { GetENSNames } from '../utils/ens-helpers.js';
-import { useResolvedThreadInfosObj } from '../utils/entity-helpers.js';
import {
ET,
entityTextToRawString,
@@ -1639,161 +1634,6 @@
} within this ${communityOrThreadNoun(threadInfo)}`;
}
-function getChatMentionCandidates(threadInfos: {
- +[id: string]: ResolvedThreadInfo,
-}): {
- chatMentionCandidatesObj: ChatMentionCandidatesObj,
- communityThreadIDForGenesisThreads: { +[id: string]: string },
-} {
- const result = {};
- const visitedGenesisThreads = new Set();
- const communityThreadIDForGenesisThreads = {};
- for (const currentThreadID in threadInfos) {
- const currentThreadInfo = threadInfos[currentThreadID];
- const { community: currentThreadCommunity } = currentThreadInfo;
- if (!currentThreadCommunity) {
- if (!result[currentThreadID]) {
- result[currentThreadID] = { [currentThreadID]: currentThreadInfo };
- }
- continue;
- }
- if (!result[currentThreadCommunity]) {
- result[currentThreadCommunity] = {
- [currentThreadCommunity]: threadInfos[currentThreadCommunity],
- };
- }
- // Handle GENESIS community case: mentioning inside GENESIS should only
- // show chats and threads inside the top level that is below GENESIS.
- if (threadInfos[currentThreadCommunity].type === threadTypes.GENESIS) {
- if (visitedGenesisThreads.has(currentThreadID)) {
- continue;
- }
- const threadTraversePath = [currentThreadInfo];
- visitedGenesisThreads.add(currentThreadID);
- let currentlySelectedThreadID = currentThreadInfo.parentThreadID;
- while (currentlySelectedThreadID) {
- const currentlySelectedThreadInfo =
- threadInfos[currentlySelectedThreadID];
- if (
- visitedGenesisThreads.has(currentlySelectedThreadID) ||
- !currentlySelectedThreadInfo ||
- currentlySelectedThreadInfo.type === threadTypes.GENESIS
- ) {
- break;
- }
- threadTraversePath.push(currentlySelectedThreadInfo);
- visitedGenesisThreads.add(currentlySelectedThreadID);
- currentlySelectedThreadID = currentlySelectedThreadInfo.parentThreadID;
- }
- const lastThreadInTraversePath =
- threadTraversePath[threadTraversePath.length - 1];
- let lastThreadInTraversePathParentID;
- if (lastThreadInTraversePath.parentThreadID) {
- lastThreadInTraversePathParentID = threadInfos[
- lastThreadInTraversePath.parentThreadID
- ]
- ? lastThreadInTraversePath.parentThreadID
- : lastThreadInTraversePath.id;
- } else {
- lastThreadInTraversePathParentID = lastThreadInTraversePath.id;
- }
- if (
- threadInfos[lastThreadInTraversePathParentID].type ===
- threadTypes.GENESIS
- ) {
- if (!result[lastThreadInTraversePath.id]) {
- result[lastThreadInTraversePath.id] = {};
- }
- for (const threadInfo of threadTraversePath) {
- result[lastThreadInTraversePath.id][threadInfo.id] = threadInfo;
- communityThreadIDForGenesisThreads[threadInfo.id] =
- lastThreadInTraversePath.id;
- }
- if (
- lastThreadInTraversePath.type !== threadTypes.PERSONAL &&
- lastThreadInTraversePath.type !== threadTypes.PRIVATE
- ) {
- result[genesis.id][lastThreadInTraversePath.id] =
- lastThreadInTraversePath;
- }
- } else {
- if (
- !communityThreadIDForGenesisThreads[lastThreadInTraversePathParentID]
- ) {
- result[lastThreadInTraversePathParentID] = {};
- communityThreadIDForGenesisThreads[lastThreadInTraversePathParentID] =
- lastThreadInTraversePathParentID;
- }
- const lastThreadInTraversePathParentCommunityThreadID =
- communityThreadIDForGenesisThreads[lastThreadInTraversePathParentID];
- for (const threadInfo of threadTraversePath) {
- result[lastThreadInTraversePathParentCommunityThreadID][
- threadInfo.id
- ] = threadInfo;
- communityThreadIDForGenesisThreads[threadInfo.id] =
- lastThreadInTraversePathParentCommunityThreadID;
- }
- }
- continue;
- }
- result[currentThreadCommunity][currentThreadID] = currentThreadInfo;
- }
- return {
- chatMentionCandidatesObj: result,
- communityThreadIDForGenesisThreads,
- };
-}
-
-function useChatMentionCandidatesObjAndUtils(): {
- chatMentionCandidatesObj: ChatMentionCandidatesObj,
- resolvedThreadInfos: ChatMentionCandidates,
- communityThreadIDForGenesisThreads: { +[id: string]: string },
-} {
- const threadInfos = useSelector(threadInfoSelector);
- const resolvedThreadInfos = useResolvedThreadInfosObj(threadInfos);
- const { chatMentionCandidatesObj, communityThreadIDForGenesisThreads } =
- React.useMemo(
- () => getChatMentionCandidates(resolvedThreadInfos),
- [resolvedThreadInfos],
- );
- return {
- chatMentionCandidatesObj,
- resolvedThreadInfos,
- communityThreadIDForGenesisThreads,
- };
-}
-
-function useChatMentionCandidatesObj(): ChatMentionCandidatesObj {
- const { chatMentionCandidatesObj } = useChatMentionCandidatesObjAndUtils();
- return chatMentionCandidatesObj;
-}
-
-function useThreadChatMentionCandidates(
- threadInfo: ThreadInfo,
-): ChatMentionCandidates {
- const { chatMentionCandidatesObj, communityThreadIDForGenesisThreads } =
- useChatMentionCandidatesObjAndUtils();
- return React.useMemo(() => {
- let communityID,
- result = {};
- if (threadInfo.community === genesis.id) {
- communityID = communityThreadIDForGenesisThreads[threadInfo.id];
- } else {
- communityID = threadInfo.community ?? threadInfo.id;
- }
- if (chatMentionCandidatesObj[communityID]) {
- result = { ...chatMentionCandidatesObj[communityID] };
- }
- delete result[threadInfo.id];
- return result;
- }, [
- chatMentionCandidatesObj,
- communityThreadIDForGenesisThreads,
- threadInfo.community,
- threadInfo.id,
- ]);
-}
-
function useUserProfileThreadInfo(userInfo: ?UserInfo): ?UserProfileThreadInfo {
const userID = userInfo?.id;
const username = userInfo?.username;
@@ -1852,50 +1692,6 @@
]);
}
-function useChatMentionSearchIndex(): {
- +[id: string]: SentencePrefixSearchIndex,
-} {
- const { chatMentionCandidatesObj } = useChatMentionCandidatesObjAndUtils();
- return React.useMemo(() => {
- const result = {};
- for (const communityThreadID in chatMentionCandidatesObj) {
- const searchIndex = new SentencePrefixSearchIndex();
- const searchIndexEntries = [];
- for (const threadID in chatMentionCandidatesObj[communityThreadID]) {
- searchIndexEntries.push({
- id: threadID,
- uiName: chatMentionCandidatesObj[communityThreadID][threadID].uiName,
- });
- }
- // Sort the keys so that the order of the search result is consistent
- searchIndexEntries.sort(({ uiName: uiNameA }, { uiName: uiNameB }) =>
- uiNameA.localeCompare(uiNameB),
- );
- for (const { id, uiName } of searchIndexEntries) {
- searchIndex.addEntry(id, uiName);
- }
- result[communityThreadID] = searchIndex;
- }
- return result;
- }, [chatMentionCandidatesObj]);
-}
-
-function useThreadChatMentionSearchIndex(
- threadInfo: ThreadInfo,
-): SentencePrefixSearchIndex {
- const chatMentionCandidatesSearchIndex = useChatMentionSearchIndex();
- const { communityThreadIDForGenesisThreads } =
- useChatMentionCandidatesObjAndUtils();
- if (threadInfo.community === genesis.id) {
- return chatMentionCandidatesSearchIndex[
- communityThreadIDForGenesisThreads[threadInfo.id]
- ];
- }
- return chatMentionCandidatesSearchIndex[
- threadInfo.community ?? threadInfo.id
- ];
-}
-
export {
threadHasPermission,
viewerIsMember,
@@ -1961,8 +1757,5 @@
useRoleMemberCountsForCommunity,
useRoleUserSurfacedPermissions,
getThreadsToDeleteText,
- useChatMentionCandidatesObj,
- useThreadChatMentionCandidates,
useUserProfileThreadInfo,
- useThreadChatMentionSearchIndex,
};
diff --git a/native/chat/chat-input-bar.react.js b/native/chat/chat-input-bar.react.js
--- a/native/chat/chat-input-bar.react.js
+++ b/native/chat/chat-input-bar.react.js
@@ -31,6 +31,10 @@
joinThread,
newThreadActionTypes,
} from 'lib/actions/thread-actions.js';
+import {
+ useChatMentionContext,
+ useThreadChatMentionCandidates,
+} from 'lib/hooks/chat-mention-hooks.js';
import { createLoadingStatusSelector } from 'lib/selectors/loading-selectors.js';
import { threadInfoSelector } from 'lib/selectors/thread-selectors.js';
import { userStoreMentionSearchIndex } from 'lib/selectors/user-selectors.js';
@@ -58,8 +62,6 @@
threadActualMembers,
checkIfDefaultMembersAreVoiced,
draftKeyFromThreadID,
- useThreadChatMentionCandidates,
- useThreadChatMentionSearchIndex,
} from 'lib/shared/thread-utils.js';
import type { CalendarQuery } from 'lib/types/entry-types.js';
import type { LoadingStatus } from 'lib/types/loading-types.js';
@@ -1266,9 +1268,8 @@
const userSearchIndex = useSelector(userStoreMentionSearchIndex);
- const chatMentionSearchIndex = useThreadChatMentionSearchIndex(
- props.threadInfo,
- );
+ const { getChatMentionSearchIndex } = useChatMentionContext();
+ const chatMentionSearchIndex = getChatMentionSearchIndex(props.threadInfo);
const { parentThreadID } = props.threadInfo;
const parentThreadInfo = useSelector(state =>
diff --git a/native/chat/message-list-types.js b/native/chat/message-list-types.js
--- a/native/chat/message-list-types.js
+++ b/native/chat/message-list-types.js
@@ -4,7 +4,7 @@
import invariant from 'invariant';
import * as React from 'react';
-import { useThreadChatMentionCandidates } from 'lib/shared/thread-utils.js';
+import { useThreadChatMentionCandidates } from 'lib/hooks/chat-mention-hooks.js';
import type { ThreadInfo } from 'lib/types/thread-types.js';
import { type UserInfo } from 'lib/types/user-types.js';
diff --git a/native/chat/message-preview.react.js b/native/chat/message-preview.react.js
--- a/native/chat/message-preview.react.js
+++ b/native/chat/message-preview.react.js
@@ -4,8 +4,8 @@
import * as React from 'react';
import { Text } from 'react-native';
+import { useThreadChatMentionCandidates } from 'lib/hooks/chat-mention-hooks.js';
import { useMessagePreview } from 'lib/shared/message-utils.js';
-import { useThreadChatMentionCandidates } from 'lib/shared/thread-utils.js';
import { type MessageInfo } from 'lib/types/message-types.js';
import { type ThreadInfo } from 'lib/types/thread-types.js';
diff --git a/native/chat/sidebar-input-bar-height-measurer.react.js b/native/chat/sidebar-input-bar-height-measurer.react.js
--- a/native/chat/sidebar-input-bar-height-measurer.react.js
+++ b/native/chat/sidebar-input-bar-height-measurer.react.js
@@ -4,7 +4,7 @@
import { View, StyleSheet } from 'react-native';
import { useLoggedInUserInfo } from 'lib/hooks/account-hooks.js';
-import { useThreadChatMentionCandidates } from 'lib/shared/thread-utils.js';
+import { useThreadChatMentionCandidates } from 'lib/hooks/chat-mention-hooks.js';
import { DummyChatInputBar } from './chat-input-bar.react.js';
import { useMessageListScreenWidth } from './composed-message-width.js';
diff --git a/native/chat/sidebar-navigation.js b/native/chat/sidebar-navigation.js
--- a/native/chat/sidebar-navigation.js
+++ b/native/chat/sidebar-navigation.js
@@ -5,10 +5,10 @@
import { ENSCacheContext } from 'lib/components/ens-cache-provider.react.js';
import { useLoggedInUserInfo } from 'lib/hooks/account-hooks.js';
+import { useThreadChatMentionCandidates } from 'lib/hooks/chat-mention-hooks.js';
import {
createPendingSidebar,
createUnresolvedPendingSidebar,
- useThreadChatMentionCandidates,
} from 'lib/shared/thread-utils.js';
import type {
ThreadInfo,
diff --git a/native/chat/utils.js b/native/chat/utils.js
--- a/native/chat/utils.js
+++ b/native/chat/utils.js
@@ -5,12 +5,10 @@
import Animated from 'react-native-reanimated';
import { useLoggedInUserInfo } from 'lib/hooks/account-hooks.js';
+import { useThreadChatMentionCandidates } from 'lib/hooks/chat-mention-hooks.js';
import { colorIsDark } from 'lib/shared/color-utils.js';
import { messageKey } from 'lib/shared/message-utils.js';
-import {
- viewerIsMember,
- useThreadChatMentionCandidates,
-} from 'lib/shared/thread-utils.js';
+import { viewerIsMember } from 'lib/shared/thread-utils.js';
import type { ThreadInfo } from 'lib/types/thread-types.js';
import { clusterEndHeight } from './chat-constants.js';
diff --git a/native/root.react.js b/native/root.react.js
--- a/native/root.react.js
+++ b/native/root.react.js
@@ -19,6 +19,7 @@
import { Provider } from 'react-redux';
import { PersistGate as ReduxPersistGate } from 'redux-persist/es/integration/react.js';
+import { ChatMentionContextProvider } from 'lib/components/chat-mention-provider.react.js';
import { EditUserAvatarProvider } from 'lib/components/edit-user-avatar-provider.react.js';
import { ENSCacheProvider } from 'lib/components/ens-cache-provider.react.js';
import IntegrityHandler from 'lib/components/integrity-handler.react.js';
@@ -259,7 +260,9 @@
-
+
+
+
diff --git a/web/app.react.js b/web/app.react.js
--- a/web/app.react.js
+++ b/web/app.react.js
@@ -15,6 +15,7 @@
fetchEntriesActionTypes,
updateCalendarQueryActionTypes,
} from 'lib/actions/entry-actions.js';
+import { ChatMentionContextProvider } from 'lib/components/chat-mention-provider.react.js';
import { EditUserAvatarProvider } from 'lib/components/edit-user-avatar-provider.react.js';
import {
ModalProvider,
@@ -193,14 +194,16 @@
-
-
-
-
-
-
-
- {content}
+
+
+
+
+
+
+
+
+ {content}
+
diff --git a/web/chat/chat-input-bar.react.js b/web/chat/chat-input-bar.react.js
--- a/web/chat/chat-input-bar.react.js
+++ b/web/chat/chat-input-bar.react.js
@@ -10,6 +10,10 @@
newThreadActionTypes,
} from 'lib/actions/thread-actions.js';
import SWMansionIcon from 'lib/components/SWMansionIcon.react.js';
+import {
+ useChatMentionContext,
+ useThreadChatMentionCandidates,
+} from 'lib/hooks/chat-mention-hooks.js';
import { createLoadingStatusSelector } from 'lib/selectors/loading-selectors.js';
import { threadInfoSelector } from 'lib/selectors/thread-selectors.js';
import { userStoreMentionSearchIndex } from 'lib/selectors/user-selectors.js';
@@ -28,8 +32,6 @@
threadFrozenDueToViewerBlock,
threadActualMembers,
checkIfDefaultMembersAreVoiced,
- useThreadChatMentionCandidates,
- useThreadChatMentionSearchIndex,
} from 'lib/shared/thread-utils.js';
import type { CalendarQuery } from 'lib/types/entry-types.js';
import type { LoadingStatus } from 'lib/types/loading-types.js';
@@ -581,9 +583,8 @@
const dispatchActionPromise = useDispatchActionPromise();
const callJoinThread = useServerCall(joinThread);
const userSearchIndex = useSelector(userStoreMentionSearchIndex);
- const chatMentionSearchIndex = useThreadChatMentionSearchIndex(
- props.threadInfo,
- );
+ const { getChatMentionSearchIndex } = useChatMentionContext();
+ const chatMentionSearchIndex = getChatMentionSearchIndex(props.threadInfo);
const { parentThreadID } = props.threadInfo;
const parentThreadInfo = useSelector(state =>
diff --git a/web/chat/chat-message-list.react.js b/web/chat/chat-message-list.react.js
--- a/web/chat/chat-message-list.react.js
+++ b/web/chat/chat-message-list.react.js
@@ -12,6 +12,7 @@
fetchMostRecentMessagesActionTypes,
fetchMostRecentMessages,
} from 'lib/actions/message-actions.js';
+import { useThreadChatMentionCandidates } from 'lib/hooks/chat-mention-hooks.js';
import { useOldestMessageServerID } from 'lib/hooks/message-hooks.js';
import { registerFetchKey } from 'lib/reducers/loading-reducer.js';
import {
@@ -19,10 +20,7 @@
useMessageListData,
} from 'lib/selectors/chat-selectors.js';
import { messageKey } from 'lib/shared/message-utils.js';
-import {
- threadIsPending,
- useThreadChatMentionCandidates,
-} from 'lib/shared/thread-utils.js';
+import { threadIsPending } from 'lib/shared/thread-utils.js';
import type { FetchMessageInfosPayload } from 'lib/types/message-types.js';
import { threadTypes } from 'lib/types/thread-types-enum.js';
import { type ThreadInfo } from 'lib/types/thread-types.js';
diff --git a/web/chat/message-preview.react.js b/web/chat/message-preview.react.js
--- a/web/chat/message-preview.react.js
+++ b/web/chat/message-preview.react.js
@@ -4,8 +4,8 @@
import invariant from 'invariant';
import * as React from 'react';
+import { useThreadChatMentionCandidates } from 'lib/hooks/chat-mention-hooks.js';
import { useMessagePreview } from 'lib/shared/message-utils.js';
-import { useThreadChatMentionCandidates } from 'lib/shared/thread-utils.js';
import { type MessageInfo } from 'lib/types/message-types.js';
import { type ThreadInfo } from 'lib/types/thread-types.js';
diff --git a/web/components/message-result.react.js b/web/components/message-result.react.js
--- a/web/components/message-result.react.js
+++ b/web/components/message-result.react.js
@@ -3,9 +3,9 @@
import classNames from 'classnames';
import * as React from 'react';
+import { useThreadChatMentionCandidates } from 'lib/hooks/chat-mention-hooks.js';
import { useStringForUser } from 'lib/hooks/ens-cache.js';
import type { ChatMessageInfoItem } from 'lib/selectors/chat-selectors.js';
-import { useThreadChatMentionCandidates } from 'lib/shared/thread-utils.js';
import type { ThreadInfo } from 'lib/types/thread-types.js';
import { longAbsoluteDate } from 'lib/utils/date-utils.js';
diff --git a/web/modals/threads/sidebars/sidebar.react.js b/web/modals/threads/sidebars/sidebar.react.js
--- a/web/modals/threads/sidebars/sidebar.react.js
+++ b/web/modals/threads/sidebars/sidebar.react.js
@@ -4,9 +4,9 @@
import * as React from 'react';
import { useModalContext } from 'lib/components/modal-provider.react.js';
+import { useThreadChatMentionCandidates } from 'lib/hooks/chat-mention-hooks.js';
import type { ChatThreadItem } from 'lib/selectors/chat-selectors.js';
import { useMessagePreview } from 'lib/shared/message-utils.js';
-import { useThreadChatMentionCandidates } from 'lib/shared/thread-utils.js';
import { shortAbsoluteDate } from 'lib/utils/date-utils.js';
import { useResolvedThreadInfo } from 'lib/utils/entity-helpers.js';
diff --git a/web/modals/threads/subchannels/subchannel.react.js b/web/modals/threads/subchannels/subchannel.react.js
--- a/web/modals/threads/subchannels/subchannel.react.js
+++ b/web/modals/threads/subchannels/subchannel.react.js
@@ -4,9 +4,9 @@
import * as React from 'react';
import { useModalContext } from 'lib/components/modal-provider.react.js';
+import { useThreadChatMentionCandidates } from 'lib/hooks/chat-mention-hooks.js';
import { type ChatThreadItem } from 'lib/selectors/chat-selectors.js';
import { useMessagePreview } from 'lib/shared/message-utils.js';
-import { useThreadChatMentionCandidates } from 'lib/shared/thread-utils.js';
import { shortAbsoluteDate } from 'lib/utils/date-utils.js';
import { useResolvedThreadInfo } from 'lib/utils/entity-helpers.js';
diff --git a/web/selectors/thread-selectors.js b/web/selectors/thread-selectors.js
--- a/web/selectors/thread-selectors.js
+++ b/web/selectors/thread-selectors.js
@@ -7,10 +7,10 @@
import { ENSCacheContext } from 'lib/components/ens-cache-provider.react.js';
import { useLoggedInUserInfo } from 'lib/hooks/account-hooks.js';
+import { useThreadChatMentionCandidates } from 'lib/hooks/chat-mention-hooks.js';
import {
createPendingSidebar,
threadInHomeChatList,
- useThreadChatMentionCandidates,
} from 'lib/shared/thread-utils.js';
import type {
ComposableMessageInfo,