268 lines
7 KiB
TypeScript
268 lines
7 KiB
TypeScript
import { useInfiniteQuery, useMutation, useQuery } from '@tanstack/react-query';
|
|
import { useEffect, useState } from 'react';
|
|
|
|
import { fetchRelationships } from 'soapbox/actions/accounts';
|
|
import snackbar from 'soapbox/actions/snackbar';
|
|
import { getNextLink } from 'soapbox/api';
|
|
import compareId from 'soapbox/compare_id';
|
|
import { useChatContext } from 'soapbox/contexts/chat-context';
|
|
import { useApi, useAppDispatch } from 'soapbox/hooks';
|
|
import { normalizeChatMessage } from 'soapbox/normalizers';
|
|
|
|
import { queryClient } from './client';
|
|
|
|
import type { IAccount } from './accounts';
|
|
|
|
export interface IChat {
|
|
id: string
|
|
unread: number
|
|
created_by_account: string
|
|
last_message: null | {
|
|
account_id: string
|
|
chat_id: string
|
|
content: string
|
|
created_at: string
|
|
discarded_at: string | null
|
|
id: string
|
|
unread: boolean
|
|
}
|
|
created_at: Date
|
|
updated_at: Date
|
|
accepted: boolean
|
|
discarded_at: null | string
|
|
account: IAccount
|
|
}
|
|
|
|
export interface IChatMessage {
|
|
account_id: string
|
|
chat_id: string
|
|
content: string
|
|
created_at: Date
|
|
id: string
|
|
unread: boolean
|
|
pending?: boolean
|
|
}
|
|
|
|
export interface IChatSilence {
|
|
id: number
|
|
account_id: number
|
|
target_account_id: number
|
|
}
|
|
|
|
export interface PaginatedResult<T> {
|
|
result: T[],
|
|
hasMore: boolean,
|
|
link?: string,
|
|
}
|
|
|
|
const reverseOrder = (a: IChat, b: IChat): number => compareId(a.id, b.id);
|
|
|
|
const useChatMessages = (chatId: string) => {
|
|
const api = useApi();
|
|
|
|
const getChatMessages = async(chatId: string, pageParam?: any): Promise<PaginatedResult<IChatMessage>> => {
|
|
const nextPageLink = pageParam?.link;
|
|
const uri = nextPageLink || `/api/v1/pleroma/chats/${chatId}/messages`;
|
|
const response = await api.get(uri);
|
|
const { data } = response;
|
|
|
|
const link = getNextLink(response);
|
|
const hasMore = !!link;
|
|
const result = data.sort(reverseOrder).map(normalizeChatMessage);
|
|
|
|
return {
|
|
result,
|
|
link,
|
|
hasMore,
|
|
};
|
|
};
|
|
|
|
const queryInfo = useInfiniteQuery(['chats', 'messages', chatId], ({ pageParam }) => getChatMessages(chatId, pageParam), {
|
|
keepPreviousData: true,
|
|
getNextPageParam: (config) => {
|
|
if (config.hasMore) {
|
|
return { link: config.link };
|
|
}
|
|
|
|
return undefined;
|
|
},
|
|
});
|
|
|
|
const data = queryInfo.data?.pages.reduce<IChatMessage[]>(
|
|
(prev: IChatMessage[], curr) => [...curr.result, ...prev],
|
|
[],
|
|
);
|
|
|
|
return {
|
|
...queryInfo,
|
|
data,
|
|
};
|
|
};
|
|
|
|
const useChats = (search?: string) => {
|
|
const api = useApi();
|
|
const dispatch = useAppDispatch();
|
|
|
|
const getChats = async(pageParam?: any): Promise<PaginatedResult<IChat>> => {
|
|
const nextPageLink = pageParam?.link;
|
|
const uri = nextPageLink || '/api/v1/pleroma/chats';
|
|
const response = await api.get<IChat[]>(uri, {
|
|
params: {
|
|
search,
|
|
},
|
|
});
|
|
const { data } = response;
|
|
|
|
const link = getNextLink(response);
|
|
const hasMore = !!link;
|
|
|
|
// Set the relationships to these users in the redux store.
|
|
dispatch(fetchRelationships(data.map((item) => item.account.id)));
|
|
|
|
return {
|
|
result: data,
|
|
hasMore,
|
|
link,
|
|
};
|
|
};
|
|
|
|
const queryInfo = useInfiniteQuery(['chats', 'search', search], ({ pageParam }) => getChats(pageParam), {
|
|
keepPreviousData: true,
|
|
getNextPageParam: (config) => {
|
|
if (config.hasMore) {
|
|
return { link: config.link };
|
|
}
|
|
|
|
return undefined;
|
|
},
|
|
});
|
|
|
|
const data = queryInfo.data?.pages.reduce<IChat[]>(
|
|
(prev: IChat[], curr) => [...prev, ...curr.result],
|
|
[],
|
|
);
|
|
|
|
const chatsQuery = {
|
|
...queryInfo,
|
|
data,
|
|
};
|
|
|
|
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 { setChat, setEditing } = useChatContext();
|
|
|
|
const markChatAsRead = (lastReadId: string) => {
|
|
api.post<IChat>(`/api/v1/pleroma/chats/${chatId}/read`, { last_read_id: lastReadId })
|
|
.then(() => queryClient.invalidateQueries(['chats', 'search']))
|
|
.catch(() => null);
|
|
};
|
|
|
|
const createChatMessage = (chatId: string, content: string) => {
|
|
return api.post<IChat>(`/api/v1/pleroma/chats/${chatId}/messages`, { content });
|
|
};
|
|
|
|
const deleteChatMessage = (chatMessageId: string) => api.delete<IChat>(`/api/v1/pleroma/chats/${chatId}/messages/${chatMessageId}`);
|
|
|
|
const acceptChat = useMutation(() => api.post<IChat>(`/api/v1/pleroma/chats/${chatId}/accept`), {
|
|
onSuccess(response) {
|
|
setChat(response.data);
|
|
queryClient.invalidateQueries(['chats', 'messages', chatId]);
|
|
queryClient.invalidateQueries(['chats', 'search']);
|
|
},
|
|
});
|
|
|
|
const deleteChat = useMutation(() => api.delete<IChat>(`/api/v1/pleroma/chats/${chatId}`), {
|
|
onSuccess(response) {
|
|
setChat(null);
|
|
setEditing(false);
|
|
queryClient.invalidateQueries(['chats', 'messages', chatId]);
|
|
queryClient.invalidateQueries(['chats', 'search']);
|
|
},
|
|
});
|
|
|
|
return { createChatMessage, markChatAsRead, deleteChatMessage, acceptChat, deleteChat };
|
|
};
|
|
|
|
const useChatSilences = () => {
|
|
const api = useApi();
|
|
|
|
const getChatSilences = async() => {
|
|
const { data } = await api.get<IChatSilence[]>('/api/v1/pleroma/chats/silences');
|
|
|
|
return data;
|
|
};
|
|
|
|
return useQuery<IChatSilence[]>(['chatSilences'], getChatSilences, {
|
|
placeholderData: [],
|
|
});
|
|
};
|
|
|
|
const useChatSilence = (chat: IChat | null) => {
|
|
const api = useApi();
|
|
const dispatch = useAppDispatch();
|
|
|
|
const [isSilenced, setSilenced] = useState<boolean>(false);
|
|
|
|
const getChatSilences = async() => {
|
|
const { data } = await api.get(`api/v1/pleroma/chats/silence?account_id=${chat?.account.id}`);
|
|
return data;
|
|
};
|
|
|
|
const fetchChatSilence = async() => {
|
|
const data = await getChatSilences();
|
|
if (data) {
|
|
setSilenced(true);
|
|
} else {
|
|
setSilenced(false);
|
|
}
|
|
};
|
|
|
|
const handleSilence = () => {
|
|
if (isSilenced) {
|
|
deleteSilence();
|
|
} else {
|
|
createSilence();
|
|
}
|
|
};
|
|
|
|
const createSilence = () => {
|
|
setSilenced(true);
|
|
|
|
api.post(`api/v1/pleroma/chats/silence?account_id=${chat?.account.id}`)
|
|
.then(() => {
|
|
dispatch(snackbar.success('Successfully silenced this chat.'));
|
|
})
|
|
.catch(() => {
|
|
dispatch(snackbar.error('Something went wrong trying to silence this chat. Please try again.'));
|
|
setSilenced(false);
|
|
});
|
|
};
|
|
|
|
const deleteSilence = () => {
|
|
setSilenced(false);
|
|
|
|
api.delete(`api/v1/pleroma/chats/silence?account_id=${chat?.account.id}`)
|
|
.then(() => {
|
|
dispatch(snackbar.success('Successfully unsilenced this chat.'));
|
|
})
|
|
.catch(() => {
|
|
dispatch(snackbar.error('Something went wrong trying to unsilence this chat. Please try again.'));
|
|
setSilenced(true);
|
|
});
|
|
};
|
|
|
|
useEffect(() => {
|
|
if (chat?.id) {
|
|
fetchChatSilence();
|
|
}
|
|
}, [chat?.id]);
|
|
|
|
return { isSilenced, handleSilence };
|
|
};
|
|
|
|
export { useChat, useChats, useChatMessages, useChatSilences, useChatSilence };
|