forked from Ivasoft/mattermost-mobile
* Use Compass icons * Update ChannelInfo and AdvancedSettings * Fix search modifiers * Fix Autocomplete item * Remove VectorIcon component * Unlink react-native-vector-icons * Revert ProgressiveImage changes * Update Mark as Unread icon * Apply review suggestion * Replace extension icons * Update video control button style * Replace (un)flag with (un)save * Replace (un)flag with (un)save - take 2 * Use bookmark-outline icon Co-authored-by: Miguel Alatzar <miguel@Miguels-MacBook-Pro.local> Co-authored-by: Miguel Alatzar <miguel@Miguels-MacBook-Pro.local>
227 lines
7.1 KiB
JavaScript
227 lines
7.1 KiB
JavaScript
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
import React from 'react';
|
|
import PropTypes from 'prop-types';
|
|
import {StyleSheet, Text} from 'react-native';
|
|
import Clipboard from '@react-native-community/clipboard';
|
|
import {intlShape} from 'react-intl';
|
|
import {displayUsername} from '@mm-redux/utils/user_utils';
|
|
|
|
import {showModal} from '@actions/navigation';
|
|
import CompassIcon from '@components/compass_icon';
|
|
import CustomPropTypes from '@constants/custom_prop_types';
|
|
import BottomSheet from '@utils/bottom_sheet';
|
|
import mattermostManaged from 'app/mattermost_managed';
|
|
|
|
export default class AtMention extends React.PureComponent {
|
|
static propTypes = {
|
|
isSearchResult: PropTypes.bool,
|
|
mentionKeys: PropTypes.array.isRequired,
|
|
mentionName: PropTypes.string.isRequired,
|
|
mentionStyle: CustomPropTypes.Style,
|
|
onPostPress: PropTypes.func,
|
|
textStyle: CustomPropTypes.Style,
|
|
teammateNameDisplay: PropTypes.string,
|
|
theme: PropTypes.object.isRequired,
|
|
usersByUsername: PropTypes.object.isRequired,
|
|
groupsByName: PropTypes.object,
|
|
};
|
|
|
|
static contextTypes = {
|
|
intl: intlShape,
|
|
};
|
|
|
|
constructor(props) {
|
|
super(props);
|
|
|
|
const user = this.getUserDetailsFromMentionName();
|
|
this.state = {
|
|
user,
|
|
};
|
|
}
|
|
|
|
componentDidUpdate(prevProps) {
|
|
if (this.props.mentionName !== prevProps.mentionName || this.props.usersByUsername !== prevProps.usersByUsername) {
|
|
this.updateUsername();
|
|
}
|
|
}
|
|
|
|
goToUserProfile = async () => {
|
|
const {intl} = this.context;
|
|
const {theme} = this.props;
|
|
const screen = 'UserProfile';
|
|
const title = intl.formatMessage({id: 'mobile.routes.user_profile', defaultMessage: 'Profile'});
|
|
const passProps = {
|
|
userId: this.state.user.id,
|
|
};
|
|
|
|
if (!this.closeButton) {
|
|
this.closeButton = await CompassIcon.getImageSource('close', 24, theme.sidebarHeaderTextColor);
|
|
}
|
|
|
|
const options = {
|
|
topBar: {
|
|
leftButtons: [{
|
|
id: 'close-settings',
|
|
icon: this.closeButton,
|
|
}],
|
|
},
|
|
};
|
|
|
|
showModal(screen, title, passProps, options);
|
|
};
|
|
|
|
getUserDetailsFromMentionName() {
|
|
const {usersByUsername} = this.props;
|
|
let mentionName = this.props.mentionName.toLowerCase();
|
|
|
|
while (mentionName.length > 0) {
|
|
if (usersByUsername.hasOwnProperty(mentionName)) {
|
|
return usersByUsername[mentionName];
|
|
}
|
|
|
|
// Repeatedly trim off trailing punctuation in case this is at the end of a sentence
|
|
if ((/[._-]$/).test(mentionName)) {
|
|
mentionName = mentionName.substring(0, mentionName.length - 1);
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
return {
|
|
username: '',
|
|
};
|
|
}
|
|
|
|
getGroupFromMentionName() {
|
|
const {groupsByName, mentionName} = this.props;
|
|
const mentionNameTrimmed = mentionName.toLowerCase().replace(/[._-]*$/, '');
|
|
return groupsByName?.[mentionNameTrimmed] || {};
|
|
}
|
|
|
|
handleLongPress = async () => {
|
|
const {formatMessage} = this.context.intl;
|
|
|
|
const config = mattermostManaged.getCachedConfig();
|
|
|
|
if (config?.copyAndPasteProtection !== 'true') {
|
|
const cancelText = formatMessage({id: 'mobile.post.cancel', defaultMessage: 'Cancel'});
|
|
const actionText = formatMessage({id: 'mobile.mention.copy_mention', defaultMessage: 'Copy Mention'});
|
|
|
|
BottomSheet.showBottomSheetWithOptions({
|
|
options: [actionText, cancelText],
|
|
cancelButtonIndex: 1,
|
|
}, (value) => {
|
|
if (value !== 1) {
|
|
this.handleCopyMention();
|
|
}
|
|
});
|
|
}
|
|
};
|
|
|
|
handleCopyMention = () => {
|
|
const {user} = this.state;
|
|
const {mentionName} = this.props;
|
|
let username = mentionName;
|
|
if (user.username) {
|
|
username = user.username;
|
|
}
|
|
|
|
Clipboard.setString(`@${username}`);
|
|
};
|
|
|
|
updateUsername = () => {
|
|
const user = this.getUserDetailsFromMentionName();
|
|
this.setState({
|
|
user,
|
|
});
|
|
}
|
|
|
|
render() {
|
|
const {isSearchResult, mentionName, mentionStyle, onPostPress, teammateNameDisplay, textStyle, mentionKeys} = this.props;
|
|
const {user} = this.state;
|
|
const mentionTextStyle = [];
|
|
|
|
let backgroundColor;
|
|
let canPress = false;
|
|
let highlighted;
|
|
let isMention = false;
|
|
let mention;
|
|
let onLongPress;
|
|
let onPress;
|
|
let suffix;
|
|
let suffixElement;
|
|
let styleText;
|
|
|
|
if (textStyle) {
|
|
const {backgroundColor: bg, ...otherStyles} = StyleSheet.flatten(textStyle);
|
|
backgroundColor = bg;
|
|
styleText = otherStyles;
|
|
}
|
|
|
|
if (user?.username) {
|
|
suffix = this.props.mentionName.substring(user.username.length);
|
|
highlighted = mentionKeys.some((item) => item.key.includes(user.username));
|
|
mention = displayUsername(user, teammateNameDisplay);
|
|
isMention = true;
|
|
canPress = true;
|
|
} else {
|
|
const group = this.getGroupFromMentionName();
|
|
if (group.allow_reference) {
|
|
highlighted = mentionKeys.some((item) => item.key === `@${group.name}`);
|
|
isMention = true;
|
|
mention = group.name;
|
|
suffix = this.props.mentionName.substring(group.name.length);
|
|
} else {
|
|
const pattern = new RegExp(/\b(all|channel|here)(?:\.\B|_\b|\b)/, 'i');
|
|
const mentionMatch = pattern.exec(mentionName);
|
|
highlighted = true;
|
|
|
|
if (mentionMatch) {
|
|
mention = mentionMatch.length > 1 ? mentionMatch[1] : mentionMatch[0];
|
|
suffix = mentionName.replace(mention, '');
|
|
isMention = true;
|
|
} else {
|
|
mention = mentionName;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (canPress) {
|
|
onLongPress = this.handleLongPress;
|
|
onPress = isSearchResult ? onPostPress : this.goToUserProfile;
|
|
}
|
|
|
|
if (suffix) {
|
|
const suffixStyle = {...styleText, color: this.props.theme.centerChannelColor};
|
|
suffixElement = (
|
|
<Text style={suffixStyle}>
|
|
{suffix}
|
|
</Text>
|
|
);
|
|
}
|
|
|
|
if (isMention) {
|
|
mentionTextStyle.push(mentionStyle);
|
|
}
|
|
|
|
if (highlighted) {
|
|
mentionTextStyle.push({backgroundColor});
|
|
}
|
|
|
|
return (
|
|
<Text
|
|
style={styleText}
|
|
onPress={onPress}
|
|
onLongPress={onLongPress}
|
|
>
|
|
<Text style={mentionTextStyle}>
|
|
{'@' + mention}
|
|
</Text>
|
|
{suffixElement}
|
|
</Text>
|
|
);
|
|
}
|
|
}
|