import React, { forwardRef, useImperativeHandle } from 'react'; import { Alert, Share } from 'react-native'; import Clipboard from '@react-native-clipboard/clipboard'; import { connect } from 'react-redux'; import moment from 'moment'; import database from '../../lib/database'; import { getMessageById } from '../../lib/database/services/Message'; import I18n from '../../i18n'; import log, { logEvent } from '../../lib/methods/helpers/log'; import Navigation from '../../lib/navigation/appNavigation'; import { getMessageTranslation } from '../message/utils'; import { LISTENER } from '../Toast'; import EventEmitter from '../../lib/methods/helpers/events'; import { showConfirmationAlert } from '../../lib/methods/helpers/info'; import { TActionSheetOptionsItem, useActionSheet, ACTION_SHEET_ANIMATION_DURATION } from '../ActionSheet'; import Header, { HEADER_HEIGHT, IHeader } from './Header'; import events from '../../lib/methods/helpers/log/events'; import { IApplicationState, IEmoji, ILoggedUser, TAnyMessage, TSubscriptionModel } from '../../definitions'; import { getPermalinkMessage } from '../../lib/methods'; import { getRoomTitle, getUidDirectMessage, hasPermission } from '../../lib/methods/helpers'; import { Services } from '../../lib/services'; export interface IMessageActionsProps { room: TSubscriptionModel; tmid?: string; user: Pick; editInit: (message: TAnyMessage) => void; reactionInit: (message: TAnyMessage) => void; onReactionPress: (shortname: IEmoji, messageId: string) => void; replyInit: (message: TAnyMessage, mention: boolean) => void; isMasterDetail: boolean; isReadOnly: boolean; Message_AllowDeleting?: boolean; Message_AllowDeleting_BlockDeleteInMinutes?: number; Message_AllowEditing?: boolean; Message_AllowEditing_BlockEditInMinutes?: number; Message_AllowPinning?: boolean; Message_AllowStarring?: boolean; Message_Read_Receipt_Store_Users?: boolean; editMessagePermission?: string[]; deleteMessagePermission?: string[]; forceDeleteMessagePermission?: string[]; deleteOwnMessagePermission?: string[]; pinMessagePermission?: string[]; createDirectMessagePermission?: string[]; } export interface IMessageActions { showMessageActions: (message: TAnyMessage) => Promise; } const MessageActions = React.memo( forwardRef( ( { room, tmid, user, editInit, reactionInit, onReactionPress, replyInit, isReadOnly, Message_AllowDeleting, Message_AllowDeleting_BlockDeleteInMinutes, Message_AllowEditing, Message_AllowEditing_BlockEditInMinutes, Message_AllowPinning, Message_AllowStarring, Message_Read_Receipt_Store_Users, isMasterDetail, editMessagePermission, deleteMessagePermission, forceDeleteMessagePermission, deleteOwnMessagePermission, pinMessagePermission, createDirectMessagePermission }, ref ) => { let permissions = { hasEditPermission: false, hasDeletePermission: false, hasForceDeletePermission: false, hasPinPermission: false, hasDeleteOwnPermission: false }; const { showActionSheet, hideActionSheet } = useActionSheet(); const getPermissions = async () => { try { const permission = [ editMessagePermission, deleteMessagePermission, forceDeleteMessagePermission, pinMessagePermission, deleteOwnMessagePermission ]; const result = await hasPermission(permission, room.rid); permissions = { hasEditPermission: result[0], hasDeletePermission: result[1], hasForceDeletePermission: result[2], hasPinPermission: result[3], hasDeleteOwnPermission: result[4] }; } catch { // Do nothing } }; const isOwn = (message: TAnyMessage) => message.u && message.u._id === user.id; const allowEdit = (message: TAnyMessage) => { if (isReadOnly) { return false; } const editOwn = isOwn(message); if (!(permissions.hasEditPermission || (Message_AllowEditing && editOwn))) { return false; } const blockEditInMinutes = Message_AllowEditing_BlockEditInMinutes; if (blockEditInMinutes) { let msgTs; if (message.ts != null) { msgTs = moment(message.ts); } let currentTsDiff = 0; if (msgTs != null) { currentTsDiff = moment().diff(msgTs, 'minutes'); } return currentTsDiff < blockEditInMinutes; } return true; }; const allowDelete = (message: TAnyMessage) => { if (isReadOnly) { return false; } // Prevent from deleting thread start message when positioned inside the thread if (tmid === message.id) { return false; } const deleteOwn = isOwn(message) && permissions.hasDeleteOwnPermission; if (!(permissions.hasDeletePermission || (Message_AllowDeleting && deleteOwn) || permissions.hasForceDeletePermission)) { return false; } if (permissions.hasForceDeletePermission) { return true; } const blockDeleteInMinutes = Message_AllowDeleting_BlockDeleteInMinutes; if (blockDeleteInMinutes != null && blockDeleteInMinutes !== 0) { let msgTs; if (message.ts != null) { msgTs = moment(message.ts); } let currentTsDiff = 0; if (msgTs != null) { currentTsDiff = moment().diff(msgTs, 'minutes'); } return currentTsDiff < blockDeleteInMinutes; } return true; }; const getPermalink = (message: TAnyMessage) => getPermalinkMessage(message); const handleReply = (message: TAnyMessage) => { logEvent(events.ROOM_MSG_ACTION_REPLY); replyInit(message, true); }; const handleEdit = (message: TAnyMessage) => { logEvent(events.ROOM_MSG_ACTION_EDIT); editInit(message); }; const handleCreateDiscussion = (message: TAnyMessage) => { logEvent(events.ROOM_MSG_ACTION_DISCUSSION); const params = { message, channel: room, showCloseModal: true }; if (isMasterDetail) { Navigation.navigate('ModalStackNavigator', { screen: 'CreateDiscussionView', params }); } else { Navigation.navigate('NewMessageStackNavigator', { screen: 'CreateDiscussionView', params }); } }; const handleUnread = async (message: TAnyMessage) => { logEvent(events.ROOM_MSG_ACTION_UNREAD); const { id: messageId, ts } = message; const { rid } = room; try { const db = database.active; const result = await Services.markAsUnread({ messageId }); if (result.success) { const subCollection = db.get('subscriptions'); const subRecord = await subCollection.find(rid); await db.write(async () => { try { await subRecord.update(sub => (sub.lastOpen = ts as Date)); // TODO: reevaluate IMessage } catch { // do nothing } }); Navigation.navigate('RoomsListView'); } } catch (e) { logEvent(events.ROOM_MSG_ACTION_UNREAD_F); log(e); } }; const handlePermalink = async (message: TAnyMessage) => { logEvent(events.ROOM_MSG_ACTION_PERMALINK); try { const permalink = await getPermalink(message); Clipboard.setString(permalink ?? ''); EventEmitter.emit(LISTENER, { message: I18n.t('Permalink_copied_to_clipboard') }); } catch { logEvent(events.ROOM_MSG_ACTION_PERMALINK_F); } }; const handleCopy = async (message: TAnyMessage) => { logEvent(events.ROOM_MSG_ACTION_COPY); await Clipboard.setString((message?.attachments?.[0]?.description || message.msg) ?? ''); EventEmitter.emit(LISTENER, { message: I18n.t('Copied_to_clipboard') }); }; const handleShare = async (message: TAnyMessage) => { logEvent(events.ROOM_MSG_ACTION_SHARE); try { const permalink = await getPermalink(message); if (permalink) { Share.share({ message: permalink }); } } catch { logEvent(events.ROOM_MSG_ACTION_SHARE_F); } }; const handleQuote = (message: TAnyMessage) => { logEvent(events.ROOM_MSG_ACTION_QUOTE); replyInit(message, false); }; const handleReplyInDM = async (message: TAnyMessage) => { if (message?.u?.username) { const result = await Services.createDirectMessage(message.u.username); if (result.success) { const { room } = result; const params = { rid: room.rid, name: getRoomTitle(room), t: room.t, roomUserId: getUidDirectMessage(room), replyInDM: message }; Navigation.replace('RoomView', params); } } }; const handleStar = async (message: TAnyMessage) => { logEvent(message.starred ? events.ROOM_MSG_ACTION_UNSTAR : events.ROOM_MSG_ACTION_STAR); try { await Services.toggleStarMessage(message.id, message.starred as boolean); // TODO: reevaluate `message.starred` type on IMessage EventEmitter.emit(LISTENER, { message: message.starred ? I18n.t('Message_unstarred') : I18n.t('Message_starred') }); } catch (e) { logEvent(events.ROOM_MSG_ACTION_STAR_F); log(e); } }; const handlePin = async (message: TAnyMessage) => { logEvent(events.ROOM_MSG_ACTION_PIN); try { await Services.togglePinMessage(message.id, message.pinned as boolean); // TODO: reevaluate `message.pinned` type on IMessage } catch (e) { logEvent(events.ROOM_MSG_ACTION_PIN_F); log(e); } }; const handleReaction: IHeader['handleReaction'] = (emoji, message) => { logEvent(events.ROOM_MSG_ACTION_REACTION); if (emoji) { onReactionPress(emoji, message.id); } else { setTimeout(() => reactionInit(message), ACTION_SHEET_ANIMATION_DURATION); } hideActionSheet(); }; const handleReadReceipt = (message: TAnyMessage) => { if (isMasterDetail) { Navigation.navigate('ModalStackNavigator', { screen: 'ReadReceiptsView', params: { messageId: message.id } }); } else { Navigation.navigate('ReadReceiptsView', { messageId: message.id }); } }; const handleToggleTranslation = async (message: TAnyMessage) => { try { if (!room.autoTranslateLanguage) { return; } const db = database.active; const messageRecord = await getMessageById(message.id); if (!messageRecord) { return; } await db.write(async () => { await messageRecord.update(m => { m.autoTranslate = !m.autoTranslate; m._updatedAt = new Date(); }); }); const translatedMessage = getMessageTranslation(message, room.autoTranslateLanguage); if (!translatedMessage) { await Services.translateMessage(message.id, room.autoTranslateLanguage); } } catch (e) { log(e); } }; const handleReport = async (message: TAnyMessage) => { logEvent(events.ROOM_MSG_ACTION_REPORT); try { await Services.reportMessage(message.id); Alert.alert(I18n.t('Message_Reported')); } catch (e) { logEvent(events.ROOM_MSG_ACTION_REPORT_F); log(e); } }; const handleDelete = (message: TAnyMessage) => { showConfirmationAlert({ message: I18n.t('You_will_not_be_able_to_recover_this_message'), confirmationText: I18n.t('Delete'), onPress: async () => { try { logEvent(events.ROOM_MSG_ACTION_DELETE); await Services.deleteMessage(message.id, message.rid); } catch (e) { logEvent(events.ROOM_MSG_ACTION_DELETE_F); log(e); } } }); }; const getOptions = (message: TAnyMessage) => { const options: TActionSheetOptionsItem[] = []; const videoConfBlock = message.t === 'videoconf'; // Quote if (!isReadOnly && !videoConfBlock) { options.push({ title: I18n.t('Quote'), icon: 'quote', onPress: () => handleQuote(message) }); } // Reply if (!isReadOnly && !tmid) { options.push({ title: I18n.t('Reply_in_Thread'), icon: 'threads', onPress: () => handleReply(message) }); } // Reply in DM if (room.t !== 'd' && room.t !== 'l' && createDirectMessagePermission && !videoConfBlock) { options.push({ title: I18n.t('Reply_in_direct_message'), icon: 'arrow-back', onPress: () => handleReplyInDM(message) }); } // Create Discussion options.push({ title: I18n.t('Start_a_Discussion'), icon: 'discussions', onPress: () => handleCreateDiscussion(message) }); // Permalink options.push({ title: I18n.t('Get_link'), icon: 'link', onPress: () => handlePermalink(message) }); // Copy if (!videoConfBlock) { options.push({ title: I18n.t('Copy'), icon: 'copy', onPress: () => handleCopy(message) }); } // Share options.push({ title: I18n.t('Share'), icon: 'share', onPress: () => handleShare(message) }); // Edit if (allowEdit(message) && !videoConfBlock) { options.push({ title: I18n.t('Edit'), icon: 'edit', onPress: () => handleEdit(message) }); } // Pin if (Message_AllowPinning && permissions?.hasPinPermission && !videoConfBlock) { options.push({ title: I18n.t(message.pinned ? 'Unpin' : 'Pin'), icon: 'pin', onPress: () => handlePin(message) }); } // Star if (Message_AllowStarring && !videoConfBlock) { options.push({ title: I18n.t(message.starred ? 'Unstar' : 'Star'), icon: message.starred ? 'star-filled' : 'star', onPress: () => handleStar(message) }); } // Mark as unread if (message.u && message.u._id !== user.id) { options.push({ title: I18n.t('Mark_unread'), icon: 'flag', onPress: () => handleUnread(message) }); } // Read Receipts if (Message_Read_Receipt_Store_Users) { options.push({ title: I18n.t('Read_Receipt'), icon: 'info', onPress: () => handleReadReceipt(message) }); } // Toggle Auto-translate if (room.autoTranslate && message.u && message.u._id !== user.id) { options.push({ title: I18n.t(message.autoTranslate ? 'View_Original' : 'Translate'), icon: 'language', onPress: () => handleToggleTranslation(message) }); } // Report options.push({ title: I18n.t('Report'), icon: 'warning', danger: true, onPress: () => handleReport(message) }); // Delete if (allowDelete(message)) { options.push({ title: I18n.t('Delete'), icon: 'delete', danger: true, onPress: () => handleDelete(message) }); } return options; }; const showMessageActions = async (message: TAnyMessage) => { logEvent(events.ROOM_SHOW_MSG_ACTIONS); await getPermissions(); showActionSheet({ options: getOptions(message), headerHeight: HEADER_HEIGHT, customHeader: !isReadOnly || room.reactWhenReadOnly ? (
) : null }); }; useImperativeHandle(ref, () => ({ showMessageActions })); return null; } ) ); const mapStateToProps = (state: IApplicationState) => ({ server: state.server.server, Message_AllowDeleting: state.settings.Message_AllowDeleting as boolean, Message_AllowDeleting_BlockDeleteInMinutes: state.settings.Message_AllowDeleting_BlockDeleteInMinutes as number, Message_AllowEditing: state.settings.Message_AllowEditing as boolean, Message_AllowEditing_BlockEditInMinutes: state.settings.Message_AllowEditing_BlockEditInMinutes as number, Message_AllowPinning: state.settings.Message_AllowPinning as boolean, Message_AllowStarring: state.settings.Message_AllowStarring as boolean, Message_Read_Receipt_Store_Users: state.settings.Message_Read_Receipt_Store_Users as boolean, isMasterDetail: state.app.isMasterDetail, editMessagePermission: state.permissions['edit-message'], deleteMessagePermission: state.permissions['delete-message'], deleteOwnMessagePermission: state.permissions['delete-own-message'], forceDeleteMessagePermission: state.permissions['force-delete-message'], pinMessagePermission: state.permissions['pin-message'], createDirectMessagePermission: state.permissions['create-d'] }); export default connect(mapStateToProps, null, null, { forwardRef: true })(MessageActions);