### Description Improve Email Thread Visibility with Collapse/Expansion Rules ### Refs #4203 ### Demo https://github.com/twentyhq/twenty/assets/140154534/ece1d783-57ef-45c9-9895-3b4b0e02b9e2 Fixes #4203 --------- Co-authored-by: gitstart-twenty <gitstart-twenty@users.noreply.github.com>
107 lines
3.7 KiB
TypeScript
107 lines
3.7 KiB
TypeScript
import styled from '@emotion/styled';
|
|
import { useRecoilCallback } from 'recoil';
|
|
|
|
import { FetchMoreLoader } from '@/activities/components/CustomResolverFetchMoreLoader';
|
|
import { EmailLoader } from '@/activities/emails/components/EmailLoader';
|
|
import { EmailThreadHeader } from '@/activities/emails/components/EmailThreadHeader';
|
|
import { EmailThreadMessage } from '@/activities/emails/components/EmailThreadMessage';
|
|
import { IntermediaryMessages } from '@/activities/emails/right-drawer/components/IntermediaryMessages';
|
|
import { useRightDrawerEmailThread } from '@/activities/emails/right-drawer/hooks/useRightDrawerEmailThread';
|
|
import { emailThreadIdWhenEmailThreadWasClosedState } from '@/activities/emails/states/lastViewableEmailThreadIdState';
|
|
import { EmailThreadMessage as EmailThreadMessageType } from '@/activities/emails/types/EmailThreadMessage';
|
|
import { RIGHT_DRAWER_CLICK_OUTSIDE_LISTENER_ID } from '@/ui/layout/right-drawer/constants/RightDrawerClickOutsideListener';
|
|
import { useClickOutsideListener } from '@/ui/utilities/pointer-event/hooks/useClickOutsideListener';
|
|
|
|
const StyledContainer = styled.div`
|
|
box-sizing: border-box;
|
|
display: flex;
|
|
flex-direction: column;
|
|
height: 100%;
|
|
justify-content: flex-start;
|
|
overflow-y: auto;
|
|
position: relative;
|
|
`;
|
|
|
|
const getVisibleMessages = (messages: EmailThreadMessageType[]) =>
|
|
messages.filter(({ messageParticipants }) => {
|
|
const from = messageParticipants.find(
|
|
(participant) => participant.role === 'from',
|
|
);
|
|
const receivers = messageParticipants.filter(
|
|
(participant) => participant.role !== 'from',
|
|
);
|
|
return from && receivers.length > 0;
|
|
});
|
|
|
|
export const RightDrawerEmailThread = () => {
|
|
const { thread, messages, fetchMoreMessages, loading } =
|
|
useRightDrawerEmailThread();
|
|
|
|
const { useRegisterClickOutsideListenerCallback } = useClickOutsideListener(
|
|
RIGHT_DRAWER_CLICK_OUTSIDE_LISTENER_ID,
|
|
);
|
|
|
|
useRegisterClickOutsideListenerCallback({
|
|
callbackId:
|
|
'EmailThreadClickOutsideCallBack-' + thread.id ?? 'no-thread-id',
|
|
callbackFunction: useRecoilCallback(
|
|
({ set }) =>
|
|
() => {
|
|
set(emailThreadIdWhenEmailThreadWasClosedState, thread.id);
|
|
},
|
|
[thread],
|
|
),
|
|
});
|
|
|
|
if (!thread) {
|
|
return null;
|
|
}
|
|
|
|
const visibleMessages = getVisibleMessages(messages);
|
|
const visibleMessagesCount = visibleMessages.length;
|
|
const is5OrMoreMessages = visibleMessagesCount >= 5;
|
|
const firstMessages = visibleMessages.slice(
|
|
0,
|
|
is5OrMoreMessages ? 2 : visibleMessagesCount - 1,
|
|
);
|
|
const intermediaryMessages = is5OrMoreMessages
|
|
? visibleMessages.slice(2, visibleMessagesCount - 1)
|
|
: [];
|
|
const lastMessage = visibleMessages[visibleMessagesCount - 1];
|
|
|
|
return (
|
|
<StyledContainer>
|
|
<EmailThreadHeader
|
|
subject={thread.subject}
|
|
lastMessageSentAt={thread.lastMessageReceivedAt}
|
|
/>
|
|
{loading ? (
|
|
<EmailLoader loadingText="Loading thread" />
|
|
) : (
|
|
<>
|
|
{firstMessages.map((message) => (
|
|
<EmailThreadMessage
|
|
key={message.id}
|
|
participants={message.messageParticipants}
|
|
body={message.text}
|
|
sentAt={message.receivedAt}
|
|
/>
|
|
))}
|
|
<IntermediaryMessages messages={intermediaryMessages} />
|
|
<EmailThreadMessage
|
|
key={lastMessage.id}
|
|
participants={lastMessage.messageParticipants}
|
|
body={lastMessage.text}
|
|
sentAt={lastMessage.receivedAt}
|
|
isExpanded
|
|
/>
|
|
<FetchMoreLoader
|
|
loading={loading}
|
|
onLastRowVisible={fetchMoreMessages}
|
|
/>
|
|
</>
|
|
)}
|
|
</StyledContainer>
|
|
);
|
|
};
|