Skip to content

Commit

Permalink
refactor: swap to brigadier logic
Browse files Browse the repository at this point in the history
  • Loading branch information
Boy0000 committed Jun 5, 2024
1 parent e136db9 commit df29060
Show file tree
Hide file tree
Showing 6 changed files with 314 additions and 389 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,306 @@
package com.mineinabyss.chatty

import com.github.shynixn.mccoroutine.bukkit.asyncDispatcher
import com.github.shynixn.mccoroutine.bukkit.launch
import com.github.shynixn.mccoroutine.bukkit.minecraftDispatcher
import com.mineinabyss.chatty.components.*
import com.mineinabyss.chatty.helpers.*
import com.mineinabyss.geary.papermc.tracking.entities.toGeary
import com.mineinabyss.geary.papermc.tracking.entities.toGearyOrNull
import com.mineinabyss.geary.serialization.getOrSetPersisting
import com.mineinabyss.geary.serialization.setPersisting
import com.mineinabyss.idofront.commands.brigadier.IdoRootCommand
import com.mineinabyss.idofront.commands.brigadier.commands
import com.mineinabyss.idofront.entities.toPlayer
import com.mineinabyss.idofront.events.call
import com.mineinabyss.idofront.textcomponents.miniMsg
import com.mojang.brigadier.arguments.StringArgumentType
import io.papermc.paper.command.brigadier.argument.ArgumentTypes
import io.papermc.paper.event.player.AsyncChatDecorateEvent
import kotlinx.coroutines.Job
import kotlinx.coroutines.delay
import kotlinx.coroutines.withContext
import net.kyori.adventure.text.Component
import net.kyori.adventure.text.format.NamedTextColor
import org.bukkit.command.CommandSender
import org.bukkit.entity.Player

