forked from Ivasoft/mattermost-mobile
* UX redesign * fix call bar profile on Android * update @mattermost/calls; use common color utils * collapseIcon and time colour fixes (not correct yet on Onyx or Quartz) * fix time and collapseIcon colors; fix unavailable wrapper * center users, scroll when full; statusBar bg and styling * better spacing; better screenshare; no reaction bar when no reactions * remove margins from bottom; speaker/react buttonOn colour is now callsBg * update calls-common; update raised hand button colors * i18n * cleaning up style sheets * fix package-lock * fix vertical alignment of phone icon * add a noBorder prop to UserAvatarsStack * typography, icon sizing, spacing changes * add rounded header; UI improvements; refactor observables * updating phone icon, join call margins * join call text; color theming * remove unneeded container * phone-outline -> phone * split CallsChannelState into boolean components * use sidebar bg to generate calls bg * fix hand icon, button texts
187 lines
6.0 KiB
TypeScript
187 lines
6.0 KiB
TypeScript
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
import React, {useCallback} from 'react';
|
|
import {useIntl} from 'react-intl';
|
|
import {type StyleProp, Text, TouchableOpacity, View, type ViewStyle} from 'react-native';
|
|
import {useSafeAreaInsets} from 'react-native-safe-area-context';
|
|
|
|
import FormattedText from '@components/formatted_text';
|
|
import {useTheme} from '@context/theme';
|
|
import {useIsTablet} from '@hooks/device';
|
|
import {TITLE_HEIGHT} from '@screens/bottom_sheet/content';
|
|
import {bottomSheet} from '@screens/navigation';
|
|
import {bottomSheetSnapPoint} from '@utils/helpers';
|
|
import {preventDoubleTap} from '@utils/tap';
|
|
import {changeOpacity, makeStyleSheetFromTheme} from '@utils/theme';
|
|
import {typography} from '@utils/typography';
|
|
|
|
import UserAvatar from './user_avatar';
|
|
import UsersList from './users_list';
|
|
|
|
import type {BottomSheetProps} from '@gorhom/bottom-sheet';
|
|
import type UserModel from '@typings/database/models/servers/user';
|
|
|
|
const OVERFLOW_DISPLAY_LIMIT = 99;
|
|
const USER_ROW_HEIGHT = 40;
|
|
|
|
type Props = {
|
|
channelId: string;
|
|
location: string;
|
|
users: UserModel[];
|
|
breakAt?: number;
|
|
style?: StyleProp<ViewStyle>;
|
|
noBorder?: boolean;
|
|
}
|
|
|
|
const getStyleSheet = makeStyleSheetFromTheme((theme: Theme) => {
|
|
const size = 24;
|
|
const imgOverlap = -6;
|
|
|
|
return {
|
|
container: {
|
|
flexDirection: 'row',
|
|
},
|
|
firstAvatar: {
|
|
justifyContent: 'center',
|
|
alignItems: 'center',
|
|
width: size,
|
|
height: size,
|
|
borderWidth: (size / 2) + 1,
|
|
borderColor: theme.centerChannelBg,
|
|
backgroundColor: theme.centerChannelBg,
|
|
borderRadius: size / 2,
|
|
},
|
|
noBorder: {
|
|
borderWidth: 0,
|
|
},
|
|
notFirstAvatars: {
|
|
justifyContent: 'center',
|
|
alignItems: 'center',
|
|
width: size,
|
|
height: size,
|
|
borderWidth: (size / 2) + 1,
|
|
borderColor: theme.centerChannelBg,
|
|
backgroundColor: theme.centerChannelBg,
|
|
borderRadius: size / 2,
|
|
marginLeft: imgOverlap,
|
|
},
|
|
overflowContainer: {
|
|
justifyContent: 'center',
|
|
alignItems: 'center',
|
|
width: size,
|
|
height: size,
|
|
borderRadius: size / 2,
|
|
borderWidth: 1,
|
|
borderColor: theme.centerChannelBg,
|
|
backgroundColor: theme.centerChannelBg,
|
|
marginLeft: imgOverlap,
|
|
},
|
|
overflowItem: {
|
|
justifyContent: 'center',
|
|
alignItems: 'center',
|
|
width: size,
|
|
height: size,
|
|
borderRadius: size / 2,
|
|
borderWidth: 1,
|
|
borderColor: theme.centerChannelBg,
|
|
backgroundColor: changeOpacity(theme.centerChannelColor, 0.08),
|
|
},
|
|
overflowText: {
|
|
fontSize: 10,
|
|
fontWeight: 'bold',
|
|
color: changeOpacity(theme.centerChannelColor, 0.64),
|
|
textAlign: 'center',
|
|
},
|
|
listHeader: {
|
|
marginBottom: 12,
|
|
},
|
|
listHeaderText: {
|
|
color: theme.centerChannelColor,
|
|
...typography('Heading', 600, 'SemiBold'),
|
|
},
|
|
};
|
|
});
|
|
|
|
const UserAvatarsStack = ({
|
|
breakAt = 3,
|
|
channelId,
|
|
location,
|
|
style: baseContainerStyle,
|
|
users,
|
|
noBorder = false,
|
|
}: Props) => {
|
|
const theme = useTheme();
|
|
const intl = useIntl();
|
|
const isTablet = useIsTablet();
|
|
const {bottom} = useSafeAreaInsets();
|
|
|
|
const showParticipantsList = useCallback(preventDoubleTap(() => {
|
|
const renderContent = () => (
|
|
<>
|
|
{!isTablet && (
|
|
<View style={style.listHeader}>
|
|
<FormattedText
|
|
id='mobile.participants.header'
|
|
defaultMessage={'Thread Participants'}
|
|
style={style.listHeaderText}
|
|
/>
|
|
</View>
|
|
)}
|
|
<UsersList
|
|
channelId={channelId}
|
|
location={location}
|
|
users={users}
|
|
/>
|
|
</>
|
|
);
|
|
|
|
const snapPoints: BottomSheetProps['snapPoints'] = [1, bottomSheetSnapPoint(Math.min(users.length, 5), USER_ROW_HEIGHT, bottom) + TITLE_HEIGHT];
|
|
if (users.length > 5) {
|
|
snapPoints.push('80%');
|
|
}
|
|
|
|
bottomSheet({
|
|
closeButtonId: 'close-set-user-status',
|
|
renderContent,
|
|
initialSnapIndex: 1,
|
|
snapPoints,
|
|
title: intl.formatMessage({id: 'mobile.participants.header', defaultMessage: 'Thread Participants'}),
|
|
theme,
|
|
});
|
|
}), [isTablet, theme, users, channelId, location, bottom]);
|
|
|
|
const displayUsers = users.slice(0, breakAt);
|
|
const overflowUsersCount = Math.min(users.length - displayUsers.length, OVERFLOW_DISPLAY_LIMIT);
|
|
|
|
const style = getStyleSheet(theme);
|
|
|
|
return (
|
|
<TouchableOpacity
|
|
onPress={showParticipantsList}
|
|
style={baseContainerStyle}
|
|
>
|
|
<View style={style.container}>
|
|
{displayUsers.map((user, index) => (
|
|
<UserAvatar
|
|
key={user.id}
|
|
style={index === 0 ? [style.firstAvatar, noBorder && style.noBorder] : [style.notFirstAvatars, noBorder && style.noBorder]}
|
|
user={user}
|
|
/>
|
|
))}
|
|
{Boolean(overflowUsersCount) && (
|
|
<View style={[style.overflowContainer, noBorder && style.noBorder]}>
|
|
<View style={[style.overflowItem, noBorder && style.noBorder]}>
|
|
<Text style={style.overflowText}>
|
|
{'+' + overflowUsersCount.toString()}
|
|
</Text>
|
|
</View>
|
|
</View>
|
|
)}
|
|
</View>
|
|
</TouchableOpacity>
|
|
);
|
|
};
|
|
|
|
export default UserAvatarsStack;
|