All Downloads are FREE. Search and download functionalities are using the official Maven repository.

commonMain.gateway.handler.MessageEventHandler.kt Maven / Gradle / Ivy

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