forked from Ivasoft/mattermost-mobile
* Ensure no unresolved types in the definition files * Address feedback and general cleanup * Move import from @constants/x to @constants where relevant * Remove unneeded "import as"
143 lines
4.2 KiB
TypeScript
143 lines
4.2 KiB
TypeScript
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
import React, {memo, useCallback, useRef, useState} from 'react';
|
|
import {StyleSheet, TouchableWithoutFeedback, View} from 'react-native';
|
|
import Animated from 'react-native-reanimated';
|
|
import parseUrl from 'url-parse';
|
|
|
|
import CompassIcon from '@components/compass_icon';
|
|
import ProgressiveImage from '@components/progressive_image';
|
|
import {useServerUrl} from '@context/server';
|
|
import {useGalleryItem} from '@hooks/gallery';
|
|
import {fileToGalleryItem, openGalleryAtIndex} from '@utils/gallery';
|
|
import {generateId} from '@utils/general';
|
|
import {calculateDimensions, isGifTooLarge} from '@utils/images';
|
|
|
|
import type {GalleryItemType} from '@typings/screens/gallery';
|
|
|
|
type MarkdownTableImageProps = {
|
|
disabled?: boolean;
|
|
imagesMetadata: Record<string, PostImage>;
|
|
location?: string;
|
|
postId: string;
|
|
serverURL?: string;
|
|
source: string;
|
|
}
|
|
|
|
const style = StyleSheet.create({
|
|
container: {
|
|
alignItems: 'center',
|
|
flex: 1,
|
|
},
|
|
});
|
|
|
|
const MarkTableImage = ({disabled, imagesMetadata, location, postId, serverURL, source}: MarkdownTableImageProps) => {
|
|
const metadata = imagesMetadata[source];
|
|
const fileId = useRef(generateId('uid')).current;
|
|
const [failed, setFailed] = useState(isGifTooLarge(metadata));
|
|
const currentServerUrl = useServerUrl();
|
|
const galleryIdentifier = `${postId}-${fileId}-${location}`;
|
|
|
|
const getImageSource = () => {
|
|
let uri = source;
|
|
let server = serverURL;
|
|
|
|
if (!serverURL) {
|
|
server = currentServerUrl;
|
|
}
|
|
|
|
if (uri.startsWith('/')) {
|
|
uri = server + uri;
|
|
}
|
|
|
|
return uri;
|
|
};
|
|
|
|
const getFileInfo = () => {
|
|
const {height, width} = metadata;
|
|
const link = decodeURIComponent(getImageSource());
|
|
let filename = parseUrl(link.substr(link.lastIndexOf('/'))).pathname.replace('/', '');
|
|
let extension = filename.split('.').pop();
|
|
|
|
if (extension === filename) {
|
|
const ext = filename.indexOf('.') === -1 ? '.png' : filename.substring(filename.lastIndexOf('.'));
|
|
filename = `${filename}${ext}`;
|
|
extension = ext;
|
|
}
|
|
|
|
return {
|
|
id: fileId,
|
|
name: filename,
|
|
extension,
|
|
has_preview_image: true,
|
|
post_id: postId,
|
|
uri: link,
|
|
width,
|
|
height,
|
|
};
|
|
};
|
|
|
|
const handlePreviewImage = useCallback(() => {
|
|
const file = getFileInfo() as FileInfo;
|
|
if (!file?.uri) {
|
|
return;
|
|
}
|
|
const item: GalleryItemType = {
|
|
...fileToGalleryItem(file),
|
|
type: 'image',
|
|
};
|
|
openGalleryAtIndex(galleryIdentifier, 0, [item]);
|
|
}, [metadata, source, serverURL, currentServerUrl, postId]);
|
|
|
|
const {ref, onGestureEvent, styles} = useGalleryItem(
|
|
galleryIdentifier,
|
|
0,
|
|
handlePreviewImage,
|
|
);
|
|
|
|
const onLoadFailed = useCallback(() => {
|
|
setFailed(true);
|
|
}, []);
|
|
|
|
let image;
|
|
if (failed) {
|
|
image = (
|
|
<CompassIcon
|
|
name='file-image-broken-outline-large'
|
|
size={24}
|
|
/>
|
|
);
|
|
} else {
|
|
const {height, width} = calculateDimensions(metadata.height, metadata.width, 100, 100);
|
|
image = (
|
|
<TouchableWithoutFeedback
|
|
disabled={disabled}
|
|
onPress={onGestureEvent}
|
|
>
|
|
<Animated.View style={[styles, {width, height}]}>
|
|
<ProgressiveImage
|
|
id={fileId}
|
|
defaultSource={{uri: source}}
|
|
forwardRef={ref}
|
|
onError={onLoadFailed}
|
|
resizeMode='contain'
|
|
style={{width, height}}
|
|
/>
|
|
</Animated.View>
|
|
</TouchableWithoutFeedback>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<View
|
|
style={style.container}
|
|
testID='markdown_table_image'
|
|
>
|
|
{image}
|
|
</View>
|
|
);
|
|
};
|
|
|
|
export default memo(MarkTableImage);
|