import React from 'react'; import { PermissionsAndroid, StyleSheet, View } from 'react-native'; import { connect } from 'react-redux'; import { StackNavigationProp } from '@react-navigation/stack'; import { RouteProp } from '@react-navigation/native'; import CameraRoll from '@react-native-community/cameraroll'; import * as mime from 'react-native-mime-types'; import RNFetchBlob from 'rn-fetch-blob'; import { Video } from 'expo-av'; import { sha256 } from 'js-sha256'; import { withSafeAreaInsets } from 'react-native-safe-area-context'; import { LISTENER } from '../containers/Toast'; import EventEmitter from '../utils/events'; import I18n from '../i18n'; import { withTheme } from '../theme'; import { ImageViewer } from '../presentation/ImageViewer'; import { themes } from '../constants/colors'; import { formatAttachmentUrl } from '../lib/utils'; import RCActivityIndicator from '../containers/ActivityIndicator'; import * as HeaderButton from '../containers/HeaderButton'; import { isAndroid } from '../utils/deviceInfo'; import { getUserSelector } from '../selectors/login'; import { withDimensions } from '../dimensions'; import { getHeaderHeight } from '../containers/Header'; import StatusBar from '../containers/StatusBar'; import { InsideStackParamList } from '../stacks/types'; import { IAttachment } from '../definitions/IAttachment'; const styles = StyleSheet.create({ container: { flex: 1 } }); interface IAttachmentViewState { attachment: IAttachment; loading: boolean; } interface IAttachmentViewProps { navigation: StackNavigationProp; route: RouteProp; theme: string; baseUrl: string; width: number; height: number; insets: { left: number; bottom: number; right: number; top: number }; user: { id: string; token: string; }; Allow_Save_Media_to_Gallery: boolean; } class AttachmentView extends React.Component { private unsubscribeBlur: (() => void) | undefined; private videoRef: any; constructor(props: IAttachmentViewProps) { super(props); const attachment = props.route.params?.attachment; this.state = { attachment, loading: true }; this.setHeader(); } componentDidMount() { const { navigation } = this.props; this.unsubscribeBlur = navigation.addListener('blur', () => { if (this.videoRef && this.videoRef.stopAsync) { this.videoRef.stopAsync(); } }); } componentWillUnmount() { if (this.unsubscribeBlur) { this.unsubscribeBlur(); } } setHeader = () => { const { route, navigation, theme, Allow_Save_Media_to_Gallery } = this.props; const attachment = route.params?.attachment; let { title } = attachment; try { if (title) { title = decodeURI(title); } } catch { // Do nothing } const options = { title, headerLeft: () => , headerRight: () => Allow_Save_Media_to_Gallery ? : null, headerBackground: () => , headerTintColor: themes[theme].previewTintColor, headerTitleStyle: { color: themes[theme].previewTintColor, marginHorizontal: 10 } }; navigation.setOptions(options); }; getVideoRef = (ref: Video) => (this.videoRef = ref); handleSave = async () => { const { attachment } = this.state; const { user, baseUrl } = this.props; const { title_link, image_url, image_type, video_url, video_type } = attachment; const url = title_link || image_url || video_url; if (!url) { return; } const mediaAttachment = formatAttachmentUrl(url, user.id, user.token, baseUrl); if (isAndroid) { const rationale = { title: I18n.t('Write_External_Permission'), message: I18n.t('Write_External_Permission_Message'), buttonPositive: 'Ok' }; const result = await PermissionsAndroid.request(PermissionsAndroid.PERMISSIONS.WRITE_EXTERNAL_STORAGE, rationale); if (!(result || result === PermissionsAndroid.RESULTS.GRANTED)) { return; } } this.setState({ loading: true }); try { const extension = image_url ? `.${mime.extension(image_type) || 'jpg'}` : `.${(video_type === 'video/quicktime' && 'mov') || mime.extension(video_type) || 'mp4'}`; // The return of mime.extension('video/quicktime') is .qt, // this format the iOS isn't recognize and can't save on gallery const documentDir = `${RNFetchBlob.fs.dirs.DocumentDir}/`; const path = `${documentDir + sha256(url!) + extension}`; const file = await RNFetchBlob.config({ path }).fetch('GET', mediaAttachment); await CameraRoll.save(path, { album: 'Rocket.Chat' }); await file.flush(); EventEmitter.emit(LISTENER, { message: I18n.t('saved_to_gallery') }); } catch (e) { EventEmitter.emit(LISTENER, { message: I18n.t(image_url ? 'error-save-image' : 'error-save-video') }); } this.setState({ loading: false }); }; renderImage = (uri: string) => { const { theme, width, height, insets } = this.props; const headerHeight = getHeaderHeight(width > height); return ( this.setState({ loading: false })} theme={theme} width={width} height={height - insets.top - insets.bottom - headerHeight} /> ); }; renderVideo = (uri: string) => (