forked from Ivasoft/mattermost-mobile
* Some extra work on voice channels interface * Fixing some TODOs * Improving styling of call in channel * Improve calls monitoring * Replacing some of the fontawesome icons with the compass ones * Improving the layout * Migrating to webrtc2 for unified plan * Add screen on and off behavior * Adding incall manager plugin * Moving everything into the products/calls folder * Make products modules routes relatives * Make products modules routes @mmproducts * Removing initiator parameter * Removing trickle parameter * Simplifying code * Removing underscore from private variables * Removing underscore from private things * More simplifications * More simplifications * More simplifications * Changing sha sum for mmjstool * Fixing typo * Migrating simple-peer to typescript * Migrating simple-peer to typescript * Improving the size of the screen share * Adding feature flag to disable the calls feature in mobile * Fixing some tests * Removing obsolte tests * Added call ended support for the post messages * Fixing some warnings in the tests * Adding JoinCall tests * Adding CallMessage tests * Adding CurrentCall unit tests * Adding CallAvatar unit tests * Adding FloatingCallContainer unit tests * Adding StartCall unit tests * Adding EnableDisableCalls unit tests * Adding CallDuration tests * Improving CallDuration tests * Adding CallScreen unit tests * Adding CallOtherActions screen tests * Fixing some dark theme styles * Fixing tests * More robustness around connecting/disconnecting * Adding FormattedRelativeTime tests * Adding tests for ChannelItem * Adding tests for ChannelInfo * Adding selectors tests * Adding reducers unit tests * Adding actions tests * Removing most of the TODOs * Removing another TODO * Updating tests snapshots * Removing the last TODO * Fixed a small problem on pressing while a call is ongoing * Remove all the inlined functions * Replacing usage of isLandscape selector with useWindowDimensions * Removed unnecesary makeStyleSheetFromTheme * Removing unneded properties from call_duration * Fixing possible null channels return from getChannel selector * Moving other inlined functions to its own constant * Simplifiying enable/disable calls component * Improving the behavior when you are in the call of the current channel * Adding missing translation strings * Simplified a bit the EnableDisableCalls component * Moving other inlined functions to its own constant * Updating snapshots * Improving usage of makeStyleSheetFromTheme * Moving data reformating from the rest client to the redux action * Adding calls to the blocklist to the redux-persist * Fixing tests * Updating snapshots * Update file icon name to the last compass icons version * Fix loading state * Only show the call connected if the websocket gets connected * Taking into consideration the indicator bar to position the calls new bars * Making the MoreMessagesButton component aware of calls components * Updating snapshots * Fixing tests * Updating snapshot * Fixing different use cases for start call channel menu * Fixing tests * Ask for confirmation to start a call when you are already in another call * Update app/products/calls/components/floating_call_container.tsx Co-authored-by: Elias Nahum <nahumhbl@gmail.com> * Memoizing userIds in join call * Applying suggestion around combine the blocklist for calls with the one for typing * Adding explicit types to the rest client * Removing unneeded permission * Making updateIntervalInSeconds prop optional in FormattedRelativeTime * Making updateIntervalInSeconds prop optional in CallDuration Co-authored-by: Elias Nahum <nahumhbl@gmail.com>
333 lines
11 KiB
TypeScript
333 lines
11 KiB
TypeScript
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
import React, {ReactNode, useRef} from 'react';
|
|
import {injectIntl, intlShape} from 'react-intl';
|
|
import {Keyboard, StyleProp, View, ViewStyle} from 'react-native';
|
|
|
|
import {showModalOverCurrentContext} from '@actions/navigation';
|
|
import ThreadFooter from '@components/global_threads/thread_footer';
|
|
import SystemAvatar from '@components/post_list/system_avatar';
|
|
import SystemHeader from '@components/post_list/system_header';
|
|
import TouchableWithFeedback from '@components/touchable_with_feedback';
|
|
import * as Screens from '@constants/screen';
|
|
import {Posts} from '@mm-redux/constants';
|
|
import {UserThread} from '@mm-redux/types/threads';
|
|
import {UserProfile} from '@mm-redux/types/users';
|
|
import EventEmitter from '@mm-redux/utils/event_emitter';
|
|
import {fromAutoResponder, isPostEphemeral, isPostPendingOrFailed, isSystemMessage} from '@mm-redux/utils/post_utils';
|
|
import CallMessage from '@mmproducts/calls/components/call_message';
|
|
import {preventDoubleTap} from '@utils/tap';
|
|
import {changeOpacity, makeStyleSheetFromTheme} from '@utils/theme';
|
|
|
|
import Avatar from './avatar';
|
|
import Body from './body';
|
|
import Header from './header';
|
|
import PreHeader from './pre_header';
|
|
import SystemMessage from './system_message';
|
|
|
|
import type {Post as PostType} from '@mm-redux/types/posts';
|
|
import type {Theme} from '@mm-redux/types/theme';
|
|
|
|
type PostProps = {
|
|
canDelete: boolean;
|
|
collapsedThreadsEnabled: boolean;
|
|
enablePostUsernameOverride: boolean;
|
|
highlight?: boolean;
|
|
highlightPinnedOrFlagged?: boolean;
|
|
intl: typeof intlShape;
|
|
isConsecutivePost?: boolean;
|
|
isFirstReply?: boolean;
|
|
isFlagged?: boolean;
|
|
isLastReply?: boolean;
|
|
location: string;
|
|
post?: PostType;
|
|
removePost: (post: PostType) => void;
|
|
rootPostAuthor?: string;
|
|
shouldRenderReplyButton?: boolean;
|
|
showAddReaction?: boolean;
|
|
showPermalink: (intl: typeof intlShape, teamName: string, postId: string) => null;
|
|
skipFlaggedHeader?: boolean;
|
|
skipPinnedHeader?: boolean;
|
|
style?: StyleProp<ViewStyle>;
|
|
teammateNameDisplay: string;
|
|
testID?: string;
|
|
theme: Theme;
|
|
thread: UserThread;
|
|
threadStarter: UserProfile;
|
|
callsFeatureEnabled: boolean;
|
|
};
|
|
|
|
const getStyleSheet = makeStyleSheetFromTheme((theme: Theme) => {
|
|
return {
|
|
consecutive: {marginTop: 0},
|
|
consecutivePostContainer: {
|
|
marginBottom: 10,
|
|
marginRight: 10,
|
|
marginLeft: 47,
|
|
marginTop: 10,
|
|
},
|
|
container: {flexDirection: 'row'},
|
|
highlight: {backgroundColor: changeOpacity(theme.mentionHighlightBg, 0.5)},
|
|
highlightBar: {
|
|
backgroundColor: theme.mentionHighlightBg,
|
|
opacity: 1,
|
|
},
|
|
highlightPinnedOrFlagged: {backgroundColor: changeOpacity(theme.mentionHighlightBg, 0.2)},
|
|
badgeContainer: {
|
|
position: 'absolute',
|
|
left: 28,
|
|
bottom: 9,
|
|
},
|
|
unreadDot: {
|
|
width: 8,
|
|
height: 8,
|
|
borderRadius: 4,
|
|
backgroundColor: theme.sidebarTextActiveBorder,
|
|
alignSelf: 'center',
|
|
top: -6,
|
|
left: 4,
|
|
},
|
|
pendingPost: {opacity: 0.5},
|
|
postStyle: {
|
|
overflow: 'hidden',
|
|
flex: 1,
|
|
},
|
|
replyBar: {
|
|
backgroundColor: theme.centerChannelColor,
|
|
opacity: 0.1,
|
|
marginLeft: 1,
|
|
marginRight: 7,
|
|
width: 3,
|
|
flexBasis: 3,
|
|
},
|
|
replyBarFirst: {paddingTop: 10},
|
|
replyBarLast: {paddingBottom: 10},
|
|
rightColumn: {
|
|
flex: 1,
|
|
flexDirection: 'column',
|
|
marginRight: 12,
|
|
},
|
|
rightColumnPadding: {paddingBottom: 3},
|
|
};
|
|
});
|
|
|
|
const Post = ({
|
|
canDelete, collapsedThreadsEnabled, enablePostUsernameOverride, highlight, highlightPinnedOrFlagged = true, intl, isConsecutivePost, isFirstReply, isFlagged, isLastReply,
|
|
location, post, removePost, rootPostAuthor, shouldRenderReplyButton, skipFlaggedHeader, skipPinnedHeader, showAddReaction = true, showPermalink, style,
|
|
teammateNameDisplay, testID, theme, thread, threadStarter, callsFeatureEnabled,
|
|
}: PostProps) => {
|
|
const pressDetected = useRef(false);
|
|
const styles = getStyleSheet(theme);
|
|
|
|
const handlePress = preventDoubleTap(() => {
|
|
pressDetected.current = true;
|
|
|
|
if (post) {
|
|
if (location === Screens.THREAD) {
|
|
Keyboard.dismiss();
|
|
} else if (location === Screens.SEARCH) {
|
|
showPermalink(intl, '', post.id);
|
|
return;
|
|
}
|
|
|
|
const isValidSystemMessage = fromAutoResponder(post) || !isSystemMessage(post);
|
|
if (post.state !== Posts.POST_DELETED && isValidSystemMessage && !isPostPendingOrFailed(post)) {
|
|
if ([Screens.CHANNEL, Screens.PERMALINK].includes(location)) {
|
|
EventEmitter.emit('goToThread', post);
|
|
}
|
|
} else if ((isPostEphemeral(post) || post.state === Posts.POST_DELETED)) {
|
|
removePost(post);
|
|
}
|
|
|
|
const pressTimeout = setTimeout(() => {
|
|
pressDetected.current = false;
|
|
clearTimeout(pressTimeout);
|
|
}, 300);
|
|
}
|
|
});
|
|
|
|
const showPostOptions = () => {
|
|
if (!post) {
|
|
return;
|
|
}
|
|
|
|
const hasBeenDeleted = (post.delete_at !== 0 || post.state === Posts.POST_DELETED);
|
|
if (isSystemMessage(post) && (!canDelete || hasBeenDeleted)) {
|
|
return;
|
|
}
|
|
|
|
if (isPostPendingOrFailed(post) || isPostEphemeral(post)) {
|
|
return;
|
|
}
|
|
|
|
const screen = 'PostOptions';
|
|
const passProps = {
|
|
location,
|
|
post,
|
|
showAddReaction,
|
|
};
|
|
|
|
Keyboard.dismiss();
|
|
const postOptionsRequest = requestAnimationFrame(() => {
|
|
showModalOverCurrentContext(screen, passProps);
|
|
cancelAnimationFrame(postOptionsRequest);
|
|
});
|
|
};
|
|
|
|
if (!post) {
|
|
return null;
|
|
}
|
|
|
|
const highlightFlagged = isFlagged && !skipFlaggedHeader;
|
|
const hightlightPinned = post.is_pinned && !skipPinnedHeader;
|
|
const itemTestID = `${testID}.${post.id}`;
|
|
const rightColumnStyle = [styles.rightColumn, (post.root_id && isLastReply && styles.rightColumnPadding)];
|
|
const pendingPostStyle: StyleProp<ViewStyle> | undefined = isPostPendingOrFailed(post) ? styles.pendingPost : undefined;
|
|
const isAutoResponder = fromAutoResponder(post);
|
|
|
|
let highlightedStyle: StyleProp<ViewStyle>;
|
|
if (highlight) {
|
|
highlightedStyle = styles.highlight;
|
|
} else if ((highlightFlagged || hightlightPinned) && highlightPinnedOrFlagged) {
|
|
highlightedStyle = styles.highlightPinnedOrFlagged;
|
|
}
|
|
|
|
let header: ReactNode;
|
|
let postAvatar: ReactNode;
|
|
let consecutiveStyle: StyleProp<ViewStyle>;
|
|
if (isConsecutivePost) {
|
|
consecutiveStyle = styles.consective;
|
|
postAvatar = <View style={styles.consecutivePostContainer}/>;
|
|
} else {
|
|
postAvatar = isAutoResponder ? (
|
|
<SystemAvatar theme={theme}/>
|
|
) : (
|
|
<Avatar
|
|
pendingPostStyle={pendingPostStyle}
|
|
post={post}
|
|
theme={theme}
|
|
/>
|
|
);
|
|
|
|
if (isSystemMessage(post) && !isAutoResponder) {
|
|
header = (
|
|
<SystemHeader
|
|
createAt={post.create_at}
|
|
theme={theme}
|
|
/>
|
|
);
|
|
} else {
|
|
header = (
|
|
<Header
|
|
collapsedThreadsEnabled={collapsedThreadsEnabled}
|
|
enablePostUsernameOverride={enablePostUsernameOverride}
|
|
location={location}
|
|
post={post}
|
|
rootPostAuthor={rootPostAuthor}
|
|
shouldRenderReplyButton={shouldRenderReplyButton}
|
|
teammateNameDisplay={teammateNameDisplay}
|
|
theme={theme}
|
|
/>
|
|
);
|
|
}
|
|
}
|
|
|
|
let body;
|
|
if (isSystemMessage(post) && !isPostEphemeral(post) && !isAutoResponder) {
|
|
body = (
|
|
<SystemMessage
|
|
post={post}
|
|
theme={theme}
|
|
/>
|
|
);
|
|
} else if (post.type === 'custom_calls' && callsFeatureEnabled) {
|
|
body = (
|
|
<CallMessage
|
|
post={post}
|
|
theme={theme}
|
|
/>
|
|
);
|
|
} else {
|
|
body = (
|
|
<Body
|
|
highlight={Boolean(highlightedStyle)}
|
|
isFirstReply={isFirstReply}
|
|
isLastReply={isLastReply}
|
|
location={location}
|
|
post={post}
|
|
rootPostAuthor={rootPostAuthor}
|
|
showAddReaction={showAddReaction}
|
|
theme={theme}
|
|
/>
|
|
);
|
|
}
|
|
|
|
let footer;
|
|
if (
|
|
collapsedThreadsEnabled &&
|
|
Boolean(thread) &&
|
|
post.state !== Posts.POST_DELETED &&
|
|
(thread?.is_following || thread?.participants?.length)
|
|
) {
|
|
footer = (
|
|
<ThreadFooter
|
|
testID={`${itemTestID}.footer`}
|
|
theme={theme}
|
|
thread={thread}
|
|
threadStarter={threadStarter}
|
|
location={Screens.CHANNEL}
|
|
/>
|
|
);
|
|
}
|
|
|
|
let badge;
|
|
if (thread?.unread_mentions || thread?.unread_replies) {
|
|
if (thread.unread_replies && thread.unread_replies > 0) {
|
|
badge = (
|
|
<View style={styles.badgeContainer}>
|
|
<View style={styles.unreadDot}/>
|
|
</View>
|
|
);
|
|
}
|
|
}
|
|
|
|
return (
|
|
<View
|
|
testID={testID}
|
|
style={[styles.postStyle, style, highlightedStyle]}
|
|
>
|
|
<TouchableWithFeedback
|
|
testID={itemTestID}
|
|
onPress={handlePress}
|
|
onLongPress={showPostOptions}
|
|
delayLongPress={200}
|
|
underlayColor={changeOpacity(theme.centerChannelColor, 0.1)}
|
|
cancelTouchOnPanning={true}
|
|
>
|
|
<>
|
|
<PreHeader
|
|
isConsecutivePost={isConsecutivePost}
|
|
isFlagged={isFlagged}
|
|
isPinned={post.is_pinned}
|
|
skipFlaggedHeader={skipFlaggedHeader}
|
|
skipPinnedHeader={skipPinnedHeader}
|
|
theme={theme}
|
|
/>
|
|
<View style={[styles.container, consecutiveStyle]}>
|
|
{postAvatar}
|
|
<View style={rightColumnStyle}>
|
|
{header}
|
|
{body}
|
|
{footer}
|
|
</View>
|
|
{badge}
|
|
</View>
|
|
</>
|
|
</TouchableWithFeedback>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
export default injectIntl(Post);
|