c32a95a885
* Fix ability to delete own comments in livestream ## Issue 5832: can't remove own comments in live stream mode (if you have multiple channels?) Looks like it was just missing `commentIsMine` for the new component. * Disable editing livestream comments It doesn't do anything at the moment, anyways. * Disable deleting hyperchats The "total tipped" will get deducted when hyperchats are deleted, which doesn't make sense (doesn't reflect actual total that the creator received).
209 lines
6.7 KiB
JavaScript
209 lines
6.7 KiB
JavaScript
// @flow
|
|
import React from 'react';
|
|
import classnames from 'classnames';
|
|
import Spinner from 'component/spinner';
|
|
import CommentCreate from 'component/commentCreate';
|
|
import LivestreamComment from 'component/livestreamComment';
|
|
import Button from 'component/button';
|
|
import UriIndicator from 'component/uriIndicator';
|
|
import CreditAmount from 'component/common/credit-amount';
|
|
import ChannelThumbnail from 'component/channelThumbnail';
|
|
import Tooltip from 'component/common/tooltip';
|
|
|
|
type Props = {
|
|
uri: string,
|
|
claim: ?StreamClaim,
|
|
activeViewers: number,
|
|
embed?: boolean,
|
|
doCommentSocketConnect: (string, string) => void,
|
|
doCommentSocketDisconnect: (string) => void,
|
|
doCommentList: (string) => void,
|
|
comments: Array<Comment>,
|
|
fetchingComments: boolean,
|
|
doSuperChatList: (string) => void,
|
|
superChats: Array<Comment>,
|
|
superChatsTotalAmount: number,
|
|
myChannels: ?Array<ChannelClaim>,
|
|
};
|
|
|
|
const VIEW_MODE_CHAT = 'view_chat';
|
|
const VIEW_MODE_SUPER_CHAT = 'view_superchat';
|
|
|
|
export default function LivestreamComments(props: Props) {
|
|
const {
|
|
claim,
|
|
uri,
|
|
embed,
|
|
doCommentSocketConnect,
|
|
doCommentSocketDisconnect,
|
|
comments,
|
|
doCommentList,
|
|
fetchingComments,
|
|
doSuperChatList,
|
|
superChats,
|
|
superChatsTotalAmount,
|
|
myChannels,
|
|
} = props;
|
|
const commentsRef = React.createRef();
|
|
const hasScrolledComments = React.useRef();
|
|
const [viewMode, setViewMode] = React.useState(VIEW_MODE_CHAT);
|
|
const [performedInitialScroll, setPerformedInitialScroll] = React.useState(false);
|
|
const claimId = claim && claim.claim_id;
|
|
const commentsLength = comments && comments.length;
|
|
const commentsToDisplay = viewMode === VIEW_MODE_CHAT ? comments : superChats;
|
|
|
|
// todo: implement comment_list --mine in SDK so redux can grab with selectCommentIsMine
|
|
function isMyComment(channelId: string) {
|
|
if (myChannels != null && channelId != null) {
|
|
for (let i = 0; i < myChannels.length; i++) {
|
|
if (myChannels[i].claim_id === channelId) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
React.useEffect(() => {
|
|
if (claimId) {
|
|
doCommentList(uri);
|
|
doSuperChatList(uri);
|
|
doCommentSocketConnect(uri, claimId);
|
|
}
|
|
|
|
return () => {
|
|
if (claimId) {
|
|
doCommentSocketDisconnect(claimId);
|
|
}
|
|
};
|
|
}, [claimId, uri, doCommentList, doSuperChatList, doCommentSocketConnect, doCommentSocketDisconnect]);
|
|
|
|
React.useEffect(() => {
|
|
const element = commentsRef.current;
|
|
|
|
function handleScroll() {
|
|
if (element) {
|
|
const scrollHeight = element.scrollHeight - element.offsetHeight;
|
|
const isAtBottom = scrollHeight <= element.scrollTop + 100;
|
|
|
|
if (!isAtBottom) {
|
|
hasScrolledComments.current = true;
|
|
} else {
|
|
hasScrolledComments.current = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (element) {
|
|
element.addEventListener('scroll', handleScroll);
|
|
|
|
if (commentsLength > 0) {
|
|
// Only update comment scroll if the user hasn't scrolled up to view old comments
|
|
// If they have, do nothing
|
|
if (!hasScrolledComments.current || !performedInitialScroll) {
|
|
setTimeout(() => (element.scrollTop = element.scrollHeight - element.offsetHeight + 100), 20);
|
|
|
|
if (!performedInitialScroll) {
|
|
setPerformedInitialScroll(true);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return () => {
|
|
if (element) {
|
|
element.removeEventListener('scroll', handleScroll);
|
|
}
|
|
};
|
|
}, [commentsRef, commentsLength, performedInitialScroll]);
|
|
|
|
if (!claim) {
|
|
return null;
|
|
}
|
|
|
|
return (
|
|
<div className="card livestream__discussion">
|
|
<div className="card__header--between livestream-discussion__header">
|
|
<div className="livestream-discussion__title">{__('Live discussion')}</div>
|
|
{superChatsTotalAmount > 0 && (
|
|
<div className="recommended-content__toggles">
|
|
<Button
|
|
className={classnames('button-toggle', {
|
|
'button-toggle--active': viewMode === VIEW_MODE_CHAT,
|
|
})}
|
|
label={__('Chat')}
|
|
onClick={() => setViewMode(VIEW_MODE_CHAT)}
|
|
/>
|
|
|
|
<Button
|
|
className={classnames('button-toggle', {
|
|
'button-toggle--active': viewMode === VIEW_MODE_SUPER_CHAT,
|
|
})}
|
|
label={
|
|
<>
|
|
<CreditAmount amount={superChatsTotalAmount} size={8} /> {__('Tipped')}
|
|
</>
|
|
}
|
|
onClick={() => setViewMode(VIEW_MODE_SUPER_CHAT)}
|
|
/>
|
|
</div>
|
|
)}
|
|
</div>
|
|
<>
|
|
{fetchingComments && !comments && (
|
|
<div className="main--empty">
|
|
<Spinner />
|
|
</div>
|
|
)}
|
|
<div ref={commentsRef} className="livestream__comments-wrapper">
|
|
{viewMode === VIEW_MODE_CHAT && superChatsTotalAmount > 0 && (
|
|
<div className="livestream-superchats__wrapper">
|
|
<div className="livestream-superchats__inner">
|
|
{superChats.map((superChat: Comment) => (
|
|
<Tooltip key={superChat.comment_id} label={superChat.comment}>
|
|
<div className="livestream-superchat">
|
|
<div className="livestream-superchat__thumbnail">
|
|
<ChannelThumbnail uri={superChat.channel_url} xsmall />
|
|
</div>
|
|
|
|
<div className="livestream-superchat__info">
|
|
<UriIndicator uri={superChat.channel_url} link />
|
|
<CreditAmount
|
|
size={10}
|
|
className="livestream-superchat__amount-large"
|
|
amount={superChat.support_amount}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</Tooltip>
|
|
))}
|
|
</div>
|
|
</div>
|
|
)}
|
|
|
|
{!fetchingComments && comments.length > 0 ? (
|
|
<div className="livestream__comments">
|
|
{commentsToDisplay.map((comment) => (
|
|
<LivestreamComment
|
|
key={comment.comment_id}
|
|
uri={uri}
|
|
authorUri={comment.channel_url}
|
|
commentId={comment.comment_id}
|
|
message={comment.comment}
|
|
supportAmount={comment.support_amount}
|
|
commentIsMine={comment.channel_id && isMyComment(comment.channel_id)}
|
|
/>
|
|
))}
|
|
</div>
|
|
) : (
|
|
<div className="main--empty" style={{ flex: 1 }} />
|
|
)}
|
|
|
|
<div className="livestream__comment-create">
|
|
<CommentCreate livestream bottom embed={embed} uri={uri} />
|
|
</div>
|
|
</div>
|
|
</>
|
|
</div>
|
|
);
|
|
}
|