forked from Ivasoft/mattermost-mobile
* Move managers to their own folder * Fix lint * Fix unit tests Co-authored-by: Elias Nahum <nahumhbl@gmail.com>
179 lines
5.2 KiB
TypeScript
179 lines
5.2 KiB
TypeScript
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
import React, {useCallback, useEffect, useMemo, useRef} from 'react';
|
|
import {
|
|
ScrollView,
|
|
Text,
|
|
View,
|
|
Platform,
|
|
} from 'react-native';
|
|
import Animated, {useAnimatedStyle, useSharedValue, withTiming} from 'react-native-reanimated';
|
|
|
|
import {GalleryInit} from '@context/gallery';
|
|
import {useTheme} from '@context/theme';
|
|
import DraftUploadManager from '@managers/draft_upload_manager';
|
|
import {fileToGalleryItem, openGalleryAtIndex} from '@utils/gallery';
|
|
import {makeStyleSheetFromTheme} from '@utils/theme';
|
|
|
|
import UploadItem from './upload_item';
|
|
|
|
const CONTAINER_HEIGHT_MAX = 67;
|
|
const CONTAINER_HEIGHT_MIN = 0;
|
|
const ERROR_HEIGHT_MAX = 20;
|
|
const ERROR_HEIGHT_MIN = 0;
|
|
|
|
type Props = {
|
|
currentUserId: string;
|
|
files: FileInfo[];
|
|
uploadFileError: React.ReactNode;
|
|
channelId: string;
|
|
rootId: string;
|
|
}
|
|
|
|
const getStyleSheet = makeStyleSheetFromTheme((theme) => {
|
|
return {
|
|
previewContainer: {
|
|
display: 'flex',
|
|
flexDirection: 'column',
|
|
},
|
|
fileContainer: {
|
|
display: 'flex',
|
|
flexDirection: 'row',
|
|
height: 0,
|
|
},
|
|
errorContainer: {
|
|
height: 0,
|
|
},
|
|
errorTextContainer: {
|
|
marginTop: Platform.select({
|
|
ios: 4,
|
|
android: 2,
|
|
}),
|
|
marginHorizontal: 12,
|
|
flex: 1,
|
|
},
|
|
scrollView: {
|
|
flex: 1,
|
|
},
|
|
scrollViewContent: {
|
|
alignItems: 'flex-end',
|
|
paddingRight: 12,
|
|
},
|
|
warning: {
|
|
color: theme.errorTextColor,
|
|
flex: 1,
|
|
flexWrap: 'wrap',
|
|
},
|
|
};
|
|
});
|
|
|
|
function Uploads({
|
|
currentUserId,
|
|
files,
|
|
uploadFileError,
|
|
channelId,
|
|
rootId,
|
|
}: Props) {
|
|
const galleryIdentifier = `${channelId}-uploadedItems-${rootId}`;
|
|
const theme = useTheme();
|
|
const style = getStyleSheet(theme);
|
|
|
|
const errorHeight = useSharedValue(ERROR_HEIGHT_MIN);
|
|
const containerHeight = useSharedValue(files.length ? CONTAINER_HEIGHT_MAX : CONTAINER_HEIGHT_MIN);
|
|
const filesForGallery = useRef(files.filter((f) => !f.failed && !DraftUploadManager.isUploading(f.clientId!)));
|
|
|
|
const errorAnimatedStyle = useAnimatedStyle(() => {
|
|
return {
|
|
height: withTiming(errorHeight.value),
|
|
};
|
|
});
|
|
|
|
const containerAnimatedStyle = useAnimatedStyle(() => {
|
|
return {
|
|
height: withTiming(containerHeight.value),
|
|
};
|
|
});
|
|
|
|
const fileContainerStyle = useMemo(() => ({
|
|
paddingBottom: files.length ? 5 : 0,
|
|
}), [files.length]);
|
|
|
|
useEffect(() => {
|
|
filesForGallery.current = files.filter((f) => !f.failed && !DraftUploadManager.isUploading(f.clientId!));
|
|
}, [files]);
|
|
|
|
useEffect(() => {
|
|
if (uploadFileError) {
|
|
errorHeight.value = ERROR_HEIGHT_MAX;
|
|
} else {
|
|
errorHeight.value = ERROR_HEIGHT_MIN;
|
|
}
|
|
}, [uploadFileError]);
|
|
|
|
useEffect(() => {
|
|
if (files.length) {
|
|
containerHeight.value = CONTAINER_HEIGHT_MAX;
|
|
return;
|
|
}
|
|
containerHeight.value = CONTAINER_HEIGHT_MIN;
|
|
}, [files.length > 0]);
|
|
|
|
const openGallery = useCallback((file: FileInfo) => {
|
|
const items = filesForGallery.current.map((f) => fileToGalleryItem(f, currentUserId));
|
|
const index = filesForGallery.current.findIndex((f) => f.clientId === file.clientId);
|
|
openGalleryAtIndex(galleryIdentifier, index, items, true);
|
|
}, [currentUserId, files]);
|
|
|
|
const buildFilePreviews = () => {
|
|
return files.map((file, index) => {
|
|
return (
|
|
<UploadItem
|
|
channelId={channelId}
|
|
galleryIdentifier={galleryIdentifier}
|
|
index={index}
|
|
file={file}
|
|
key={file.clientId}
|
|
openGallery={openGallery}
|
|
rootId={rootId}
|
|
/>
|
|
);
|
|
});
|
|
};
|
|
|
|
return (
|
|
<GalleryInit galleryIdentifier={galleryIdentifier}>
|
|
<View style={style.previewContainer}>
|
|
<Animated.View
|
|
style={[style.fileContainer, fileContainerStyle, containerAnimatedStyle]}
|
|
>
|
|
<ScrollView
|
|
horizontal={true}
|
|
style={style.scrollView}
|
|
contentContainerStyle={style.scrollViewContent}
|
|
keyboardShouldPersistTaps={'handled'}
|
|
>
|
|
{buildFilePreviews()}
|
|
</ScrollView>
|
|
</Animated.View>
|
|
|
|
<Animated.View
|
|
style={[style.errorContainer, errorAnimatedStyle]}
|
|
>
|
|
{Boolean(uploadFileError) &&
|
|
<View style={style.errorTextContainer}>
|
|
|
|
<Text style={style.warning}>
|
|
{uploadFileError}
|
|
</Text>
|
|
|
|
</View>
|
|
}
|
|
</Animated.View>
|
|
</View>
|
|
</GalleryInit>
|
|
);
|
|
}
|
|
|
|
export default React.memo(Uploads);
|