2022-10-27 10:59:54 -07:00
|
|
|
import { InfiniteData, useInfiniteQuery, useMutation, useQuery } from '@tanstack/react-query';
|
2022-09-27 13:05:19 -07:00
|
|
|
import sumBy from 'lodash/sumBy';
|
2022-08-10 05:38:49 -07:00
|
|
|
|
2022-09-29 10:36:35 -07:00
|
|
|
import { importFetchedAccount, importFetchedAccounts } from 'soapbox/actions/importer';
|
2022-10-26 10:08:02 -07:00
|
|
|
import snackbar from 'soapbox/actions/snackbar';
|
2022-09-15 07:49:09 -07:00
|
|
|
import { getNextLink } from 'soapbox/api';
|
2022-11-02 12:28:16 -07:00
|
|
|
import { ChatWidgetScreens, useChatContext } from 'soapbox/contexts/chat-context';
|
2022-09-27 13:05:19 -07:00
|
|
|
import { useStatContext } from 'soapbox/contexts/stat-context';
|
2022-11-09 09:24:44 -08:00
|
|
|
import { useApi, useAppDispatch, useAppSelector, useFeatures, useOwnAccount } from 'soapbox/hooks';
|
2022-11-02 12:23:39 -07:00
|
|
|
import { normalizeChatMessage } from 'soapbox/normalizers';
|
2022-09-27 12:43:30 -07:00
|
|
|
import { flattenPages, PaginatedResult, updatePageItem } from 'soapbox/utils/queries';
|
2022-08-10 05:38:49 -07:00
|
|
|
|
2022-08-16 10:38:17 -07:00
|
|
|
import { queryClient } from './client';
|
2022-11-10 08:56:36 -08:00
|
|
|
import { useFetchRelationships } from './relationships';
|
2022-08-16 10:38:17 -07:00
|
|
|
|
2022-09-08 09:47:19 -07:00
|
|
|
import type { IAccount } from './accounts';
|
|
|
|
|
2022-11-09 05:37:50 -08:00
|
|
|
export const messageExpirationOptions = [120, 604800, 1209600, 2592000, 7776000];
|
2022-10-26 10:28:50 -07:00
|
|
|
|
2022-10-26 10:08:02 -07:00
|
|
|
export enum MessageExpirationValues {
|
2022-11-09 05:37:50 -08:00
|
|
|
'TWO_MINUTES' = messageExpirationOptions[0],
|
|
|
|
'SEVEN' = messageExpirationOptions[1],
|
|
|
|
'FOURTEEN' = messageExpirationOptions[2],
|
|
|
|
'THIRTY' = messageExpirationOptions[3],
|
|
|
|
'NINETY' = messageExpirationOptions[4]
|
2022-10-26 10:08:02 -07:00
|
|
|
}
|
|
|
|
|
2022-08-10 05:38:49 -07:00
|
|
|
export interface IChat {
|
2022-10-26 10:08:02 -07:00
|
|
|
accepted: boolean
|
|
|
|
account: IAccount
|
2022-10-28 10:01:39 -07:00
|
|
|
created_at: string
|
2022-08-16 10:38:17 -07:00
|
|
|
created_by_account: string
|
2022-10-26 10:08:02 -07:00
|
|
|
discarded_at: null | string
|
|
|
|
id: string
|
2022-08-26 09:41:25 -07:00
|
|
|
last_message: null | {
|
|
|
|
account_id: string
|
|
|
|
chat_id: string
|
|
|
|
content: string
|
|
|
|
created_at: string
|
|
|
|
discarded_at: string | null
|
|
|
|
id: string
|
|
|
|
unread: boolean
|
|
|
|
}
|
2022-11-27 13:34:43 -08:00
|
|
|
latest_read_message_by_account?: {
|
2022-11-02 12:28:16 -07:00
|
|
|
id: string,
|
|
|
|
date: string
|
2022-10-26 10:08:02 -07:00
|
|
|
}[]
|
2022-10-28 10:01:39 -07:00
|
|
|
latest_read_message_created_at: null | string
|
2022-11-02 11:53:41 -07:00
|
|
|
message_expiration?: MessageExpirationValues
|
2022-10-26 10:08:02 -07:00
|
|
|
unread: number
|
2022-08-10 05:38:49 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
export interface IChatMessage {
|
|
|
|
account_id: string
|
|
|
|
chat_id: string
|
|
|
|
content: string
|
2022-10-28 10:01:39 -07:00
|
|
|
created_at: string
|
2022-08-10 05:38:49 -07:00
|
|
|
id: string
|
|
|
|
unread: boolean
|
2022-08-31 10:20:37 -07:00
|
|
|
pending?: boolean
|
2022-08-10 05:38:49 -07:00
|
|
|
}
|
|
|
|
|
2022-10-26 10:08:02 -07:00
|
|
|
type UpdateChatVariables = {
|
|
|
|
message_expiration: MessageExpirationValues
|
|
|
|
}
|
|
|
|
|
2022-10-05 12:15:16 -07:00
|
|
|
const ChatKeys = {
|
2022-09-28 13:38:05 -07:00
|
|
|
chat: (chatId?: string) => ['chats', 'chat', chatId] as const,
|
2022-09-27 12:42:24 -07:00
|
|
|
chatMessages: (chatId: string) => ['chats', 'messages', chatId] as const,
|
2022-09-30 07:28:19 -07:00
|
|
|
chatSearch: (searchQuery?: string) => searchQuery ? ['chats', 'search', searchQuery] : ['chats', 'search'] as const,
|
2022-09-27 12:42:24 -07:00
|
|
|
};
|
|
|
|
|
2022-11-01 06:44:21 -07:00
|
|
|
/** Check if item is most recent */
|
|
|
|
const isLastMessage = (chatMessageId: string): boolean => {
|
|
|
|
const queryData = queryClient.getQueryData<InfiniteData<PaginatedResult<IChat>>>(ChatKeys.chatSearch());
|
2022-11-10 12:42:59 -08:00
|
|
|
const items = flattenPages<IChat>(queryData);
|
2022-11-01 06:44:21 -07:00
|
|
|
const chat = items?.find((item) => item.last_message?.id === chatMessageId);
|
|
|
|
|
|
|
|
return !!chat;
|
|
|
|
};
|
|
|
|
|
2022-10-25 10:10:53 -07:00
|
|
|
const useChatMessages = (chat: IChat) => {
|
2022-08-10 05:38:49 -07:00
|
|
|
const api = useApi();
|
2022-10-25 10:10:53 -07:00
|
|
|
const isBlocked = useAppSelector((state) => state.getIn(['relationships', chat.account.id, 'blocked_by']));
|
2022-08-10 05:38:49 -07:00
|
|
|
|
2022-09-27 12:43:30 -07:00
|
|
|
const getChatMessages = async (chatId: string, pageParam?: any): Promise<PaginatedResult<IChatMessage>> => {
|
2022-09-15 07:49:09 -07:00
|
|
|
const nextPageLink = pageParam?.link;
|
|
|
|
const uri = nextPageLink || `/api/v1/pleroma/chats/${chatId}/messages`;
|
2022-12-06 13:12:24 -08:00
|
|
|
const response = await api.get<any[]>(uri);
|
2022-09-15 07:49:09 -07:00
|
|
|
const { data } = response;
|
2022-08-10 05:38:49 -07:00
|
|
|
|
2022-09-15 07:49:09 -07:00
|
|
|
const link = getNextLink(response);
|
|
|
|
const hasMore = !!link;
|
2022-12-06 13:12:24 -08:00
|
|
|
const result = data.map(normalizeChatMessage);
|
2022-08-10 05:38:49 -07:00
|
|
|
|
|
|
|
return {
|
|
|
|
result,
|
2022-09-15 07:49:09 -07:00
|
|
|
link,
|
2022-08-10 05:38:49 -07:00
|
|
|
hasMore,
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2022-10-25 10:10:53 -07:00
|
|
|
const queryInfo = useInfiniteQuery(ChatKeys.chatMessages(chat.id), ({ pageParam }) => getChatMessages(chat.id, pageParam), {
|
|
|
|
enabled: !isBlocked,
|
2022-11-03 10:55:33 -07:00
|
|
|
cacheTime: 0,
|
|
|
|
staleTime: 0,
|
2022-08-10 05:38:49 -07:00
|
|
|
getNextPageParam: (config) => {
|
|
|
|
if (config.hasMore) {
|
2022-09-15 07:49:09 -07:00
|
|
|
return { link: config.link };
|
2022-08-10 05:38:49 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return undefined;
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2022-12-06 13:12:24 -08:00
|
|
|
const data = flattenPages(queryInfo.data)?.reverse();
|
2022-08-10 05:38:49 -07:00
|
|
|
|
|
|
|
return {
|
|
|
|
...queryInfo,
|
|
|
|
data,
|
|
|
|
};
|
|
|
|
};
|
2022-08-26 09:41:25 -07:00
|
|
|
|
2022-09-09 07:24:25 -07:00
|
|
|
const useChats = (search?: string) => {
|
2022-08-10 05:38:49 -07:00
|
|
|
const api = useApi();
|
2022-09-08 09:47:19 -07:00
|
|
|
const dispatch = useAppDispatch();
|
2022-09-22 14:04:52 -07:00
|
|
|
const features = useFeatures();
|
2022-09-27 13:05:19 -07:00
|
|
|
const { setUnreadChatsCount } = useStatContext();
|
2022-11-10 08:56:36 -08:00
|
|
|
const fetchRelationships = useFetchRelationships();
|
2022-08-10 05:38:49 -07:00
|
|
|
|
2022-09-27 12:43:30 -07:00
|
|
|
const getChats = async (pageParam?: any): Promise<PaginatedResult<IChat>> => {
|
2022-09-22 14:04:52 -07:00
|
|
|
const endpoint = features.chatsV2 ? '/api/v2/pleroma/chats' : '/api/v1/pleroma/chats';
|
2022-09-15 07:49:09 -07:00
|
|
|
const nextPageLink = pageParam?.link;
|
2022-09-22 14:04:52 -07:00
|
|
|
const uri = nextPageLink || endpoint;
|
2022-09-15 07:49:09 -07:00
|
|
|
const response = await api.get<IChat[]>(uri, {
|
2022-12-05 16:38:32 -08:00
|
|
|
params: search ? {
|
2022-09-09 07:24:25 -07:00
|
|
|
search,
|
2022-12-05 16:38:32 -08:00
|
|
|
} : undefined,
|
2022-08-26 09:41:25 -07:00
|
|
|
});
|
2022-09-15 07:49:09 -07:00
|
|
|
const { data } = response;
|
2022-08-26 09:41:25 -07:00
|
|
|
|
2022-09-15 07:49:09 -07:00
|
|
|
const link = getNextLink(response);
|
|
|
|
const hasMore = !!link;
|
2022-08-26 09:41:25 -07:00
|
|
|
|
2022-10-27 10:59:54 -07:00
|
|
|
setUnreadChatsCount(Number(response.headers['x-unread-messages-count']) || sumBy(data, (chat) => chat.unread));
|
2022-09-27 13:05:19 -07:00
|
|
|
|
2022-09-08 09:47:19 -07:00
|
|
|
// Set the relationships to these users in the redux store.
|
2022-11-10 08:56:36 -08:00
|
|
|
fetchRelationships.mutate({ accountIds: data.map((item) => item.account.id) });
|
2022-09-29 10:36:35 -07:00
|
|
|
dispatch(importFetchedAccounts(data.map((item) => item.account)));
|
2022-09-08 09:47:19 -07:00
|
|
|
|
2022-08-26 09:41:25 -07:00
|
|
|
return {
|
|
|
|
result: data,
|
|
|
|
hasMore,
|
2022-09-15 07:49:09 -07:00
|
|
|
link,
|
2022-08-26 09:41:25 -07:00
|
|
|
};
|
2022-08-10 05:38:49 -07:00
|
|
|
};
|
|
|
|
|
2022-10-05 12:15:16 -07:00
|
|
|
const queryInfo = useInfiniteQuery(ChatKeys.chatSearch(search), ({ pageParam }) => getChats(pageParam), {
|
2022-08-26 09:41:25 -07:00
|
|
|
keepPreviousData: true,
|
|
|
|
getNextPageParam: (config) => {
|
|
|
|
if (config.hasMore) {
|
2022-09-15 07:49:09 -07:00
|
|
|
return { link: config.link };
|
2022-08-26 09:41:25 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return undefined;
|
|
|
|
},
|
2022-08-10 05:38:49 -07:00
|
|
|
});
|
|
|
|
|
2022-12-06 13:12:24 -08:00
|
|
|
const data = flattenPages(queryInfo.data);
|
2022-08-26 09:41:25 -07:00
|
|
|
|
|
|
|
const chatsQuery = {
|
|
|
|
...queryInfo,
|
|
|
|
data,
|
|
|
|
};
|
|
|
|
|
2022-08-10 05:38:49 -07:00
|
|
|
const getOrCreateChatByAccountId = (accountId: string) => api.post<IChat>(`/api/v1/pleroma/chats/by-account-id/${accountId}`);
|
|
|
|
|
|
|
|
return { chatsQuery, getOrCreateChatByAccountId };
|
|
|
|
};
|
|
|
|
|
2022-09-28 13:38:05 -07:00
|
|
|
const useChat = (chatId?: string) => {
|
|
|
|
const api = useApi();
|
2022-09-29 10:36:35 -07:00
|
|
|
const dispatch = useAppDispatch();
|
2022-11-10 08:56:36 -08:00
|
|
|
const fetchRelationships = useFetchRelationships();
|
2022-09-28 13:38:05 -07:00
|
|
|
|
|
|
|
const getChat = async () => {
|
|
|
|
if (chatId) {
|
|
|
|
const { data } = await api.get<IChat>(`/api/v1/pleroma/chats/${chatId}`);
|
2022-09-29 10:36:35 -07:00
|
|
|
|
2022-11-10 08:56:36 -08:00
|
|
|
fetchRelationships.mutate({ accountIds: [data.account.id] });
|
2022-09-29 10:36:35 -07:00
|
|
|
dispatch(importFetchedAccount(data.account));
|
|
|
|
|
2022-09-28 13:38:05 -07:00
|
|
|
return data;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-11-02 12:28:16 -07:00
|
|
|
return useQuery<IChat | undefined>(ChatKeys.chat(chatId), getChat, {
|
2022-11-07 12:40:19 -08:00
|
|
|
cacheTime: 0,
|
2022-11-02 12:28:16 -07:00
|
|
|
enabled: !!chatId,
|
|
|
|
});
|
2022-09-28 13:38:05 -07:00
|
|
|
};
|
|
|
|
|
2022-09-28 13:20:59 -07:00
|
|
|
const useChatActions = (chatId: string) => {
|
2022-11-09 09:24:44 -08:00
|
|
|
const account = useOwnAccount();
|
2022-08-10 05:38:49 -07:00
|
|
|
const api = useApi();
|
2022-10-26 10:08:02 -07:00
|
|
|
const dispatch = useAppDispatch();
|
2022-11-09 09:24:44 -08:00
|
|
|
|
2022-10-27 10:59:54 -07:00
|
|
|
const { setUnreadChatsCount } = useStatContext();
|
2022-10-26 10:08:02 -07:00
|
|
|
|
2022-11-02 12:28:16 -07:00
|
|
|
const { chat, changeScreen } = useChatContext();
|
2022-08-10 05:38:49 -07:00
|
|
|
|
2022-10-28 10:01:39 -07:00
|
|
|
const markChatAsRead = async (lastReadId: string) => {
|
|
|
|
return api.post<IChat>(`/api/v1/pleroma/chats/${chatId}/read`, { last_read_id: lastReadId })
|
2022-10-27 10:59:54 -07:00
|
|
|
.then(({ data }) => {
|
|
|
|
updatePageItem(ChatKeys.chatSearch(), data, (o, n) => o.id === n.id);
|
|
|
|
const queryData = queryClient.getQueryData<InfiniteData<PaginatedResult<unknown>>>(ChatKeys.chatSearch());
|
2022-10-28 10:01:39 -07:00
|
|
|
|
2022-10-27 10:59:54 -07:00
|
|
|
if (queryData) {
|
|
|
|
const flattenedQueryData: any = flattenPages(queryData)?.map((chat: any) => {
|
|
|
|
if (chat.id === data.id) {
|
|
|
|
return data;
|
|
|
|
} else {
|
|
|
|
return chat;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
setUnreadChatsCount(sumBy(flattenedQueryData, (chat: IChat) => chat.unread));
|
|
|
|
}
|
2022-10-28 10:01:39 -07:00
|
|
|
|
|
|
|
return data;
|
2022-10-27 10:59:54 -07:00
|
|
|
})
|
2022-08-30 07:10:31 -07:00
|
|
|
.catch(() => null);
|
|
|
|
};
|
2022-08-10 05:38:49 -07:00
|
|
|
|
2022-11-09 09:24:44 -08:00
|
|
|
const createChatMessage = useMutation(
|
|
|
|
(
|
2022-12-06 14:27:07 -08:00
|
|
|
{ chatId, content, mediaId }: { chatId: string, content: string, mediaId?: string },
|
|
|
|
) => api.post<IChatMessage>(`/api/v1/pleroma/chats/${chatId}/messages`, { content, media_id: mediaId }),
|
2022-11-09 09:24:44 -08:00
|
|
|
{
|
|
|
|
retry: false,
|
|
|
|
onMutate: async (variables) => {
|
|
|
|
// Cancel any outgoing refetches (so they don't overwrite our optimistic update)
|
|
|
|
await queryClient.cancelQueries(['chats', 'messages', variables.chatId]);
|
|
|
|
|
|
|
|
// Snapshot the previous value
|
|
|
|
const prevContent = variables.content;
|
|
|
|
const prevChatMessages = queryClient.getQueryData(['chats', 'messages', variables.chatId]);
|
|
|
|
|
|
|
|
// Optimistically update to the new value
|
|
|
|
queryClient.setQueryData(ChatKeys.chatMessages(variables.chatId), (prevResult: any) => {
|
|
|
|
const newResult = { ...prevResult };
|
|
|
|
newResult.pages = newResult.pages.map((page: any, idx: number) => {
|
|
|
|
if (idx === 0) {
|
|
|
|
return {
|
|
|
|
...page,
|
2022-11-09 10:17:10 -08:00
|
|
|
result: [
|
|
|
|
normalizeChatMessage({
|
|
|
|
content: variables.content,
|
|
|
|
id: String(Number(new Date())),
|
|
|
|
created_at: new Date(),
|
|
|
|
account_id: account?.id,
|
|
|
|
pending: true,
|
|
|
|
unread: true,
|
|
|
|
}),
|
2022-12-07 14:18:03 -08:00
|
|
|
...page.result,
|
2022-11-09 10:17:10 -08:00
|
|
|
],
|
2022-11-09 09:24:44 -08:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return page;
|
|
|
|
});
|
|
|
|
|
|
|
|
return newResult;
|
|
|
|
});
|
|
|
|
|
|
|
|
return { prevChatMessages, prevContent };
|
|
|
|
},
|
|
|
|
// If the mutation fails, use the context returned from onMutate to roll back
|
|
|
|
onError: (_error: any, variables, context: any) => {
|
|
|
|
queryClient.setQueryData(['chats', 'messages', variables.chatId], context.prevChatMessages);
|
|
|
|
},
|
2022-11-10 13:41:24 -08:00
|
|
|
onSuccess: (response, variables) => {
|
|
|
|
const nextChat = { ...chat, last_message: response.data };
|
|
|
|
updatePageItem(ChatKeys.chatSearch(), nextChat, (o, n) => o.id === n.id);
|
|
|
|
|
2022-11-09 09:24:44 -08:00
|
|
|
queryClient.invalidateQueries(ChatKeys.chatMessages(variables.chatId));
|
|
|
|
},
|
|
|
|
},
|
|
|
|
);
|
2022-08-10 05:38:49 -07:00
|
|
|
|
2022-10-26 10:08:02 -07:00
|
|
|
const updateChat = useMutation((data: UpdateChatVariables) => api.patch<IChat>(`/api/v1/pleroma/chats/${chatId}`, data), {
|
|
|
|
onMutate: async (data) => {
|
|
|
|
// Cancel any outgoing refetches (so they don't overwrite our optimistic update)
|
|
|
|
await queryClient.cancelQueries(ChatKeys.chat(chatId));
|
|
|
|
|
|
|
|
// Snapshot the previous value
|
|
|
|
const prevChat = { ...chat };
|
|
|
|
const nextChat = { ...chat, ...data };
|
|
|
|
|
|
|
|
// Optimistically update to the new value
|
|
|
|
queryClient.setQueryData(ChatKeys.chat(chatId), nextChat);
|
2022-11-02 12:28:16 -07:00
|
|
|
|
2022-10-26 10:08:02 -07:00
|
|
|
// Return a context object with the snapshotted value
|
|
|
|
return { prevChat };
|
|
|
|
},
|
|
|
|
// If the mutation fails, use the context returned from onMutate to roll back
|
|
|
|
onError: (_error: any, _newData: any, context: any) => {
|
2022-11-02 12:28:16 -07:00
|
|
|
changeScreen(ChatWidgetScreens.CHAT, context.prevChat.id);
|
2022-10-26 10:08:02 -07:00
|
|
|
queryClient.setQueryData(ChatKeys.chat(chatId), context.prevChat);
|
|
|
|
dispatch(snackbar.error('Chat Settings failed to update.'));
|
|
|
|
},
|
2022-11-04 06:49:35 -07:00
|
|
|
onSuccess() {
|
2022-10-26 10:08:02 -07:00
|
|
|
queryClient.invalidateQueries(ChatKeys.chat(chatId));
|
2022-10-26 10:28:50 -07:00
|
|
|
queryClient.invalidateQueries(ChatKeys.chatSearch());
|
2022-10-26 10:08:02 -07:00
|
|
|
dispatch(snackbar.success('Chat Settings updated successfully'));
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2022-08-16 05:39:58 -07:00
|
|
|
const deleteChatMessage = (chatMessageId: string) => api.delete<IChat>(`/api/v1/pleroma/chats/${chatId}/messages/${chatMessageId}`);
|
|
|
|
|
2022-08-16 10:38:17 -07:00
|
|
|
const acceptChat = useMutation(() => api.post<IChat>(`/api/v1/pleroma/chats/${chatId}/accept`), {
|
|
|
|
onSuccess(response) {
|
2022-11-02 12:28:16 -07:00
|
|
|
changeScreen(ChatWidgetScreens.CHAT, response.data.id);
|
2022-11-04 04:37:05 -07:00
|
|
|
queryClient.invalidateQueries(ChatKeys.chat(chatId));
|
2022-10-05 12:15:16 -07:00
|
|
|
queryClient.invalidateQueries(ChatKeys.chatMessages(chatId));
|
|
|
|
queryClient.invalidateQueries(ChatKeys.chatSearch());
|
2022-08-16 10:38:17 -07:00
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2022-08-16 10:54:36 -07:00
|
|
|
const deleteChat = useMutation(() => api.delete<IChat>(`/api/v1/pleroma/chats/${chatId}`), {
|
2022-11-02 12:28:16 -07:00
|
|
|
onSuccess() {
|
|
|
|
changeScreen(ChatWidgetScreens.INBOX);
|
2022-10-05 12:15:16 -07:00
|
|
|
queryClient.invalidateQueries(ChatKeys.chatMessages(chatId));
|
|
|
|
queryClient.invalidateQueries(ChatKeys.chatSearch());
|
2022-08-16 10:54:36 -07:00
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2022-10-26 10:08:02 -07:00
|
|
|
return { createChatMessage, markChatAsRead, deleteChatMessage, updateChat, acceptChat, deleteChat };
|
2022-08-10 05:38:49 -07:00
|
|
|
};
|
|
|
|
|
2022-11-01 06:44:21 -07:00
|
|
|
export { ChatKeys, useChat, useChatActions, useChats, useChatMessages, isLastMessage };
|