diff --git a/native/chat/chat-thread-list.react.js b/native/chat/chat-thread-list.react.js
index 81bf4b425..d8609c218 100644
--- a/native/chat/chat-thread-list.react.js
+++ b/native/chat/chat-thread-list.react.js
@@ -1,642 +1,646 @@
// @flow
import IonIcon from '@expo/vector-icons/Ionicons.js';
import invariant from 'invariant';
import _sum from 'lodash/fp/sum.js';
import * as React from 'react';
import {
View,
FlatList,
Platform,
TextInput,
TouchableWithoutFeedback,
BackHandler,
} from 'react-native';
import { FloatingAction } from 'react-native-floating-action';
import Animated from 'react-native-reanimated';
import { createSelector } from 'reselect';
import { searchUsers } from 'lib/actions/user-actions.js';
+import { useLoggedInUserInfo } from 'lib/hooks/account-hooks.js';
import {
type ChatThreadItem,
useFlattenedChatListData,
} from 'lib/selectors/chat-selectors.js';
import { useGlobalThreadSearchIndex } from 'lib/selectors/nav-selectors.js';
import { usersWithPersonalThreadSelector } from 'lib/selectors/user-selectors.js';
import SearchIndex from 'lib/shared/search-index.js';
import {
createPendingThread,
getThreadListSearchResults,
} from 'lib/shared/thread-utils.js';
import type { UserSearchResult } from 'lib/types/search-types.js';
import type { ThreadInfo } from 'lib/types/thread-types.js';
import { threadTypes } from 'lib/types/thread-types.js';
-import type { GlobalAccountUserInfo, UserInfo } from 'lib/types/user-types.js';
+import type {
+ GlobalAccountUserInfo,
+ UserInfo,
+ LoggedInUserInfo,
+} from 'lib/types/user-types.js';
import { useServerCall } from 'lib/utils/action-utils.js';
import {
ChatThreadListItem,
chatThreadListItemHeight,
spacerHeight,
} from './chat-thread-list-item.react.js';
import type {
ChatTopTabsNavigationProp,
ChatNavigationProp,
} from './chat.react.js';
import {
type MessageListParams,
useNavigateToThread,
} from './message-list-types.js';
import { sidebarHeight } from './sidebar-item.react.js';
import Button from '../components/button.react.js';
import Search from '../components/search.react.js';
import {
SidebarListModalRouteName,
HomeChatThreadListRouteName,
BackgroundChatThreadListRouteName,
type NavigationRoute,
} from '../navigation/route-names.js';
import type { TabNavigationProp } from '../navigation/tab-navigator.react.js';
import { useSelector } from '../redux/redux-utils.js';
import {
type IndicatorStyle,
indicatorStyleSelector,
useStyles,
} from '../themes/colors.js';
import type { ScrollEvent } from '../types/react-native.js';
import { AnimatedView, type AnimatedStyleObj } from '../types/styles.js';
import { animateTowards } from '../utils/animation-utils.js';
const floatingActions = [
{
text: 'Compose',
icon: ,
name: 'compose',
position: 1,
},
];
/* eslint-disable import/no-named-as-default-member */
const { Value, Node, interpolateNode } = Animated;
/* eslint-enable import/no-named-as-default-member */
type Item =
| ChatThreadItem
| { +type: 'search', +searchText: string }
| { +type: 'empty', +emptyItem: React.ComponentType<{}> };
type BaseProps = {
+navigation:
| ChatTopTabsNavigationProp<'HomeChatThreadList'>
| ChatTopTabsNavigationProp<'BackgroundChatThreadList'>,
+route:
| NavigationRoute<'HomeChatThreadList'>
| NavigationRoute<'BackgroundChatThreadList'>,
+filterThreads: (threadItem: ThreadInfo) => boolean,
+emptyItem?: React.ComponentType<{}>,
};
type Props = {
...BaseProps,
// Redux state
+chatListData: $ReadOnlyArray,
- +viewerID: ?string,
+ +loggedInUserInfo: ?LoggedInUserInfo,
+threadSearchIndex: SearchIndex,
+styles: typeof unboundStyles,
+indicatorStyle: IndicatorStyle,
+usersWithPersonalThread: $ReadOnlySet,
+navigateToThread: (params: MessageListParams) => void,
// async functions that hit server APIs
+searchUsers: (usernamePrefix: string) => Promise,
};
type SearchStatus = 'inactive' | 'activating' | 'active';
type State = {
+searchStatus: SearchStatus,
+searchText: string,
+threadsSearchResults: Set,
+usersSearchResults: $ReadOnlyArray,
+openedSwipeableId: string,
+numItemsToDisplay: number,
};
type PropsAndState = { ...Props, ...State };
class ChatThreadList extends React.PureComponent {
state: State = {
searchStatus: 'inactive',
searchText: '',
threadsSearchResults: new Set(),
usersSearchResults: [],
openedSwipeableId: '',
numItemsToDisplay: 25,
};
searchInput: ?React.ElementRef;
flatList: ?FlatList- ;
scrollPos = 0;
clearNavigationBlurListener: ?() => mixed;
searchCancelButtonOpen: Value = new Value(0);
searchCancelButtonProgress: Node;
searchCancelButtonOffset: Node;
constructor(props: Props) {
super(props);
this.searchCancelButtonProgress = animateTowards(
this.searchCancelButtonOpen,
100,
);
this.searchCancelButtonOffset = interpolateNode(
this.searchCancelButtonProgress,
{ inputRange: [0, 1], outputRange: [0, 56] },
);
}
componentDidMount() {
this.clearNavigationBlurListener = this.props.navigation.addListener(
'blur',
() => {
this.setState({ numItemsToDisplay: 25 });
},
);
const chatNavigation: ?ChatNavigationProp<'ChatThreadList'> =
this.props.navigation.getParent();
invariant(chatNavigation, 'ChatNavigator should be within TabNavigator');
const tabNavigation: ?TabNavigationProp<'Chat'> =
chatNavigation.getParent();
invariant(tabNavigation, 'ChatNavigator should be within TabNavigator');
tabNavigation.addListener('tabPress', this.onTabPress);
BackHandler.addEventListener('hardwareBackPress', this.hardwareBack);
}
componentWillUnmount() {
this.clearNavigationBlurListener && this.clearNavigationBlurListener();
const chatNavigation: ?ChatNavigationProp<'ChatThreadList'> =
this.props.navigation.getParent();
invariant(chatNavigation, 'ChatNavigator should be within TabNavigator');
const tabNavigation: ?TabNavigationProp<'Chat'> =
chatNavigation.getParent();
invariant(tabNavigation, 'ChatNavigator should be within TabNavigator');
tabNavigation.removeListener('tabPress', this.onTabPress);
BackHandler.removeEventListener('hardwareBackPress', this.hardwareBack);
}
hardwareBack = () => {
if (!this.props.navigation.isFocused()) {
return false;
}
const { searchStatus } = this.state;
const isActiveOrActivating =
searchStatus === 'active' || searchStatus === 'activating';
if (!isActiveOrActivating) {
return false;
}
this.onSearchCancel();
return true;
};
componentDidUpdate(prevProps: Props, prevState: State) {
const { searchStatus } = this.state;
const prevSearchStatus = prevState.searchStatus;
const isActiveOrActivating =
searchStatus === 'active' || searchStatus === 'activating';
const wasActiveOrActivating =
prevSearchStatus === 'active' || prevSearchStatus === 'activating';
if (isActiveOrActivating && !wasActiveOrActivating) {
this.searchCancelButtonOpen.setValue(1);
} else if (!isActiveOrActivating && wasActiveOrActivating) {
this.searchCancelButtonOpen.setValue(0);
}
const { flatList } = this;
if (!flatList) {
return;
}
if (this.state.searchText !== prevState.searchText) {
flatList.scrollToOffset({ offset: 0, animated: false });
return;
}
if (searchStatus === 'activating' && prevSearchStatus === 'inactive') {
flatList.scrollToOffset({ offset: 0, animated: true });
}
}
onTabPress = () => {
if (!this.props.navigation.isFocused()) {
return;
}
if (this.scrollPos > 0 && this.flatList) {
this.flatList.scrollToOffset({ offset: 0, animated: true });
} else if (this.props.route.name === BackgroundChatThreadListRouteName) {
this.props.navigation.navigate({ name: HomeChatThreadListRouteName });
}
};
onSearchFocus = () => {
if (this.state.searchStatus !== 'inactive') {
return;
}
if (this.scrollPos === 0) {
this.setState({ searchStatus: 'active' });
} else {
this.setState({ searchStatus: 'activating' });
}
};
clearSearch() {
const { flatList } = this;
flatList && flatList.scrollToOffset({ offset: 0, animated: false });
this.setState({ searchStatus: 'inactive' });
}
onSearchBlur = () => {
if (this.state.searchStatus !== 'active') {
return;
}
this.clearSearch();
};
onSearchCancel = () => {
this.onChangeSearchText('');
this.clearSearch();
};
renderSearch(additionalProps?: $Shape>) {
const animatedSearchBoxStyle: AnimatedStyleObj = {
marginRight: this.searchCancelButtonOffset,
};
const searchBoxStyle = [
this.props.styles.searchBox,
animatedSearchBoxStyle,
];
const buttonStyle = [
this.props.styles.cancelSearchButtonText,
{ opacity: this.searchCancelButtonProgress },
];
return (
);
}
searchInputRef = (searchInput: ?React.ElementRef) => {
this.searchInput = searchInput;
};
renderItem = (row: { item: Item, ... }) => {
const item = row.item;
if (item.type === 'search') {
return (
{this.renderSearch({ active: false })}
);
}
if (item.type === 'empty') {
const EmptyItem = item.emptyItem;
return ;
}
return (
);
};
static keyExtractor = (item: Item) => {
if (item.type === 'chatThreadItem') {
return item.threadInfo.id;
} else if (item.type === 'empty') {
return 'empty';
} else {
return 'search';
}
};
static getItemLayout = (data: ?$ReadOnlyArray
- , index: number) => {
if (!data) {
return { length: 0, offset: 0, index };
}
const offset = ChatThreadList.heightOfItems(
data.filter((_, i) => i < index),
);
const item = data[index];
const length = item ? ChatThreadList.itemHeight(item) : 0;
return { length, offset, index };
};
static itemHeight = (item: Item) => {
if (item.type === 'search') {
return Platform.OS === 'ios' ? 54.5 : 55;
}
// itemHeight for emptyItem might be wrong because of line wrapping
// but we don't care because we'll only ever be rendering this item
// by itself and it should always be on-screen
if (item.type === 'empty') {
return 123;
}
let height = chatThreadListItemHeight;
height += item.sidebars.length * sidebarHeight;
if (item.sidebars.length > 0) {
height += spacerHeight;
}
return height;
};
static heightOfItems(data: $ReadOnlyArray
- ): number {
return _sum(data.map(ChatThreadList.itemHeight));
}
listDataSelector = createSelector(
(propsAndState: PropsAndState) => propsAndState.chatListData,
(propsAndState: PropsAndState) => propsAndState.searchStatus,
(propsAndState: PropsAndState) => propsAndState.searchText,
(propsAndState: PropsAndState) => propsAndState.threadsSearchResults,
(propsAndState: PropsAndState) => propsAndState.emptyItem,
(propsAndState: PropsAndState) => propsAndState.usersSearchResults,
(propsAndState: PropsAndState) => propsAndState.filterThreads,
- (propsAndState: PropsAndState) => propsAndState.viewerID,
+ (propsAndState: PropsAndState) => propsAndState.loggedInUserInfo,
(
reduxChatListData: $ReadOnlyArray,
searchStatus: SearchStatus,
searchText: string,
threadsSearchResults: Set,
emptyItem?: React.ComponentType<{}>,
usersSearchResults: $ReadOnlyArray,
filterThreads: ThreadInfo => boolean,
- viewerID: ?string,
+ loggedInUserInfo: ?LoggedInUserInfo,
): $ReadOnlyArray
- => {
const chatThreadItems = getThreadListSearchResults(
reduxChatListData,
searchText,
filterThreads,
threadsSearchResults,
usersSearchResults,
- viewerID,
+ loggedInUserInfo?.id,
);
const chatItems: Item[] = [...chatThreadItems];
if (emptyItem && chatItems.length === 0) {
chatItems.push({ type: 'empty', emptyItem });
}
if (searchStatus === 'inactive' || searchStatus === 'activating') {
chatItems.unshift({ type: 'search', searchText });
}
return chatItems;
},
);
partialListDataSelector = createSelector(
this.listDataSelector,
(propsAndState: PropsAndState) => propsAndState.numItemsToDisplay,
(items: $ReadOnlyArray
- , numItemsToDisplay: number) =>
items.slice(0, numItemsToDisplay),
);
get fullListData() {
return this.listDataSelector({ ...this.props, ...this.state });
}
get listData() {
return this.partialListDataSelector({ ...this.props, ...this.state });
}
onEndReached = () => {
if (this.listData.length === this.fullListData.length) {
return;
}
this.setState(prevState => ({
numItemsToDisplay: prevState.numItemsToDisplay + 25,
}));
};
render() {
let floatingAction;
if (Platform.OS === 'android') {
floatingAction = (
);
}
let fixedSearch;
const { searchStatus } = this.state;
if (searchStatus === 'active') {
fixedSearch = this.renderSearch({ autoFocus: true });
}
const scrollEnabled =
searchStatus === 'inactive' || searchStatus === 'active';
- // this.props.viewerID is in extraData since it's used by MessagePreview
+ // viewerID is in extraData since it's used by MessagePreview
// within ChatThreadListItem
+ const viewerID = this.props.loggedInUserInfo?.id;
+ const extraData = `${viewerID || ''} ${this.state.openedSwipeableId}`;
return (
{fixedSearch}
{floatingAction}
);
}
flatListRef = (flatList: ?FlatList
- ) => {
this.flatList = flatList;
};
onScroll = (event: ScrollEvent) => {
const oldScrollPos = this.scrollPos;
this.scrollPos = event.nativeEvent.contentOffset.y;
if (this.scrollPos !== 0 || oldScrollPos === 0) {
return;
}
if (this.state.searchStatus === 'activating') {
this.setState({ searchStatus: 'active' });
}
};
async searchUsers(usernamePrefix: string) {
if (usernamePrefix.length === 0) {
return [];
}
const { userInfos } = await this.props.searchUsers(usernamePrefix);
return userInfos.filter(
info =>
!this.props.usersWithPersonalThread.has(info.id) &&
- info.id !== this.props.viewerID,
+ info.id !== this.props.loggedInUserInfo?.id,
);
}
onChangeSearchText = async (searchText: string) => {
const results = this.props.threadSearchIndex.getSearchResults(searchText);
this.setState({
searchText,
threadsSearchResults: new Set(results),
numItemsToDisplay: 25,
});
const usersSearchResults = await this.searchUsers(searchText);
this.setState({ usersSearchResults });
};
onPressItem = (
threadInfo: ThreadInfo,
pendingPersonalThreadUserInfo?: UserInfo,
) => {
this.onChangeSearchText('');
if (this.searchInput) {
this.searchInput.blur();
}
this.props.navigateToThread({ threadInfo, pendingPersonalThreadUserInfo });
};
onPressSeeMoreSidebars = (threadInfo: ThreadInfo) => {
this.onChangeSearchText('');
if (this.searchInput) {
this.searchInput.blur();
}
this.props.navigation.navigate<'SidebarListModal'>({
name: SidebarListModalRouteName,
params: { threadInfo },
});
};
onSwipeableWillOpen = (threadInfo: ThreadInfo) => {
this.setState(state => ({ ...state, openedSwipeableId: threadInfo.id }));
};
composeThread = () => {
- if (!this.props.viewerID) {
+ const { loggedInUserInfo } = this.props;
+ if (!loggedInUserInfo) {
return;
}
const threadInfo = createPendingThread({
- viewerID: this.props.viewerID,
+ viewerID: loggedInUserInfo.id,
threadType: threadTypes.PRIVATE,
});
this.props.navigateToThread({ threadInfo, searching: true });
};
}
const unboundStyles = {
icon: {
fontSize: 28,
},
container: {
flex: 1,
},
searchContainer: {
backgroundColor: 'listBackground',
display: 'flex',
justifyContent: 'center',
flexDirection: 'row',
},
searchBox: {
flex: 1,
},
search: {
marginBottom: 8,
marginHorizontal: 18,
marginTop: 16,
},
cancelSearchButton: {
position: 'absolute',
right: 0,
top: 0,
bottom: 0,
display: 'flex',
justifyContent: 'center',
},
cancelSearchButtonText: {
color: 'link',
fontSize: 16,
paddingHorizontal: 16,
paddingTop: 8,
},
flatList: {
flex: 1,
backgroundColor: 'listBackground',
},
};
const ConnectedChatThreadList: React.ComponentType =
React.memo(function ConnectedChatThreadList(props: BaseProps) {
const boundChatListData = useFlattenedChatListData();
- const viewerID = useSelector(
- state => state.currentUserInfo && state.currentUserInfo.id,
- );
+ const loggedInUserInfo = useLoggedInUserInfo();
const threadSearchIndex = useGlobalThreadSearchIndex();
const styles = useStyles(unboundStyles);
const indicatorStyle = useSelector(indicatorStyleSelector);
const callSearchUsers = useServerCall(searchUsers);
const usersWithPersonalThread = useSelector(
usersWithPersonalThreadSelector,
);
const navigateToThread = useNavigateToThread();
return (
);
});
export default ConnectedChatThreadList;