forked from Ivasoft/mattermost-mobile
* Disable attempt to open User Profile screen * Fix loading state alignment when switching servers * Fix user is typing style and show in thread view * Add additional space at the bottom of post list when CRT is disabled * header back button position * add margin bottom to typing component
166 lines
5.0 KiB
TypeScript
166 lines
5.0 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 [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;
|
|
}
|
|
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(() => {
|
|
setRefresh(Date.now());
|
|
timeoutToDisappear.current = undefined;
|
|
}, 500);
|
|
} else {
|
|
setRefresh(Date.now());
|
|
}
|
|
}, []);
|
|
|
|
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]);
|
|
|
|
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);
|
|
|