object ChattyBrigadierCommands {

fun registerCommands() {
chatty.plugin.commands {
"chatty" {
"reload" {
executes {
chatty.plugin.createChattyContext()
sender.sendMessage(Component.text("Chatty has been reloaded!", NamedTextColor.GREEN))
}
}
"ping" {
"toggle" {
playerExecutes {
val gearyPlayer = player.toGeary()
val oldData = gearyPlayer.get<ChannelData>() ?: return@playerExecutes
gearyPlayer.setPersisting(oldData.copy(disablePingSound = !oldData.disablePingSound))
player.sendFormattedMessage(chatty.messages.ping.toggledPingSound)
}
}
"sound" {
val soundName by StringArgumentType.word()
playerExecutes {
val gearyPlayer = player.toGeary()
val oldData = gearyPlayer.get<ChannelData>() ?: return@playerExecutes
if (soundName() in alternativePingSounds) {
gearyPlayer.setPersisting(oldData.copy(pingSound = soundName()))
player.sendFormattedMessage(chatty.messages.ping.changedPingSound)
} else player.sendFormattedMessage(chatty.messages.ping.invalidPingSound)
}
}
}
"channels" {
executes {
(sender as? Player)?.sendFormattedMessage(chatty.messages.channels.availableChannels)
?: sender.sendRichMessage(chatty.messages.channels.availableChannels)
}
}
"commandspy" {
playerExecutes {
val gearyPlayer = player.toGeary()
if (gearyPlayer.has<CommandSpy>()) {
gearyPlayer.remove<CommandSpy>()
player.sendFormattedMessage(chatty.messages.spying.commandSpyOff)
} else {
gearyPlayer.getOrSetPersisting { CommandSpy() }
player.sendFormattedMessage(chatty.messages.spying.commandSpyOn)
}
}
}
"spy" {
val channelName by StringArgumentType.word()
playerExecutes {
val channel = chatty.config.channels[channelName()] ?: run {
player.sendFormattedMessage(chatty.messages.channels.noChannelWithName)
return@playerExecutes
}
val spy = player.toGeary().getOrSetPersisting { SpyOnChannels() }

when {
channel.channelType == ChannelType.GLOBAL ->
player.sendFormattedMessage(chatty.messages.spying.cannotSpyOnChannel)

!player.hasPermission(channel.permission) ->
player.sendFormattedMessage(chatty.messages.spying.cannotSpyOnChannel)

channel.key in spy.channels -> {
player.sendFormattedMessage(chatty.messages.spying.stopSpyingOnChannel)
spy.channels.remove(channel.key)
}

else -> {
spy.channels.add(channel.key)
player.sendFormattedMessage(chatty.messages.spying.startSpyingOnChannel)
}
}
}
}
"nickname" {
val nickname by StringArgumentType.greedyString()
playerExecutes {
val nickMessage = chatty.messages.nicknames
val bypassFormatPerm = sender.hasPermission(ChattyPermissions.BYPASS_TAG_PERM)

when {
!sender.hasPermission(ChattyPermissions.NICKNAME) ->
sender.sendFormattedMessage(nickMessage.selfDenied)

player.uniqueId != (sender as? Player)?.uniqueId -> {
when {
!sender.hasPermission(ChattyPermissions.NICKNAME_OTHERS) ->
sender.sendFormattedMessage(nickMessage.otherDenied, player)

nickname().isNullOrEmpty() -> {
player.chattyNickname = null
player.sendFormattedMessage(nickMessage.selfEmpty)
sender.sendFormattedMessage(nickMessage.otherEmpty, player)
}

!bypassFormatPerm && !nickname().verifyNickLength() ->
sender.sendFormattedMessage(nickMessage.tooLong)

nickname().isNotEmpty() -> {
player.chattyNickname = nickname()
sender.sendFormattedMessage(nickMessage.otherSuccess, player)
}
}
}

else -> {
when {
nickname().isNullOrEmpty() -> {
player.chattyNickname = null
sender.sendFormattedMessage(nickMessage.selfEmpty)
}

!bypassFormatPerm && !nickname().verifyNickLength() ->
sender.sendFormattedMessage(nickMessage.tooLong)

else -> {
(sender as? Player)?.chattyNickname = nickname()
sender.sendFormattedMessage(nickMessage.selfSuccess)
}
}
}
}
}
}
chatty.config.channels.values
.flatMap { it.channelAliases + it.key }
.forEach { channelName ->
channelName {
playerExecutes {
swapChannel(player, chatty.config.channels[channelName])
}
}
}
}
"global" { handleShortCutChannel(globalChannel()) }
"g" { handleShortCutChannel(globalChannel()) }
"local" { handleShortCutChannel(radiusChannel()) }
"l" { handleShortCutChannel(radiusChannel()) }
"admin" { handleShortCutChannel(adminChannel()) }
"a" { handleShortCutChannel(adminChannel()) }
"message" { handleMessage() }
"msg" { handleMessage() }
"reply" { handleReply() }

}
}

private fun IdoRootCommand.handleShortCutChannel(channel: MutableMap.MutableEntry<String, ChattyChannel>?) {
val message by StringArgumentType.greedyString()
playerExecutes {
player.shortcutCommand(channel, message())
}
}

private fun IdoRootCommand.handleMessage() {
val player by ArgumentTypes.player()
val message by StringArgumentType.greedyString()
playerExecutes {
this.player.handleSendingPrivateMessage(context.getArgument("player", Player::class.java), message(), false)
}
}

private fun IdoRootCommand.handleReply() {
val message by StringArgumentType.greedyString()
playerExecutes {
val player = sender as? Player ?: return@playerExecutes
player.toGeary().get<ChannelData>()?.lastMessager?.toPlayer()
?.let { player.handleSendingPrivateMessage(it, message(), true) }
?: player.sendFormattedMessage(chatty.messages.privateMessages.emptyReply)
}
}

private fun Player.shortcutCommand(
channel: Map.Entry<String, ChattyChannel>?,
message: String
) {
val chattyData = toGeary().get<ChannelData>() ?: return
val currentChannel = chattyData.channelId
when {
channel == null -> sendFormattedMessage(chatty.messages.channels.noChannelWithName)
channel.value.permission.isNotBlank() && !hasPermission(channel.value.permission) ->
sendFormattedMessage(chatty.messages.channels.missingChannelPermission)

message.isEmpty() || !chatty.config.chat.disableChatSigning -> swapChannel(this, channel.value)
else -> {
toGeary().setPersisting(chattyData.copy(channelId = channel.key, lastChannelUsedId = channel.key))
chatty.plugin.launch(chatty.plugin.asyncDispatcher) {
AsyncChatDecorateEvent(this@shortcutCommand, message.miniMsg()).call<AsyncChatDecorateEvent> {
GenericChattyChatEvent(this@shortcutCommand, result()).callEvent()
}
withContext(chatty.plugin.minecraftDispatcher) {
// chance that player logged out by now
toGearyOrNull()?.setPersisting(chattyData.copy(channelId = currentChannel))
}
}
}
}
}

fun swapChannel(player: Player, newChannel: ChattyChannel?) {
when {
newChannel == null ->
player.sendFormattedMessage(chatty.messages.channels.noChannelWithName)

newChannel.permission.isNotBlank() && !player.hasPermission(newChannel.permission) ->
player.sendFormattedMessage(chatty.messages.channels.missingChannelPermission)

else -> {
val gearyPlayer = player.toGeary()
val chattyData = gearyPlayer.get<ChannelData>() ?: return
gearyPlayer.setPersisting(
chattyData.copy(
channelId = newChannel.key,
lastChannelUsedId = newChannel.key
)
)
player.sendFormattedMessage(chatty.messages.channels.channelChanged)
}
}
}

private fun CommandSender.sendFormattedMessage(message: String, optionalPlayer: Player? = null) =
(optionalPlayer ?: this as? Player)?.let { player ->
this.sendMessage(translatePlaceholders(player, message).miniMsg(player.buildTagResolver(true)))
}

private fun Player.sendFormattedPrivateMessage(messageFormat: String, message: String, receiver: Player) =
this.sendMessage(
Component.textOfChildren(
translatePlaceholders(receiver, messageFormat).miniMsg(receiver.buildTagResolver(true)),
message.miniMsg(receiver.buildTagResolver(true))
)
)

private val replyMap = mutableMapOf<Player, Job>()
private fun handleReplyTimer(player: Player, chattyData: ChannelData): Job {
replyMap[player]?.let { return it }
replyMap[player]?.cancel()
return chatty.plugin.launch {
delay(chatty.config.privateMessages.messageReplyTime)
replyMap[player]?.cancel()
replyMap.remove(player)
player.toGeary().setPersisting(chattyData.copy(lastMessager = null))
}
}

private fun Player.handleSendingPrivateMessage(other: Player, message: String, isReply: Boolean = false) {
val chattyData = toGeary().get<ChannelData>() ?: return
when {
!chatty.config.privateMessages.enabled ->
sendFormattedMessage(chatty.messages.privateMessages.disabled)

isReply && chattyData.lastMessager == null ->
sendFormattedMessage(chatty.messages.privateMessages.emptyReply)

else -> {
if (message.isEmpty() || this == other) return

replyMap[other] = handleReplyTimer(other, chattyData)

this.sendFormattedPrivateMessage(chatty.config.privateMessages.messageSendFormat, message, other)
other.sendFormattedPrivateMessage(chatty.config.privateMessages.messageReceiveFormat, message, this)
val gearyOther = other.toGeary()
val otherChannelData = gearyOther.get<ChannelData>()
if (otherChannelData != null) {
gearyOther.setPersisting(otherChannelData.copy(lastMessager = uniqueId))
}
if (chatty.config.privateMessages.messageSendSound.isNotEmpty())
this.playSound(other.location, chatty.config.privateMessages.messageSendSound, 1f, 1f)
if (chatty.config.privateMessages.messageReceivedSound.isNotEmpty())
other.playSound(other.location, chatty.config.privateMessages.messageReceivedSound, 1f, 1f)
}
}
}
}
Loading

0 comments on commit df29060

Please sign in to comment.