const reply = await channel.sendMessage({
text: "This is a reply in a thread",
parent_id: parentMessageId,
show_in_channel: false,
});Threads & Replies
Threads allow users to reply to specific messages without cluttering the main channel conversation. A thread is created when a message is sent with a parent_id referencing another message.
Starting a Thread
Send a message with a parent_id to start a thread or add a reply to an existing thread.
Thread Parameters
| Name | Type | Description | Default | Optional |
|---|---|---|---|---|
| parent_id | string | ID of the parent message to reply to | ||
| show_in_channel | boolean | If true, the reply appears both in the thread and the main channel | false | ✓ |
Messages in threads support the same features as regular messages: reactions, attachments, and mentions.
Paginating Thread Replies
When querying a channel, thread replies are not included by default. The parent message includes a reply_count field. Use getReplies to fetch thread messages.
// Get the latest 20 replies
const replies = await channel.getReplies(parentMessageId, { limit: 20 });
// Get older replies (before message with ID "42")
const olderReplies = await channel.getReplies(parentMessageId, {
limit: 20,
id_lte: "42",
});
// Get oldest replies first
const oldestFirst = await channel.getReplies(parentMessageId, { limit: 20 }, [
{ created_at: 1 },
]);Inline Replies
Reply to a message inline without creating a thread. The referenced message appears within the new message. Use quoted_message_id instead of parent_id.
const message = await channel.sendMessage({
text: "I agree with this point",
quoted_message_id: originalMessageId,
});When querying messages, the quoted_message field is automatically populated:
{
"id": "new-message-id",
"text": "I agree with this point",
"quoted_message_id": "original-message-id",
"quoted_message": {
"id": "original-message-id",
"text": "The original message text"
}
}Inline replies are only available one level deep. If Message A replies to Message B, and Message B replies to Message C, you cannot access Message C through Message A. Fetch Message B directly to access its referenced message.
Thread List
Query all threads that the current user participates in. This is useful for building thread list views similar to Slack or Discord.
Querying Threads
Threads are returned with unread replies first, sorted by the latest reply timestamp in descending order.
const { threads } = await client.queryThreads();
for (const thread of threads) {
const state = thread.state.getLatestValue();
console.log(state.parentMessage.text);
console.log(state.replies);
console.log(state.participants);
console.log(state.read);
}Query Options
| Name | Type | Description | Default | Optional |
|---|---|---|---|---|
| reply_limit | number | Number of latest replies to fetch per thread | 2 | ✓ |
| participant_limit | number | Number of thread participants to fetch per thread | 100 | ✓ |
| limit | number | Maximum number of threads to return | 10 | ✓ |
| watch | boolean | If true, watch channels for the returned threads | true | ✓ |
| member_limit | number | Number of members to fetch per thread channel | 100 | ✓ |
Filtering and Sorting
Filter and sort threads using MongoDB-style query operators.
Supported Filter Fields
| Field | Type | Operators | Description |
|---|---|---|---|
channel_cid | string or list of strings | $eq, $in | Channel CID |
channel.disabled | boolean | $eq | Channel disabled status |
channel.team | string or list of strings | $eq, $in | Channel team |
parent_message_id | string or list of strings | $eq, $in | Parent message ID |
created_by_user_id | string or list of strings | $eq, $in | Thread creator's user ID |
created_at | string (RFC3339) | $eq, $gt, $lt, $gte, $lte | Thread creation timestamp |
updated_at | string (RFC3339) | $eq, $gt, $lt, $gte, $lte | Thread update timestamp |
last_message_at | string (RFC3339) | $eq, $gt, $lt, $gte, $lte | Last message timestamp |
Supported Sort Fields
active_participant_countcreated_atlast_message_atparent_message_idparticipant_countreply_countupdated_at
Use 1 for ascending order and -1 for descending order.
// Get threads created by a specific user, sorted by creation date
const { threads, next } = await client.queryThreads({
filter: {
created_by_user_id: { $eq: "user-1" },
updated_at: { $gte: "2024-01-01T00:00:00Z" },
},
sort: [{ created_at: -1 }],
limit: 10,
});
// Get next page
const { threads: page2 } = await client.queryThreads({
filter: {
created_by_user_id: { $eq: "user-1" },
updated_at: { $gte: "2024-01-01T00:00:00Z" },
},
sort: [{ created_at: -1 }],
limit: 10,
next,
});Getting a Thread by ID
Retrieve a specific thread using the parent message ID.
const thread = await client.getThread(parentMessageId, {
watch: true,
reply_limit: 10,
participant_limit: 25,
});Updating Thread Title and Custom Data
Assign a title and custom data to a thread.
// Set properties
const { thread } = await client.partialUpdateThread(threadId, {
set: {
title: "Project Discussion",
priority: "high",
},
});
// Remove properties
await client.partialUpdateThread(threadId, {
unset: ["priority"],
});Thread Unread Counts
Total Unread Threads
The total number of unread threads is available after connecting.
const { me } = await client.connectUser({ id: "user-id" }, token);
console.log(me.unread_threads);
// Or access via thread manager
client.threads.registerSubscriptions();
const { unreadThreadCount } = client.threads.state.getLatestValue();Marking Threads as Read or Unread
// Mark thread as read
await channel.markRead({ thread_id: parentMessageId });
// Mark thread as unread
await channel.markUnread({ thread_id: parentMessageId });Unread Count Per Thread
const response = await client.getUnreadCount();
console.log(response.total_unread_threads_count);
for (const thread of response.threads) {
console.log(thread.parent_message_id);
console.log(thread.unread_count);
console.log(thread.last_read);
}Thread Manager
The ThreadManager class provides built-in pagination and state management for threads.
// Access the client's thread manager
const threadManager = client.threads;
// Subscribe to state updates
const unsubscribe = threadManager.state.subscribe((state) => {
console.log(state.threads);
console.log(state.unreadThreadCount);
});
// Load threads
await threadManager.loadNextPage();
// Access current state
const { threads } = threadManager.state.getLatestValue();Event Handling
Register subscriptions to receive real-time updates for threads.
const { threads } = await client.queryThreads({ watch: true, limit: 10 });
const [thread] = threads;
// Register event handlers for a single thread
thread.registerSubscriptions();
const unsubscribe = thread.state.subscribe((state) => {
console.log(state.replies);
});The watch parameter is required when querying threads to receive real-time updates.
For ThreadManager, call registerSubscriptions once to automatically manage subscriptions for all loaded threads:
const threadManager = client.threads;
threadManager.registerSubscriptions();
await threadManager.loadNextPage();
// All threads are now listening to channel events
const { threads } = threadManager.state.getLatestValue();