Files
mattermost-mobile/app/components/post_draft/typing/index.tsx
2022-08-02 19:01:45 +02:00

187 lines
5.6 KiB
TypeScript

// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
import React, {useCallback, useEffect, useRef, useState} from 'react';
import {DeviceEventEmitter} from 'react-native';
import Animated, {useAnimatedStyle, useSharedValue, withTiming} from 'react-native-reanimated';
import FormattedText from '@components/formatted_text';
import {Events} from '@constants';
import {TYPING_HEIGHT} from '@constants/post_draft';
import {useTheme} from '@context/theme';
import {changeOpacity, makeStyleSheetFromTheme} from '@utils/theme';
import {typography} from '@utils/typography';
type Props = {
channelId: string;
rootId: string;
}
const getStyleSheet = makeStyleSheetFromTheme((theme: Theme) => {
return {
typing: {
color: changeOpacity(theme.centerChannelColor, 0.7),
paddingHorizontal: 10,
...typography('Body', 75),
},
};
});
function Typing({
channelId,
rootId,
}: Props) {
const typingHeight = useSharedValue(0);
const typing = useRef<Array<{id: string; now: number; username: string}>>([]);
const timeoutToDisappear = useRef<NodeJS.Timeout>();
const mounted = useRef(false);
const [refresh, setRefresh] = useState(0);
const theme = useTheme();
const style = getStyleSheet(theme);
// This moves the list of post up. This may be rethought by UX in https://mattermost.atlassian.net/browse/MM-39681
const typingAnimatedStyle = useAnimatedStyle(() => {
return {
height: withTiming(typingHeight.value),
marginBottom: 4,
};
});
const onUserStartTyping = useCallback((msg: any) => {
if (channelId !== msg.channelId) {
return;
}
const msgRootId = msg.parentId || msg.rootId || '';
if (rootId !== msgRootId) {
return;
}
typing.current = typing.current.filter(({id}) => id !== msg.userId);
typing.current.push({id: msg.userId, now: msg.now, username: msg.username});
if (timeoutToDisappear.current) {
clearTimeout(timeoutToDisappear.current);
timeoutToDisappear.current = undefined;
}
if (mounted.current) {
setRefresh(Date.now());
}
}, [channelId, rootId]);
const onUserStopTyping = useCallback((msg: any) => {
if (channelId !== msg.channelId) {
return;
}
const msgRootId = msg.parentId || msg.rootId || '';
if (rootId !== msgRootId) {
return;
}
typing.current = typing.current.filter(({id, now}) => id !== msg.userId && now !== msg.now);
if (timeoutToDisappear.current) {
clearTimeout(timeoutToDisappear.current);
timeoutToDisappear.current = undefined;
}
if (typing.current.length === 0) {
timeoutToDisappear.current = setTimeout(() => {
if (mounted.current) {
setRefresh(Date.now());
}
timeoutToDisappear.current = undefined;
}, 500);
} else if (mounted.current) {
setRefresh(Date.now());
}
}, [channelId, rootId]);
useEffect(() => {
mounted.current = true;
return () => {
mounted.current = false;
};
}, []);
useEffect(() => {
const listener = DeviceEventEmitter.addListener(Events.USER_TYPING, onUserStartTyping);
return () => {
listener.remove();
};
}, [onUserStartTyping]);
useEffect(() => {
const listener = DeviceEventEmitter.addListener(Events.USER_STOP_TYPING, onUserStopTyping);
return () => {
listener.remove();
};
}, [onUserStopTyping]);
useEffect(() => {
typingHeight.value = typing.current.length ? TYPING_HEIGHT : 0;
}, [refresh]);
useEffect(() => {
typing.current = [];
typingHeight.value = 0;
if (timeoutToDisappear.current) {
clearTimeout(timeoutToDisappear.current);
timeoutToDisappear.current = undefined;
}
}, [channelId, rootId]);
const renderTyping = () => {
const nextTyping = typing.current.map(({username}) => username);
// Max three names
nextTyping.splice(3);
const numUsers = nextTyping.length;
switch (numUsers) {
case 0:
return null;
case 1:
return (
<FormattedText
id='msg_typing.isTyping'
defaultMessage='{user} is typing...'
style={style.typing}
ellipsizeMode='tail'
numberOfLines={1}
values={{
user: nextTyping[0],
}}
/>
);
default: {
const last = nextTyping.pop();
return (
<FormattedText
id='msg_typing.areTyping'
defaultMessage='{users} and {last} are typing...'
style={style.typing}
ellipsizeMode='tail'
numberOfLines={1}
values={{
users: (nextTyping.join(', ')),
last,
}}
/>
);
}
}
};
return (
<Animated.View style={typingAnimatedStyle}>
{renderTyping()}
</Animated.View>
);
}
export default React.memo(Typing);