import React, { Component } from 'react'; import PropTypes from 'prop-types'; import { View, Text, NativeModules } from 'react-native'; import { connect } from 'react-redux'; import ShareExtension from 'rn-extensions-share'; import { themes } from '../../constants/colors'; import I18n from '../../i18n'; import styles from './styles'; import Loading from '../../containers/Loading'; import * as HeaderButton from '../../containers/HeaderButton'; import { isBlocked } from '../../utils/room'; import { isReadOnly } from '../../utils/isReadOnly'; import { withTheme } from '../../theme'; import Header from './Header'; import RocketChat from '../../lib/rocketchat'; import TextInput from '../../containers/TextInput'; import Preview from './Preview'; import Thumbs from './Thumbs'; import MessageBox from '../../containers/MessageBox'; import SafeAreaView from '../../containers/SafeAreaView'; import { getUserSelector } from '../../selectors/login'; import StatusBar from '../../containers/StatusBar'; import database from '../../lib/database'; import { canUploadFile } from '../../utils/media'; import { isAndroid } from '../../utils/deviceInfo'; class ShareView extends Component { constructor(props) { super(props); this.messagebox = React.createRef(); this.files = props.route.params?.attachments ?? []; this.isShareExtension = props.route.params?.isShareExtension; this.serverInfo = props.route.params?.serverInfo ?? {}; this.state = { selected: {}, loading: false, readOnly: false, attachments: [], text: props.route.params?.text ?? '', room: props.route.params?.room ?? {}, thread: props.route.params?.thread ?? {}, maxFileSize: this.isShareExtension ? this.serverInfo?.FileUpload_MaxFileSize : props.FileUpload_MaxFileSize, mediaAllowList: this.isShareExtension ? this.serverInfo?.FileUpload_MediaTypeWhiteList : props.FileUpload_MediaTypeWhiteList }; this.getServerInfo(); } componentDidMount = async() => { const readOnly = await this.getReadOnly(); const { attachments, selected } = await this.getAttachments(); this.setState({ readOnly, attachments, selected }, () => this.setHeader()); } componentWillUnmount = () => { console.countReset(`${ this.constructor.name }.render calls`); } setHeader = () => { const { room, thread, readOnly, attachments } = this.state; const { navigation, theme } = this.props; const options = { headerTitle: () =>
, headerTitleAlign: 'left', headerTintColor: themes[theme].previewTintColor }; // if is share extension show default back button if (!this.isShareExtension) { options.headerLeft = () => ; } if (!attachments.length && !readOnly) { options.headerRight = () => ( ); } options.headerBackground = () => ; navigation.setOptions(options); } // fetch server info getServerInfo = async() => { const { server } = this.props; const serversDB = database.servers; const serversCollection = serversDB.collections.get('servers'); try { this.serverInfo = await serversCollection.find(server); } catch (error) { // Do nothing } } getReadOnly = async() => { const { room } = this.state; const { user } = this.props; const readOnly = await isReadOnly(room, user); return readOnly; } getAttachments = async() => { const { mediaAllowList, maxFileSize } = this.state; const items = await Promise.all(this.files.map(async(item) => { // Check server settings const { success: canUpload, error } = canUploadFile(item, mediaAllowList, maxFileSize); item.canUpload = canUpload; item.error = error; // get video thumbnails if (isAndroid && this.files.length > 1 && item.mime?.match?.(/video/)) { try { const VideoThumbnails = require('expo-video-thumbnails'); const { uri } = await VideoThumbnails.getThumbnailAsync(item.path); item.uri = uri; } catch { // Do nothing } } // Set a filename, if there isn't any if (!item.filename) { item.filename = new Date().toISOString(); } return item; })); return { attachments: items, selected: items[0] }; } send = async() => { const { loading, selected } = this.state; if (loading) { return; } // update state await this.selectFile(selected); const { attachments, room, text, thread } = this.state; const { navigation, server, user } = this.props; // if it's share extension this should show loading if (this.isShareExtension) { this.setState({ loading: true }); // if it's not share extension this can close } else { navigation.pop(); } try { // Send attachment if (attachments.length) { await Promise.all(attachments.map(({ filename: name, mime: type, description, size, path, canUpload }) => { if (canUpload) { return RocketChat.sendFileMessage( room.rid, { name, description, size, type, path, store: 'Uploads' }, thread?.id, server, { id: user.id, token: user.token } ); } return Promise.resolve(); })); // Send text message } else if (text.length) { await RocketChat.sendMessage(room.rid, text, thread?.id, { id: user.id, token: user.token }); } } catch { // Do nothing } // if it's share extension this should close if (this.isShareExtension) { ShareExtension.close(); } }; selectFile = (item) => { const { attachments, selected } = this.state; if (attachments.length > 0) { const { text } = this.messagebox.current; const newAttachments = attachments.map((att) => { if (att.path === selected.path) { att.description = text; } return att; }); return this.setState({ attachments: newAttachments, selected: item }); } } removeFile = (item) => { const { selected, attachments } = this.state; let newSelected; if (item.path === selected.path) { const selectedIndex = attachments.findIndex(att => att.path === selected.path); // Selects the next one, if available if (attachments[selectedIndex + 1]?.path) { newSelected = attachments[selectedIndex + 1]; // If it's the last thumb, selects the previous one } else { newSelected = attachments[selectedIndex - 1] || {}; } } this.setState({ attachments: attachments.filter(att => att.path !== item.path), selected: newSelected ?? selected }, () => { this.messagebox?.current?.forceUpdate?.(); }); } onChangeText = (text) => { this.setState({ text }); } renderContent = () => { const { attachments, selected, room, text } = this.state; const { theme, navigation } = this.props; if (attachments.length) { return ( ); } return ( ); }; render() { console.count(`${ this.constructor.name }.render calls`); const { readOnly, room, loading } = this.state; const { theme } = this.props; if (readOnly || isBlocked(room)) { return ( {isBlocked(room) ? I18n.t('This_room_is_blocked') : I18n.t('This_room_is_read_only')} ); } return ( {this.renderContent()} ); } } ShareView.propTypes = { navigation: PropTypes.object, route: PropTypes.object, theme: PropTypes.string, user: PropTypes.shape({ id: PropTypes.string.isRequired, username: PropTypes.string.isRequired, token: PropTypes.string.isRequired }), server: PropTypes.string, FileUpload_MediaTypeWhiteList: PropTypes.string, FileUpload_MaxFileSize: PropTypes.string }; const mapStateToProps = state => ({ user: getUserSelector(state), server: state.share.server.server || state.server.server, FileUpload_MediaTypeWhiteList: state.settings.FileUpload_MediaTypeWhiteList, FileUpload_MaxFileSize: state.settings.FileUpload_MaxFileSize }); export default connect(mapStateToProps)(withTheme(ShareView));