-
Notifications
You must be signed in to change notification settings - Fork 281
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
fix: initial message load issues #2292
Conversation
const channelQueryOptions: ChannelQueryOptions<StreamChatGenerics> & { | ||
messages: { limit: number }; | ||
} = useMemo( | ||
() => | ||
defaultsDeep(propChannelQueryOptions, { | ||
messages: { limit: DEFAULT_INITIAL_CHANNEL_PAGE_SIZE }, | ||
}), | ||
[propChannelQueryOptions], | ||
); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Use DEFAULT_INITIAL_CHANNEL_PAGE_SIZE
when loading the first page of messages.
hasMore: hasMoreMessagesProbably( | ||
channel.state.messages.length, | ||
channelQueryOptions?.messages?.limit ?? DEFAULT_INITIAL_CHANNEL_PAGE_SIZE, | ||
), | ||
hasMore: | ||
channelInitializedExternally || | ||
hasMoreMessagesProbably( | ||
channel.state.messages.length, | ||
channelQueryOptions.messages.limit, | ||
), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If it wasn't the Channel
component itself that loaded the first page of messages, don't trust the message count to determine if there's more messages to load, since the function that loaded messages could have totally different limits.
@@ -79,6 +80,7 @@ export const usePaginatedChannels = < | |||
|
|||
const newOptions = { | |||
limit: options?.limit ?? MAX_QUERY_CHANNELS_LIMIT, | |||
message_limit: options?.message_limit ?? DEFAULT_INITIAL_CHANNEL_PAGE_SIZE, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Use DEFAULT_INITIAL_CHANNEL_PAGE_SIZE
to limit the number of messages when loading channel list.
|
||
return () => { | ||
scrollElement.removeEventListener('scroll', scrollListener, useCapture); | ||
scrollElement.removeEventListener('resize', scrollListener, useCapture); | ||
}; | ||
}, [initialLoad, scrollListener, useCapture]); | ||
}, [initialLoad, useCapture]); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Prevent this effect from firing excessively. Note that scrollListener
is now a ref, so it always has access to the latest props.
@@ -68,7 +69,7 @@ const MessageListWithContext = < | |||
headerPosition, | |||
read, | |||
renderMessages = defaultRenderMessages, | |||
messageLimit = 100, | |||
messageLimit = DEFAULT_NEXT_CHANNEL_PAGE_SIZE, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Remove hard-coded limit for message page size.
@@ -184,7 +185,7 @@ const VirtualizedMessageListWithContext = < | |||
loadMoreNewer, | |||
Message: MessageUIComponentFromProps, | |||
messageActions, | |||
messageLimit = 100, | |||
messageLimit = DEFAULT_NEXT_CHANNEL_PAGE_SIZE, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Remove hard-coded limit for message page size.
Size Change: +30.1 kB (+2%) Total Size: 1.79 MB
ℹ️ View Unchanged
|
scrollElement.addEventListener('scroll', scrollListener, useCapture); | ||
scrollElement.addEventListener('resize', scrollListener, useCapture); | ||
scrollListener(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Note that we'll only load a single additional page of messages in case the first page is not enough to cause overflow of the message list.
To keep loading pages until there is overflow, or there is no more pages to load, we can do something like this:
useLayoutEffect(() => {
if (!isLoading) {
scrollListenerRef.current?.();
}
}, [isLoading]);
However, I don't see a way to have the same behavior in the VirtualizedMessageList
, and since our default page size is rather big (100 messages), I think it's better to keep things consistent.
Codecov ReportAttention: Patch coverage is
Additional details and impacted files@@ Coverage Diff @@
## master #2292 +/- ##
==========================================
+ Coverage 85.03% 85.04% +0.01%
==========================================
Files 344 344
Lines 7689 7702 +13
Branches 2140 2143 +3
==========================================
+ Hits 6538 6550 +12
+ Misses 807 806 -1
- Partials 344 346 +2 ☔ View full report in Codecov by Sentry. |
Thank you, @myandrienko, closing #2275. |
## [11.10.0](v11.9.0...v11.10.0) (2024-02-23) ### Bug Fixes * initial message load issues ([#2292](#2292)) ([3685030](3685030)) ### Features * add customizable reaction details sorting ([#2290](#2290)) ([652e3a5](652e3a5)), closes [#2289](#2289) * add customizable reactions sorting ([#2289](#2289)) ([78c6107](78c6107))
🎉 This PR is included in version 11.10.0 🎉 The release is available on: Your semantic-release bot 📦🚀 |
🎯 Goal
This PR covers a range of issues that originate from incorrect initialization of message list state. All the cases are covered below.
🛠 Implementation details
ChannelList
,Channel
,MessageList
,VirtualizedMessageList
not respecting the message limitsWe have several default limits listed in the
limit.ts
file. However they were not always respected:MessageList
,VirtualizedMessageList
didn't useDEFAULT_NEXT_CHANNEL_PAGE_SIZE
and defaulted to the hard-coded message limit of 100.Channel
didn't useDEFAULT_INITIAL_CHANNEL_PAGE_SIZE
when fetching the first page of messages.ChannelList
didn't useDEFAULT_INITIAL_CHANNEL_PAGE_SIZE
as the message limit per channel when fetching channels.This is now fixed.
False-negative
hasMore
when messages are fetched byChannelList
We have two main ways to initialize the message list: it can be initialized by the
Channel
component; or it can be initialized by theChannelList
component (usingusePaginatedChannels
hook). If the message list is initialized by theChannelList
, theChannel
component will not try to query the channel, and will use messages fetched by theChannelList
instead.The problem here is that to determine if the channel has more messages to load, we used to compare the number of messages currently in the channel with the limit set on the
Channel
. But since it was not theChannel
but theChannelList
that fetched the messages, this check makes no sense:ChannelList
has its own separate limits.Consider this example:
The
Channel
component will compare the number of messages fetched by theChannelList
(5) with its own limit (10) and determine that the channel has no more messages.This is fixed by always setting
hasMore
to false in channel if it was not theChannel
component that fetched the messages. In the worst case we'll just make one redundant query.The "tall window" problem
In case the message container is very tall, or the message limit for the initial load is very small, we can run into a situation when the first page of messages is not long enough to trigger overflow in the message list. Without overflow, scroll events don't fire, so the next page of messages is never fetched.
This is not usually a problem in the
VirtualizedMessageList
, since it will immediately load the next page once in this case. However, theMessageList
didn't have this logic in place. It is now added (by invoking thescrollListener
once on mount).Note: both
VirtualizedMessageList
andMessageList
will only try loading the next page once if the first page of messages is not long enough to cause overflow. A better solution would be to keep loading pages until there's overflow or there is no more pages. This solution is not currently possible to implement inVirtualizedMessageList
, so for the sake of consistency I didn't implement itMessageList
(event though it is possible).In most cases this not an issue, since our default page size is 100 messages which is enough to fill even very tall containers.