Custom Message List

On this page you’ll learn how to create a simple message list screen, without using any of our SDK’s UI components. We will only use the low-level ChatClient for state, a completely custom view model and Jetpack Compose for the UI.

Custom Message List

Custom View Model

First, we create a view model that receives a ChatClient instance. It exposes a uiState property of type StateFlow<MessageListUiState>. This allows the UI layer to observe state changes and update the UI accordingly.

Next, we define the getMessages(cid: String) method. It receives the cid (channel id) as an argument from the UI and calls the chatClient.watchChannelAsState method, passing the cid to it. This returns a StateFlow<ChannelState>, which represents the state of that specific channel. We store this state in the channelStateFlow variable.

Finally, we start collecting channelStateFlow and it’s messages: StateFlow<List<Message>> property, and update _uiState accordingly.

Here is the full code for the view model:

class CustomMessageListViewModel(val chatClient: ChatClient = ChatClient.instance()) : ViewModel() {
    private val _uiState = MutableStateFlow(MessageListUiState())
    val uiState = _uiState.asStateFlow()

    fun getMessages(cid: String) { // cid will be provided in the UI, see section below
        val channelStateFlow: StateFlow<ChannelState?> = chatClient.watchChannelAsState(
            cid = cid,
            messageLimit = 30,
            coroutineScope = viewModelScope
        )

        viewModelScope.launch {
            channelStateFlow.collect { channelState ->
                if (channelState != null) {
                    channelState.messages.collect { messages ->
                        _uiState.update { it.copy(messages = messages, error = null) }
                    }
                } else {
                    _uiState.update { it.copy(error = "Cannot load messages") }
                }
            }
        }
    }
}

data class MessageListUiState(
    val messages: List<Message> = emptyList(),
    val error: String? = null,
)

Custom UI

For the UI, we create a composable called CustomMessageListScreen that gets the view model that we defined in the previous section and the cid (channel id) as parameters.

The cid (channel id) is usually provided when the app navigates from a channel list screen to a message list screen, like we have here.

The composable then starts observing our view model’s uiState: StateFlow property for changes and triggers message list loading by calling the view model’s getMessages method with the channel cid.

Next, we pass the state to other functions to build the appropriate UI.

@Composable
fun CustomMessageListScreen(
    viewModel: CustomMessageListViewModel = viewModel(), // We use a custom view model, see section above
    cid: String?,
) {
    val uiState by viewModel.uiState.collectAsStateWithLifecycle()

    LaunchedEffect(key1 = Unit) { cid?.let { viewModel.getMessages(it) } }

    if (uiState.error == null) {
        CustomMessageList(messages = uiState.messages)
    } else {
        Error(message = uiState.error!!)
    }
}

The other composable functions used to create the screen are listed below:

@Composable
private fun CustomMessageList(messages: List<Message>) {
    LazyColumn(
        modifier = Modifier.fillMaxSize(),
        contentPadding = PaddingValues(all = 15.dp),
        verticalArrangement = Arrangement.spacedBy(15.dp),
        reverseLayout = true,
    ) {
        items(messages) { message ->
            if (message.text != "") CustomMessageListItem(message = message)
        }
    }
}

@Composable
private fun CustomMessageListItem(message: Message) {
    val timeFormat = SimpleDateFormat("HH:mm", Locale.getDefault())

    Column {
        Text(text = "${message.user.name} said:", fontSize = 12.sp, fontWeight = FontWeight.Light)
        Spacer(modifier = Modifier.height(5.dp))
        Row(
            horizontalArrangement = Arrangement.SpaceBetween,
            verticalAlignment = Alignment.CenterVertically,
            modifier = Modifier
                .background(
                    color = Color(0xFFEEEEEE),
                    shape = RoundedCornerShape(topStart = 0.dp, topEnd = 10.dp, bottomEnd = 10.dp, bottomStart = 10.dp)
                )
                .padding(all = 10.dp)
        ) {
            Text(text = message.text)
            Spacer(modifier = Modifier.width(15.dp))
            message.createdAt?.let {
                Text(text = timeFormat.format(it), fontSize = 12.sp, fontWeight = FontWeight.Light)
            }
        }
    }
}

@Composable
private fun Error(message: String) {
    Box(
        modifier = Modifier.fillMaxSize(),
        contentAlignment = Alignment.Center
    ) {
        Text(text = message)
    }
}

Paginating Messages

Check if the message list is scrolled to the end

To create simple pagination for the message list, we first extend the LazyListState class with a composable function named OnListEndReached.

This function takes two parameters: buffer and handler. handler is a lambda that will be executed when the end of the list (considering the buffer) is reached.

Inside the function, a derivedStateOf block is used to create a boolean state that determines whether handler should be called. This state, named shouldCallHandler, is true when the index of the last visible item equals the total number of items minus the buffer.

Finally, a LaunchedEffect block is used to call the handler function when shouldCallHandler is true. This effect is re-launched every time shouldCallHandler changes. In other words, handler will be called every time the user scrolls the list to its end (considering the buffer).

@Composable
fun LazyListState.OnListEndReached(buffer: Int = 0, handler: () -> Unit) {
    val shouldCallHandler by remember {
        derivedStateOf {
            layoutInfo.visibleItemsInfo.lastOrNull()?.let { lastVisibleItem ->
                lastVisibleItem.index == layoutInfo.totalItemsCount - 1 - buffer
            } ?: false
        }
    }

    LaunchedEffect(shouldCallHandler) {
        if (shouldCallHandler) handler()
    }
}

Load more messages in the View Model

In order to support pagination, we must add a method to the View Model that will load more messages when the list reaches the end. In our case, we’ll fetch older messages:

fun loadMoreMessages(cid: String) {
    chatClient.loadOlderMessages(cid = cid, messageLimit = 30).enqueue(
        onError = { streamError ->
            Log.e("[Messages]", "Cannot load more messages. Error: ${streamError.message}")
        }
    )   
}

Put everything together in the UI

To make it work, we just need to make a few changes to the UI.

We add a new lambda parameter named onListEndReached to the CustomMessageList composable and we use the list state OnListEndReached extension method what we created earlier:

private fun CustomMessageList(messages: List<Message>, onListEndReached: () -> Unit) {
    val listState = rememberLazyListState()
    listState.OnListEndReached(buffer = 5, handler = onListEndReached)

    LazyColumn(
        // ...
        state = listState,
        // ...
    ) {
        // ...
    }
}

In CustomMessageListScreen, we call the view model method that loads more messages when the onListEndReached lambda is called:

// ...
CustomMessageList(
    // ...
    onListEndReached = { cid?.let { viewModel.loadMoreMessages(it) } }
)
//...

Now, when the user scrolls to the end of the list, older messages will be loaded.

There might be edge cases that are not covered by this pagination implementation. Its purpose is to demonstrate how more messages can be fetched in a pagination scenario. Make sure you test your production implementation.

More Resources

If you want to learn how to use and customize our Compose UI Components, see this page.

© Getstream.io, Inc. All Rights Reserved.