commonMain.gateway.handler.MessageEventHandler.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of kord-core Show documentation
Show all versions of kord-core Show documentation
Idiomatic Kotlin Wrapper for The Discord API
The newest version!
package dev.kord.core.gateway.handler
import dev.kord.cache.api.put
import dev.kord.cache.api.query
import dev.kord.common.entity.optional.*
import dev.kord.core.Kord
import dev.kord.core.cache.data.*
import dev.kord.core.cache.idEq
import dev.kord.core.entity.Member
import dev.kord.core.entity.Message
import dev.kord.core.entity.ReactionEmoji
import dev.kord.core.event.message.*
import dev.kord.gateway.*
import kotlinx.coroutines.flow.map
import kotlinx.coroutines.flow.singleOrNull
import kotlinx.coroutines.flow.toSet
internal class MessageEventHandler : BaseGatewayEventHandler() {
override suspend fun handle(
event: Event,
shard: Int,
kord: Kord,
context: LazyContext?,
): dev.kord.core.event.Event? = when (event) {
is MessageCreate -> handle(event, shard, kord, context)
is MessageUpdate -> handle(event, shard, kord, context)
is MessageDelete -> handle(event, shard, kord, context)
is MessageDeleteBulk -> handle(event, shard, kord, context)
is MessageReactionAdd -> handle(event, shard, kord, context)
is MessageReactionRemove -> handle(event, shard, kord, context)
is MessageReactionRemoveAll -> handle(event, shard, kord, context)
is MessageReactionRemoveEmoji -> handle(event, shard, kord, context)
else -> null
}
private suspend fun handle(
event: MessageCreate,
shard: Int,
kord: Kord,
context: LazyContext?,
): MessageCreateEvent = with(event.message) {
val data = MessageData.from(this)
kord.cache.put(data)
kord.cache.query { idEq(ChannelData::id, channelId) }.update { channel ->
channel.copy(
lastMessageId = data.id.optionalSnowflake(),
messageCount = channel.messageCount.map { it + 1 },
totalMessageSent = channel.totalMessageSent.map { it + 1 },
)
}
//get the user data only if it exists and the user isn't a webhook
val userData = if (webhookId is OptionalSnowflake.Missing) {
UserData.from(author).also { kord.cache.put(it) }
} else null
//get the member and cache the member. We need the user, guild id and member to be present
val member = if (userData != null && guildId is OptionalSnowflake.Value && member is Optional.Value) {
val memberData = MemberData.from(author.id, guildId.value!!, member.value!!)
kord.cache.put(memberData)
Member(memberData, userData, kord)
} else null
//cache interaction user if present.
if (interaction is Optional.Value) {
val interactionUserData = UserData.from(interaction.value!!.user)
kord.cache.put(interactionUserData)
}
mentions.forEach {
val user = UserData.from(it)
kord.cache.put(user)
it.member.value?.let { member ->
kord.cache.put(MemberData.from(userId = user.id, guildId = guildId.value!!, member))
}
}
MessageCreateEvent(Message(data, kord), guildId.value, member, shard, context?.get())
}
private suspend fun handle(
event: MessageUpdate,
shard: Int,
kord: Kord,
context: LazyContext?,
): MessageUpdateEvent = with(event.message) {
val query = kord.cache.query { idEq(MessageData::id, id) }
val old = query.asFlow().map { Message(it, kord) }.singleOrNull()
query.update { it + this }
mentions.orEmpty().forEach {
val user = UserData.from(it)
kord.cache.put(user)
it.member.value?.let { member ->
kord.cache.put(MemberData.from(userId = user.id, guildId = guildId.value!!, member))
}
}
MessageUpdateEvent(id, channelId, this, old, kord, shard, context?.get())
}
private suspend fun handle(
event: MessageDelete,
shard: Int,
kord: Kord,
context: LazyContext?,
): MessageDeleteEvent = with(event.message) {
val query = kord.cache.query { idEq(MessageData::id, id) }
val removed = query.singleOrNull()?.let { Message(it, kord) }
query.remove()
kord.cache.query { idEq(ChannelData::id, channelId) }.update { channel ->
channel.copy(messageCount = channel.messageCount.map { it - 1 })
}
MessageDeleteEvent(id, channelId, guildId.value, removed, kord, shard, context?.get())
}
private suspend fun handle(
event: MessageDeleteBulk,
shard: Int,
kord: Kord,
context: LazyContext?,
): MessageBulkDeleteEvent =
with(event.messageBulk) {
val ids = ids.toSet()
val query = kord.cache.query { MessageData::id `in` ids }
val removed = query.asFlow().map { Message(it, kord) }.toSet()
query.remove()
kord.cache.query { idEq(ChannelData::id, channelId) }.update { channel ->
channel.copy(messageCount = channel.messageCount.map { it - ids.size })
}
MessageBulkDeleteEvent(
ids,
removed,
channelId,
guildId.value,
kord,
shard,
context?.get(),
)
}
private suspend fun handle(
event: MessageReactionAdd,
shard: Int,
kord: Kord,
context: LazyContext?,
): ReactionAddEvent =
with(event.reaction) {
/**
* Reactions added will *always* have a name, the only case in which name is null is when a guild reaction
* no longer exists (only id is kept). Reacting with a non-existing reaction *should* be impossible.
**/
val reaction = when (val id = emoji.id) {
null -> ReactionEmoji.Unicode(emoji.name!!)
else -> ReactionEmoji.Custom(id, emoji.name!!, emoji.animated.orElse(false))
}
kord.cache.query { idEq(MessageData::id, messageId) }.update {
val isMe = kord.selfId == event.reaction.userId
val reactions = if (it.reactions.value.isNullOrEmpty()) {
listOf(ReactionData.from(1, isMe, emoji))
} else {
val reactions = it.reactions.orEmpty()
val reactionData = reactions.firstOrNull { reaction ->
if (emoji.id == null) reaction.emojiName == emoji.name
else reaction.emojiId == emoji.id && reaction.emojiName == emoji.name
}
when (reactionData) {
null -> reactions + ReactionData.from(1, isMe, emoji)
else -> (reactions - reactionData) + reactionData.copy(
count = reactionData.count + 1,
me = isMe
)
}
}
it.copy(reactions = Optional.Value(reactions))
}
ReactionAddEvent(
userId,
channelId,
messageId,
guildId.value,
reaction,
messageAuthorId.value,
kord,
shard,
context?.get(),
)
}
private suspend fun handle(
event: MessageReactionRemove,
shard: Int,
kord: Kord,
context: LazyContext?,
): ReactionRemoveEvent =
with(event.reaction) {
/**
* Reactions removed will *sometimes* have a name, the only case in which name is null is when a guild reaction
* no longer exists (only id is kept). Removing a non-existing reaction *should* be possible.
**/
val reaction = when (val id = emoji.id) {
null -> ReactionEmoji.Unicode(emoji.name!!)
else -> ReactionEmoji.Custom(id, emoji.name ?: "", emoji.animated.orElse(false))
}
kord.cache.query { idEq(MessageData::id, messageId) }.update {
val oldReactions = it.reactions.value ?: return@update it
if (oldReactions.isEmpty()) return@update it
val me = kord.selfId == event.reaction.userId
val reactionData = oldReactions.firstOrNull { reaction ->
if (emoji.id == null) reaction.emojiName == emoji.name
else reaction.emojiId == emoji.id && reaction.emojiName == emoji.name
} ?: return@update it
val reactions = when (val count = reactionData.count - 1) {
0 -> (oldReactions - reactionData)
else -> (oldReactions - reactionData) + reactionData.copy(
count = count,
me = reactionData.me xor me
)
}
it.copy(reactions = Optional.Value(reactions))
}
ReactionRemoveEvent(
userId,
channelId,
messageId,
guildId.value,
reaction,
kord,
shard,
context?.get(),
)
}
private suspend fun handle(
event: MessageReactionRemoveAll,
shard: Int,
kord: Kord,
context: LazyContext?,
): ReactionRemoveAllEvent =
with(event.reactions) {
kord.cache.query { idEq(MessageData::id, messageId) }
.update { it.copy(reactions = Optional.Missing()) }
ReactionRemoveAllEvent(
channelId,
messageId,
guildId.value,
kord,
shard,
context?.get(),
)
}
private suspend fun handle(
event: MessageReactionRemoveEmoji,
shard: Int,
kord: Kord,
context: LazyContext?,
): ReactionRemoveEmojiEvent =
with(event.reaction) {
kord.cache.query { idEq(MessageData::id, messageId) }
.update { it.copy(reactions = it.reactions.map { list -> list.filter { data -> data.emojiName != emoji.name } }) }
val data = ReactionRemoveEmojiData.from(this)
ReactionRemoveEmojiEvent(data, kord, shard, context?.get())
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy