2023-01-03 19:55:46 +00:00
|
|
|
// Copyright 2020 Signal Messenger, LLC
|
2021-10-05 16:47:06 +00:00
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
import React from 'react';
|
|
|
|
|
|
|
|
export type PropsType = {
|
2022-12-08 01:26:59 +00:00
|
|
|
conversationId: string;
|
|
|
|
processAttachments: (options: {
|
|
|
|
conversationId: string;
|
|
|
|
files: ReadonlyArray<File>;
|
|
|
|
}) => void;
|
2021-10-05 16:47:06 +00:00
|
|
|
renderCompositionArea: () => JSX.Element;
|
|
|
|
renderConversationHeader: () => JSX.Element;
|
|
|
|
renderTimeline: () => JSX.Element;
|
2022-12-14 18:41:04 +00:00
|
|
|
renderPanel: () => JSX.Element | undefined;
|
2021-10-05 16:47:06 +00:00
|
|
|
};
|
|
|
|
|
2022-11-18 00:45:19 +00:00
|
|
|
export function ConversationView({
|
2022-12-08 01:26:59 +00:00
|
|
|
conversationId,
|
|
|
|
processAttachments,
|
2021-10-05 16:47:06 +00:00
|
|
|
renderCompositionArea,
|
|
|
|
renderConversationHeader,
|
|
|
|
renderTimeline,
|
2022-12-14 18:41:04 +00:00
|
|
|
renderPanel,
|
2022-11-18 00:45:19 +00:00
|
|
|
}: PropsType): JSX.Element {
|
2022-12-08 01:26:59 +00:00
|
|
|
const onDrop = React.useCallback(
|
|
|
|
(event: React.DragEvent<HTMLDivElement>) => {
|
|
|
|
if (!event.dataTransfer) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (event.dataTransfer.types[0] !== 'Files') {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
|
|
const { files } = event.dataTransfer;
|
|
|
|
processAttachments({
|
|
|
|
conversationId,
|
|
|
|
files: Array.from(files),
|
|
|
|
});
|
|
|
|
},
|
|
|
|
[conversationId, processAttachments]
|
|
|
|
);
|
|
|
|
|
|
|
|
const onPaste = React.useCallback(
|
|
|
|
(event: React.ClipboardEvent<HTMLDivElement>) => {
|
|
|
|
if (!event.clipboardData) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const { items } = event.clipboardData;
|
|
|
|
|
|
|
|
const anyImages = [...items].some(
|
|
|
|
item => item.type.split('/')[0] === 'image'
|
|
|
|
);
|
|
|
|
if (!anyImages) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
|
|
const files: Array<File> = [];
|
|
|
|
for (let i = 0; i < items.length; i += 1) {
|
|
|
|
if (items[i].type.split('/')[0] === 'image') {
|
|
|
|
const file = items[i].getAsFile();
|
|
|
|
if (file) {
|
|
|
|
files.push(file);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
processAttachments({
|
|
|
|
conversationId,
|
|
|
|
files,
|
|
|
|
});
|
|
|
|
},
|
|
|
|
[conversationId, processAttachments]
|
|
|
|
);
|
|
|
|
|
2021-10-05 16:47:06 +00:00
|
|
|
return (
|
2022-12-08 01:26:59 +00:00
|
|
|
<div className="ConversationView" onDrop={onDrop} onPaste={onPaste}>
|
2021-10-05 16:47:06 +00:00
|
|
|
<div className="ConversationView__header">
|
|
|
|
{renderConversationHeader()}
|
|
|
|
</div>
|
|
|
|
<div className="ConversationView__pane main panel">
|
|
|
|
<div className="ConversationView__timeline--container">
|
|
|
|
<div aria-live="polite" className="ConversationView__timeline">
|
|
|
|
{renderTimeline()}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div className="ConversationView__composition-area">
|
|
|
|
{renderCompositionArea()}
|
|
|
|
</div>
|
|
|
|
</div>
|
2022-12-14 18:41:04 +00:00
|
|
|
{renderPanel()}
|
2021-10-05 16:47:06 +00:00
|
|
|
</div>
|
|
|
|
);
|
2022-11-18 00:45:19 +00:00
|
|
|
}
|