forked from Ivasoft/mattermost-mobile
211 lines
6.2 KiB
TypeScript
211 lines
6.2 KiB
TypeScript
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
import React, {useCallback, useRef} from 'react';
|
|
import {View, TouchableWithoutFeedback} from 'react-native';
|
|
import Animated from 'react-native-reanimated';
|
|
|
|
import TouchableWithFeedback from '@components/touchable_with_feedback';
|
|
import {useTheme} from '@context/theme';
|
|
import {useGalleryItem} from '@hooks/gallery';
|
|
import {isDocument, isImage, isVideo} from '@utils/file';
|
|
import {changeOpacity, makeStyleSheetFromTheme} from '@utils/theme';
|
|
|
|
import DocumentFile, {DocumentFileRef} from './document_file';
|
|
import FileIcon from './file_icon';
|
|
import FileInfo from './file_info';
|
|
import FileOptionsIcon from './file_options_icon';
|
|
import ImageFile from './image_file';
|
|
import ImageFileOverlay from './image_file_overlay';
|
|
import VideoFile from './video_file';
|
|
|
|
type FileProps = {
|
|
canDownloadFiles: boolean;
|
|
file: FileInfo;
|
|
galleryIdentifier: string;
|
|
index: number;
|
|
inViewPort: boolean;
|
|
isSingleImage?: boolean;
|
|
nonVisibleImagesCount: number;
|
|
onPress: (index: number) => void;
|
|
publicLinkEnabled: boolean;
|
|
channelName?: string;
|
|
onOptionsPress?: (fileInfo: FileInfo) => void;
|
|
optionSelected?: boolean;
|
|
wrapperWidth?: number;
|
|
showDate?: boolean;
|
|
updateFileForGallery: (idx: number, file: FileInfo) => void;
|
|
asCard?: boolean;
|
|
};
|
|
|
|
const getStyleSheet = makeStyleSheetFromTheme((theme: Theme) => {
|
|
return {
|
|
fileWrapper: {
|
|
flex: 1,
|
|
flexDirection: 'row',
|
|
alignItems: 'center',
|
|
marginTop: 10,
|
|
borderWidth: 1,
|
|
borderColor: changeOpacity(theme.centerChannelColor, 0.24),
|
|
borderRadius: 4,
|
|
},
|
|
iconWrapper: {
|
|
marginTop: 8,
|
|
marginRight: 7,
|
|
marginBottom: 8,
|
|
marginLeft: 6,
|
|
},
|
|
imageVideo: {
|
|
height: 40,
|
|
width: 40,
|
|
margin: 4,
|
|
},
|
|
};
|
|
});
|
|
|
|
const File = ({
|
|
asCard = false,
|
|
canDownloadFiles,
|
|
channelName,
|
|
file,
|
|
galleryIdentifier,
|
|
inViewPort = false,
|
|
index,
|
|
isSingleImage = false,
|
|
nonVisibleImagesCount = 0,
|
|
onOptionsPress,
|
|
onPress,
|
|
optionSelected,
|
|
publicLinkEnabled,
|
|
showDate = false,
|
|
updateFileForGallery,
|
|
wrapperWidth = 300,
|
|
}: FileProps) => {
|
|
const document = useRef<DocumentFileRef>(null);
|
|
const theme = useTheme();
|
|
const style = getStyleSheet(theme);
|
|
|
|
const handlePreviewPress = useCallback(() => {
|
|
if (document.current) {
|
|
document.current.handlePreviewPress();
|
|
} else {
|
|
onPress(index);
|
|
}
|
|
}, [index]);
|
|
|
|
const {styles, onGestureEvent, ref} = useGalleryItem(galleryIdentifier, index, handlePreviewPress);
|
|
|
|
const handleOnOptionsPress = useCallback(() => {
|
|
onOptionsPress?.(file);
|
|
}, [file, onOptionsPress]);
|
|
|
|
const optionsButton = (
|
|
<FileOptionsIcon
|
|
onPress={handleOnOptionsPress}
|
|
selected={optionSelected}
|
|
/>
|
|
);
|
|
|
|
const fileInfo = (
|
|
<FileInfo
|
|
file={file}
|
|
showDate={showDate}
|
|
channelName={channelName}
|
|
onPress={handlePreviewPress}
|
|
/>
|
|
);
|
|
|
|
const renderImageFileOverlay = (
|
|
<ImageFileOverlay
|
|
value={nonVisibleImagesCount}
|
|
/>
|
|
);
|
|
|
|
const renderImageFile = (
|
|
<TouchableWithoutFeedback onPress={onGestureEvent}>
|
|
<Animated.View style={[styles, asCard ? style.imageVideo : null]}>
|
|
<ImageFile
|
|
file={file}
|
|
forwardRef={ref}
|
|
inViewPort={inViewPort}
|
|
isSingleImage={isSingleImage}
|
|
resizeMode={'cover'}
|
|
wrapperWidth={wrapperWidth}
|
|
/>
|
|
{Boolean(nonVisibleImagesCount) && renderImageFileOverlay}
|
|
</Animated.View>
|
|
</TouchableWithoutFeedback>
|
|
);
|
|
|
|
const renderVideoFile = (
|
|
<TouchableWithoutFeedback onPress={onGestureEvent}>
|
|
<Animated.View style={[styles, asCard ? style.imageVideo : null]}>
|
|
<VideoFile
|
|
file={file}
|
|
forwardRef={ref}
|
|
inViewPort={inViewPort}
|
|
isSingleImage={isSingleImage}
|
|
resizeMode={'cover'}
|
|
wrapperWidth={wrapperWidth}
|
|
updateFileForGallery={updateFileForGallery}
|
|
index={index}
|
|
/>
|
|
{Boolean(nonVisibleImagesCount) && renderImageFileOverlay}
|
|
</Animated.View>
|
|
</TouchableWithoutFeedback>
|
|
);
|
|
|
|
const renderDocumentFile = (
|
|
<View style={style.iconWrapper}>
|
|
<DocumentFile
|
|
ref={document}
|
|
canDownloadFiles={canDownloadFiles}
|
|
file={file}
|
|
/>
|
|
</View>
|
|
);
|
|
|
|
const renderCardWithImage = (fileIcon: JSX.Element) => {
|
|
return (
|
|
<View style={[style.fileWrapper]}>
|
|
<View style={style.iconWrapper}>
|
|
{fileIcon}
|
|
</View>
|
|
{fileInfo}
|
|
{onOptionsPress && optionsButton}
|
|
</View>
|
|
);
|
|
};
|
|
|
|
let fileComponent;
|
|
if (isVideo(file) && publicLinkEnabled) {
|
|
fileComponent = asCard ? renderCardWithImage(renderVideoFile) : renderVideoFile;
|
|
} else if (isImage(file)) {
|
|
fileComponent = asCard ? renderCardWithImage(renderImageFile) : renderImageFile;
|
|
} else if (isDocument(file)) {
|
|
fileComponent = (
|
|
<View style={[style.fileWrapper]}>
|
|
{renderDocumentFile}
|
|
{fileInfo}
|
|
{onOptionsPress && optionsButton}
|
|
</View>
|
|
);
|
|
} else {
|
|
const touchableWithPreview = (
|
|
<TouchableWithFeedback
|
|
onPress={handlePreviewPress}
|
|
type={'opacity'}
|
|
>
|
|
<FileIcon
|
|
file={file}
|
|
/>
|
|
</TouchableWithFeedback>
|
|
);
|
|
|
|
fileComponent = renderCardWithImage(touchableWithPreview);
|
|
}
|
|
return fileComponent;
|
|
};
|
|
|
|
export default File;
|