Files
mattermost-mobile/app/components/user_list/index.tsx

276 lines
8.7 KiB
TypeScript

// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
import React, {useCallback, useMemo} from 'react';
import {useIntl} from 'react-intl';
import {FlatList, Keyboard, ListRenderItemInfo, Platform, SectionList, SectionListData, Text, View} from 'react-native';
import {storeProfile} from '@actions/local/user';
import Loading from '@components/loading';
import NoResultsWithTerm from '@components/no_results_with_term';
import UserListRow from '@components/user_list_row';
import {General, Screens} from '@constants';
import {useServerUrl} from '@context/server';
import {useTheme} from '@context/theme';
import {useKeyboardHeight} from '@hooks/device';
import {openAsBottomSheet} from '@screens/navigation';
import {
changeOpacity,
makeStyleSheetFromTheme,
} from '@utils/theme';
import {typography} from '@utils/typography';
const INITIAL_BATCH_TO_RENDER = 15;
const SCROLL_EVENT_THROTTLE = 60;
const keyboardDismissProp = Platform.select({
android: {
onScrollBeginDrag: Keyboard.dismiss,
},
ios: {
keyboardDismissMode: 'on-drag' as const,
},
});
const keyExtractor = (item: UserProfile) => {
return item.id;
};
const sectionKeyExtractor = (profile: UserProfile) => {
// Group items alphabetically by first letter of username
return profile.username[0].toUpperCase();
};
export function createProfilesSections(profiles: UserProfile[]) {
const sections: {[key: string]: UserProfile[]} = {};
const sectionKeys: string[] = [];
for (const profile of profiles) {
const sectionKey = sectionKeyExtractor(profile);
if (!sections[sectionKey]) {
sections[sectionKey] = [];
sectionKeys.push(sectionKey);
}
sections[sectionKey].push(profile);
}
sectionKeys.sort();
return sectionKeys.map((sectionKey, index) => {
return {
id: sectionKey,
first: index === 0,
data: sections[sectionKey],
};
});
}
const getStyleFromTheme = makeStyleSheetFromTheme((theme) => {
return {
list: {
backgroundColor: theme.centerChannelBg,
flex: 1,
},
container: {
flexGrow: 1,
},
loadingContainer: {
flex: 1,
justifyContent: 'center' as const,
alignItems: 'center' as const,
},
noResultContainer: {
flexGrow: 1,
alignItems: 'center' as const,
justifyContent: 'center' as const,
},
sectionContainer: {
backgroundColor: changeOpacity(theme.centerChannelColor, 0.08),
paddingLeft: 16,
justifyContent: 'center',
height: 24,
},
sectionWrapper: {
backgroundColor: theme.centerChannelBg,
},
sectionText: {
color: theme.centerChannelColor,
...typography('Body', 75, 'SemiBold'),
},
};
});
type Props = {
profiles: UserProfile[];
currentUserId: string;
teammateNameDisplay: string;
handleSelectProfile: (user: UserProfile) => void;
fetchMore: () => void;
loading: boolean;
showNoResults: boolean;
selectedIds: {[id: string]: UserProfile};
testID?: string;
term?: string;
tutorialWatched: boolean;
}
export default function UserList({
profiles,
selectedIds,
currentUserId,
teammateNameDisplay,
handleSelectProfile,
fetchMore,
loading,
showNoResults,
term,
testID,
tutorialWatched,
}: Props) {
const intl = useIntl();
const theme = useTheme();
const serverUrl = useServerUrl();
const style = getStyleFromTheme(theme);
const keyboardHeight = useKeyboardHeight();
const noResutsStyle = useMemo(() => [
style.noResultContainer,
{paddingBottom: keyboardHeight},
], [style, keyboardHeight]);
const data = useMemo(() => {
if (term) {
return profiles;
}
return createProfilesSections(profiles);
}, [term, profiles]);
const openUserProfile = useCallback(async (profile: UserProfile) => {
const {user} = await storeProfile(serverUrl, profile);
if (user) {
const screen = Screens.USER_PROFILE;
const title = intl.formatMessage({id: 'mobile.routes.user_profile', defaultMessage: 'Profile'});
const closeButtonId = 'close-user-profile';
const props = {
closeButtonId,
userId: user.id,
location: Screens.USER_PROFILE,
};
Keyboard.dismiss();
openAsBottomSheet({screen, title, theme, closeButtonId, props});
}
}, []);
const renderItem = useCallback(({item, index, section}: ListRenderItemInfo<UserProfile> & {section?: SectionListData<UserProfile>}) => {
// The list will re-render when the selection changes because it's passed into the list as extraData
const selected = Boolean(selectedIds[item.id]);
const canAdd = Object.keys(selectedIds).length < General.MAX_USERS_IN_GM;
return (
<UserListRow
key={item.id}
highlight={section?.first && index === 0}
id={item.id}
isMyUser={currentUserId === item.id}
onPress={handleSelectProfile}
onLongPress={openUserProfile}
disabled={!canAdd}
selectable={true}
selected={selected}
testID='create_direct_message.user_list.user_item'
teammateNameDisplay={teammateNameDisplay}
tutorialWatched={tutorialWatched}
user={item}
/>
);
}, [selectedIds, currentUserId, handleSelectProfile, teammateNameDisplay, tutorialWatched]);
const renderLoading = useCallback(() => {
if (!loading) {
return null;
}
return (
<Loading
color={theme.buttonBg}
containerStyle={style.loadingContainer}
size='large'
/>
);
}, [loading, theme]);
const renderNoResults = useCallback(() => {
if (!showNoResults || !term) {
return null;
}
return (
<View style={noResutsStyle}>
<NoResultsWithTerm term={term}/>
</View>
);
}, [showNoResults && style, term, noResutsStyle]);
const renderSectionHeader = useCallback(({section}: {section: SectionListData<UserProfile>}) => {
return (
<View style={style.sectionWrapper}>
<View style={style.sectionContainer}>
<Text style={style.sectionText}>{section.id}</Text>
</View>
</View>
);
}, [style]);
const renderFlatList = (items: UserProfile[]) => {
return (
<FlatList
contentContainerStyle={style.container}
data={items}
keyboardShouldPersistTaps='always'
{...keyboardDismissProp}
keyExtractor={keyExtractor}
initialNumToRender={INITIAL_BATCH_TO_RENDER}
ListEmptyComponent={renderNoResults}
ListFooterComponent={renderLoading}
maxToRenderPerBatch={INITIAL_BATCH_TO_RENDER + 1}
removeClippedSubviews={true}
renderItem={renderItem}
scrollEventThrottle={SCROLL_EVENT_THROTTLE}
style={style.list}
testID={`${testID}.flat_list`}
/>
);
};
const renderSectionList = (sections: Array<SectionListData<UserProfile>>) => {
return (
<SectionList
contentContainerStyle={style.container}
keyboardShouldPersistTaps='always'
{...keyboardDismissProp}
keyExtractor={keyExtractor}
initialNumToRender={INITIAL_BATCH_TO_RENDER}
ListEmptyComponent={renderNoResults}
ListFooterComponent={renderLoading}
maxToRenderPerBatch={INITIAL_BATCH_TO_RENDER + 1}
removeClippedSubviews={true}
renderItem={renderItem}
renderSectionHeader={renderSectionHeader}
scrollEventThrottle={SCROLL_EVENT_THROTTLE}
sections={sections}
style={style.list}
stickySectionHeadersEnabled={false}
testID={`${testID}.section_list`}
onEndReached={fetchMore}
/>
);
};
if (term) {
return renderFlatList(data as UserProfile[]);
}
return renderSectionList(data as Array<SectionListData<UserProfile>>);
}