diff --git a/app/soapbox/features/chats/components/chat_message_list.js b/app/soapbox/features/chats/components/chat_message_list.js index 4c86677a4..a0e645856 100644 --- a/app/soapbox/features/chats/components/chat_message_list.js +++ b/app/soapbox/features/chats/components/chat_message_list.js @@ -10,6 +10,8 @@ import emojify from 'soapbox/features/emoji/emoji'; import classNames from 'classnames'; import { escape, throttle } from 'lodash'; +const scrollBottom = (elem) => elem.scrollHeight - elem.offsetHeight - elem.scrollTop; + const makeEmojiMap = record => record.get('emojis', ImmutableList()).reduce((map, emoji) => { return map.set(`:${emoji.get('shortcode')}:`, emoji); }, ImmutableMap()); @@ -19,7 +21,7 @@ const mapStateToProps = (state, { chatMessageIds }) => ({ chatMessages: chatMessageIds.reduce((acc, curr) => { const chatMessage = state.getIn(['chat_messages', curr]); return chatMessage ? acc.push(chatMessage) : acc; - }, ImmutableList()).sort(), + }, ImmutableList()).sort().reverse(), }); export default @connect(mapStateToProps) @@ -39,6 +41,10 @@ class ChatMessageList extends ImmutablePureComponent { chatMessages: ImmutableList(), } + state = { + isLoading: false, + } + scrollToBottom = () => { if (!this.messagesEnd) return; this.messagesEnd.scrollIntoView(); @@ -46,7 +52,6 @@ class ChatMessageList extends ImmutablePureComponent { setMessageEndRef = (el) => { this.messagesEnd = el; - this.scrollToBottom(); }; getFormattedTimestamp = (chatMessage) => { @@ -75,11 +80,19 @@ class ChatMessageList extends ImmutablePureComponent { } componentDidMount() { + this.node.addEventListener('scroll', this.handleScroll); } componentDidUpdate(prevProps) { - if (prevProps.chatMessages !== this.props.chatMessages) - this.scrollToBottom(); + const oldCount = prevProps.chatMessages.count(); + const newCount = this.props.chatMessages.count(); + const isNearBottom = scrollBottom(this.node) < 150; + const historyAdded = prevProps.chatMessages.getIn([-1, 'id']) !== this.props.chatMessages.getIn([-1, 'id']); + + if (oldCount !== newCount) { + if (isNearBottom) this.scrollToBottom(); + if (historyAdded) this.setState({ isLoading: false }); + } } componentWillUnmount() { @@ -87,13 +100,14 @@ class ChatMessageList extends ImmutablePureComponent { } handleLoadMore = () => { - const { dispatch, chatId, chatMessageIds } = this.props; - const maxId = chatMessageIds.last(); + const { dispatch, chatId, chatMessages } = this.props; + const maxId = chatMessages.getIn([-1, 'id']); dispatch(fetchChatMessages(chatId, maxId)); + this.setState({ isLoading: true }); } handleScroll = throttle(() => { - if (this.node.scrollTop < 100) this.handleLoadMore(); + if (this.node.scrollTop < 150 && !this.state.isLoading) this.handleLoadMore(); }, 150, { trailing: true, }); @@ -112,7 +126,6 @@ class ChatMessageList extends ImmutablePureComponent { setRef = (c) => { this.node = c; - this.node.addEventListener('scroll', this.handleScroll); } render() { @@ -120,6 +133,7 @@ class ChatMessageList extends ImmutablePureComponent { return (
+
{chatMessages.map(chatMessage => (
))} -
); } diff --git a/app/styles/chats.scss b/app/styles/chats.scss index 011543ca3..38550ac3b 100644 --- a/app/styles/chats.scss +++ b/app/styles/chats.scss @@ -99,12 +99,18 @@ .chat-messages { overflow-y: scroll; flex: 1; + display: flex; + flex-direction: column-reverse; } .chat-message { - margin: 14px 10px; + padding: 7px 10px; display: flex; + &:last-child { + padding-top: 14px; + } + &__bubble { font-size: 15px; padding: 4px 10px;