forked from Ivasoft/mattermost-mobile
276 lines
8.7 KiB
TypeScript
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>>);
|
|
}
|
|
|