Files
mattermost-mobile/app/components/global_threads/thread_item/thread_item.tsx
Dean Whillier fb8238ab0b [MM-37553] Update default themes (#5648)
* new themes and theme type updates

* update theme processing

port newer functionality from webapp

* update theme UI

including new svg-based thumbnail

* lint fixes

* update snapshots and tests

* update theme tile border approach

* remove unused path component

* remove old variable typo

* remove old variable type from theme type

* lint and snapshot updates

* update snapshots
2021-09-03 10:50:24 -04:00

271 lines
8.0 KiB
TypeScript

// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
import React from 'react';
import {injectIntl, intlShape} from 'react-intl';
import {View, Text, TouchableHighlight} from 'react-native';
import {goToScreen} from '@actions/navigation';
import FriendlyDate from '@components/friendly_date';
import RemoveMarkdown from '@components/remove_markdown';
import {GLOBAL_THREADS, THREAD} from '@constants/screen';
import {Posts, Preferences} from '@mm-redux/constants';
import {Channel} from '@mm-redux/types/channels';
import {Post} from '@mm-redux/types/posts';
import {UserThread} from '@mm-redux/types/threads';
import {UserProfile} from '@mm-redux/types/users';
import {displayUsername} from '@mm-redux/utils/user_utils';
import {changeOpacity, makeStyleSheetFromTheme} from '@utils/theme';
import ThreadFooter from '../thread_footer';
import type {Theme} from '@mm-redux/types/theme';
export type DispatchProps = {
actions: {
getPost: (postId: string) => void;
getPostThread: (postId: string) => void;
selectPost: (postId: string) => void;
};
}
export type OwnProps = {
testID: string;
theme: Theme;
threadId: string;
};
export type StateProps = {
channel: Channel;
post: Post;
thread: UserThread | null;
threadStarter: UserProfile;
}
type Props = DispatchProps & OwnProps & StateProps & {
intl: typeof intlShape;
};
function ThreadItem({actions, channel, intl, post, threadId, testID, theme, thread, threadStarter}: Props) {
const style = getStyleSheet(theme);
if (!thread) {
return null;
}
const postItem = post || thread.post;
React.useEffect(() => {
// Get the latest post
if (!post) {
actions.getPost(threadId);
}
}, []);
const channelName = channel?.display_name;
const threadStarterName = displayUsername(threadStarter, Preferences.DISPLAY_PREFER_FULL_NAME);
const showThread = () => {
actions.getPostThread(postItem.id);
actions.selectPost(postItem.id);
const passProps = {
channelId: postItem.channel_id,
rootId: postItem.id,
};
goToScreen(THREAD, '', passProps);
};
const testIDPrefix = `${testID}.${postItem?.id}`;
const needBadge = thread.unread_mentions || thread.unread_replies;
let badgeComponent;
if (needBadge) {
if (thread.unread_mentions && thread.unread_mentions > 0) {
badgeComponent = (
<View
style={style.mentionBadge}
testID={`${testIDPrefix}.unread_mentions`}
>
<Text style={style.mentionBadgeText}>{thread.unread_mentions > 99 ? '99+' : thread.unread_mentions}</Text>
</View>
);
} else if (thread.unread_replies && thread.unread_replies > 0) {
badgeComponent = (
<View
style={style.unreadDot}
testID={`${testIDPrefix}.unread_dot`}
/>
);
}
}
let name;
if (postItem.state === Posts.POST_DELETED) {
name = (
<Text
style={[style.threadStarter, style.threadDeleted]}
numberOfLines={1}
>{intl.formatMessage({
id: 'threads.deleted',
defaultMessage: 'Original Message Deleted',
})}</Text>
);
} else {
name = (
<Text
style={style.threadStarter}
numberOfLines={1}
>{threadStarterName}</Text>
);
}
let postBody;
if (postItem.state !== Posts.POST_DELETED) {
postBody = (
<Text
style={style.message}
numberOfLines={2}
>
<RemoveMarkdown value={postItem.message || ''}/>
</Text>
);
}
return (
<TouchableHighlight
underlayColor={changeOpacity(theme.buttonBg, 0.08)}
onPress={showThread}
testID={`${testIDPrefix}.item`}
>
<View style={style.container}>
<View style={style.badgeContainer}>
{badgeComponent}
</View>
<View style={style.postContainer}>
<View style={style.header}>
<View style={style.headerInfoContainer}>
{name}
<View style={style.channelNameContainer}>
<Text
style={style.channelName}
numberOfLines={1}
>{channelName}</Text>
</View>
</View>
<FriendlyDate
value={thread.last_reply_at}
style={style.date}
/>
</View>
{postBody}
<ThreadFooter
testID={`${testIDPrefix}.footer`}
thread={thread}
theme={theme}
threadStarter={threadStarter}
location={GLOBAL_THREADS}
/>
</View>
</View>
</TouchableHighlight>
);
}
const getStyleSheet = makeStyleSheetFromTheme((theme: Theme) => {
return {
container: {
paddingTop: 16,
paddingRight: 16,
flex: 1,
flexDirection: 'row',
borderBottomColor: changeOpacity(theme.centerChannelColor, 0.08),
borderBottomWidth: 1,
},
badgeContainer: {
marginTop: 3,
width: 32,
},
postContainer: {
flex: 1,
},
header: {
alignItems: 'center',
flex: 1,
flexDirection: 'row',
marginBottom: 9,
},
headerInfoContainer: {
alignItems: 'center',
flex: 1,
flexDirection: 'row',
marginRight: 12,
overflow: 'hidden',
},
threadDeleted: {
color: changeOpacity(theme.centerChannelColor, 0.72),
fontStyle: 'italic',
},
threadStarter: {
color: theme.centerChannelColor,
fontSize: 15,
fontWeight: '600',
lineHeight: 22,
paddingRight: 8,
},
channelNameContainer: {
backgroundColor: changeOpacity(theme.centerChannelColor, 0.08),
borderRadius: 4,
maxWidth: '50%',
},
channelName: {
color: theme.centerChannelColor,
fontSize: 10,
fontWeight: '600',
lineHeight: 16,
letterSpacing: 0.1,
textTransform: 'uppercase',
marginLeft: 6,
marginRight: 6,
marginTop: 2,
marginBottom: 2,
},
date: {
color: changeOpacity(theme.centerChannelColor, 0.64),
fontSize: 12,
fontWeight: '400',
},
message: {
color: theme.centerChannelColor,
fontSize: 15,
lineHeight: 20,
},
unreadDot: {
width: 8,
height: 8,
borderRadius: 4,
backgroundColor: theme.sidebarTextActiveBorder,
alignSelf: 'center',
marginTop: 5,
},
mentionBadge: {
width: 18,
height: 18,
borderRadius: 9,
backgroundColor: theme.mentionColor,
alignSelf: 'center',
},
mentionBadgeText: {
fontFamily: 'Open Sans',
fontSize: 10,
lineHeight: 16,
fontWeight: '700',
alignSelf: 'center',
color: theme.centerChannelBg,
},
};
});
export {ThreadItem}; // Used for shallow render test cases
export default injectIntl(ThreadItem);