pleroma/app/soapbox/features/conversations/components/conversations-list.tsx
2023-02-28 15:24:04 -05:00

75 lines
2.3 KiB
TypeScript

import debounce from 'lodash/debounce';
import React, { useRef } from 'react';
import { FormattedMessage } from 'react-intl';
import { expandConversations } from 'soapbox/actions/conversations';
import ScrollableList from 'soapbox/components/scrollable-list';
import { useAppDispatch, useAppSelector } from 'soapbox/hooks';
import Conversation from './conversation';
import type { VirtuosoHandle } from 'react-virtuoso';
const ConversationsList: React.FC = () => {
const dispatch = useAppDispatch();
const ref = useRef<VirtuosoHandle>(null);
const conversations = useAppSelector((state) => state.conversations.items);
const isLoading = useAppSelector((state) => state.conversations.isLoading);
const hasMore = useAppSelector((state) => state.conversations.hasMore === true);
const getCurrentIndex = (id: string) => conversations.findIndex(x => x.id === id);
const handleMoveUp = (id: string) => {
const elementIndex = getCurrentIndex(id) - 1;
selectChild(elementIndex);
};
const handleMoveDown = (id: string) => {
const elementIndex = getCurrentIndex(id) + 1;
selectChild(elementIndex);
};
const selectChild = (index: number) => {
ref.current?.scrollIntoView({
index,
behavior: 'smooth',
done: () => {
const element = document.querySelector<HTMLDivElement>(`#direct-list [data-index="${index}"] .focusable`);
if (element) {
element.focus();
}
},
});
};
const handleLoadOlder = debounce(() => {
const maxId = conversations.getIn([-1, 'id']);
if (maxId) dispatch(expandConversations({ maxId }));
}, 300, { leading: true });
return (
<ScrollableList
hasMore={hasMore}
onLoadMore={handleLoadOlder}
id='direct-list'
scrollKey='direct'
ref={ref}
isLoading={isLoading}
showLoading={isLoading && conversations.size === 0}
emptyMessage={<FormattedMessage id='empty_column.direct' defaultMessage="You don't have any direct messages yet. When you send or receive one, it will show up here." />}
>
{conversations.map((item: any) => (
<Conversation
key={item.id}
conversationId={item.id}
onMoveUp={handleMoveUp}
onMoveDown={handleMoveDown}
/>
))}
</ScrollableList>
);
};
export default ConversationsList;