bigbuffet-rw/app/soapbox/queries/chats.ts

209 lines
6.1 KiB
TypeScript
Raw Normal View History

2022-09-16 05:57:09 -07:00
import { useInfiniteQuery, useMutation, useQuery } from '@tanstack/react-query';
import sumBy from 'lodash/sumBy';
2022-08-10 05:38:49 -07:00
2022-09-08 09:47:19 -07:00
import { fetchRelationships } from 'soapbox/actions/accounts';
2022-09-29 10:36:35 -07:00
import { importFetchedAccount, importFetchedAccounts } from 'soapbox/actions/importer';
2022-09-15 07:49:09 -07:00
import { getNextLink } from 'soapbox/api';
import compareId from 'soapbox/compare_id';
2022-08-16 10:38:17 -07:00
import { useChatContext } from 'soapbox/contexts/chat-context';
import { useStatContext } from 'soapbox/contexts/stat-context';
2022-10-25 10:10:53 -07:00
import { useApi, useAppDispatch, useAppSelector, useFeatures } from 'soapbox/hooks';
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-09-08 09:47:19 -07:00
import type { IAccount } from './accounts';
2022-08-10 05:38:49 -07:00
export interface IChat {
id: string
unread: number
2022-08-16 10:38:17 -07:00
created_by_account: 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-08-10 05:38:49 -07:00
created_at: Date
updated_at: Date
accepted: boolean
discarded_at: null | string
2022-09-08 09:47:19 -07:00
account: IAccount
2022-08-10 05:38:49 -07:00
}
export interface IChatMessage {
account_id: string
chat_id: string
content: string
created_at: Date
id: string
unread: boolean
2022-08-31 10:20:37 -07:00
pending?: boolean
2022-08-10 05:38:49 -07:00
}
const ChatKeys = {
chat: (chatId?: string) => ['chats', 'chat', chatId] as const,
2022-09-27 12:42:24 -07:00
chatMessages: (chatId: string) => ['chats', 'messages', chatId] as const,
chatSearch: (searchQuery?: string) => searchQuery ? ['chats', 'search', searchQuery] : ['chats', 'search'] as const,
2022-09-27 12:42:24 -07:00
};
const reverseOrder = (a: IChat, b: IChat): number => compareId(a.id, b.id);
2022-08-10 05:38:49 -07:00
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
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`;
const response = await api.get(uri);
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-10-04 07:48:37 -07:00
const result = data.sort(reverseOrder);
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-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;
},
});
const data = flattenPages(queryInfo.data);
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();
const { setUnreadChatsCount } = useStatContext();
2022-08-10 05:38:49 -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-08-26 09:41:25 -07:00
params: {
2022-09-09 07:24:25 -07:00
search,
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
// TODO: change to response header
setUnreadChatsCount(sumBy(data, (chat) => chat.unread));
2022-09-08 09:47:19 -07:00
// Set the relationships to these users in the redux store.
dispatch(fetchRelationships(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
};
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
});
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 };
};
const useChat = (chatId?: string) => {
const api = useApi();
const actions = useChatActions(chatId!);
2022-09-29 10:36:35 -07:00
const dispatch = useAppDispatch();
const getChat = async () => {
if (chatId) {
const { data } = await api.get<IChat>(`/api/v1/pleroma/chats/${chatId}`);
2022-09-29 10:36:35 -07:00
dispatch(importFetchedAccount(data.account));
return data;
}
};
const chat = useQuery<IChat | undefined>(ChatKeys.chat(chatId), getChat);
return { ...actions, chat };
};
2022-09-28 13:20:59 -07:00
const useChatActions = (chatId: string) => {
2022-08-10 05:38:49 -07:00
const api = useApi();
2022-08-17 12:48:04 -07:00
const { setChat, setEditing } = useChatContext();
2022-08-10 05:38:49 -07:00
2022-08-30 07:10:31 -07:00
const markChatAsRead = (lastReadId: string) => {
api.post<IChat>(`/api/v1/pleroma/chats/${chatId}/read`, { last_read_id: lastReadId })
.then(({ data }) => updatePageItem(['chats', 'search'], data, (o, n) => o.id === n.id))
2022-08-30 07:10:31 -07:00
.catch(() => null);
};
2022-08-10 05:38:49 -07:00
const createChatMessage = (chatId: string, content: string) => {
return api.post<IChat>(`/api/v1/pleroma/chats/${chatId}/messages`, { content });
};
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) {
setChat(response.data);
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}`), {
onSuccess(response) {
setChat(null);
2022-08-17 12:48:04 -07:00
setEditing(false);
queryClient.invalidateQueries(ChatKeys.chatMessages(chatId));
queryClient.invalidateQueries(ChatKeys.chatSearch());
2022-08-16 10:54:36 -07:00
},
});
return { createChatMessage, markChatAsRead, deleteChatMessage, acceptChat, deleteChat };
2022-08-10 05:38:49 -07:00
};
2022-10-25 05:23:33 -07:00
export { ChatKeys, useChat, useChatActions, useChats, useChatMessages };