summaryrefslogtreecommitdiff
path: root/src/main/kotlin/io/dico/parcels2/command/NamedParcelTargetType.kt
blob: ceb226f3a524ecae0dec5805c7780bd22aa48af3 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
package io.dico.parcels2.command

import io.dico.dicore.command.parameter.ArgumentBuffer
import io.dico.dicore.command.parameter.Parameter
import io.dico.dicore.command.parameter.type.ParameterConfig
import io.dico.dicore.command.parameter.type.ParameterType
import io.dico.parcels2.ParcelWorld
import io.dico.parcels2.Worlds
import org.bukkit.Bukkit
import org.bukkit.OfflinePlayer
import org.bukkit.command.CommandSender
import org.bukkit.entity.Player

class NamedParcelTarget(val world: ParcelWorld, val player: OfflinePlayer, val index: Int)

@Target(AnnotationTarget.VALUE_PARAMETER)
@Retention(AnnotationRetention.RUNTIME)
annotation class NamedParcelDefault(val value: NamedParcelDefaultValue)

enum class NamedParcelDefaultValue {
    FIRST_OWNED,
    NULL
}

class NamedParcelTargetConfig : ParameterConfig<NamedParcelDefault,
    NamedParcelDefaultValue>(NamedParcelDefault::class.java) {

    override fun toParameterInfo(annotation: NamedParcelDefault): NamedParcelDefaultValue {
        return annotation.value
    }
}

class ParcelHomeParameterType(val worlds: Worlds) : ParameterType<NamedParcelTarget,
    NamedParcelDefaultValue>(NamedParcelTarget::class.java, NamedParcelTargetConfig()) {

    val regex = Regex.fromLiteral("((.+)->)?(.+)|((.+):([0-9]+))")

    private fun requirePlayer(sender: CommandSender, parameter: Parameter<*, *>): Player {
        if (sender !is Player) invalidInput(parameter, "console cannot omit the player name")
        return sender
    }

    @Suppress("UsePropertyAccessSyntax")
    private fun getOfflinePlayer(input: String, parameter: Parameter<*, *>) = Bukkit.getOfflinePlayer(input)
        ?.takeIf { it.isOnline() || it.hasPlayedBefore() }
        ?: invalidInput(parameter, "do not know who $input is")

    override fun parse(parameter: Parameter<NamedParcelTarget, NamedParcelDefaultValue>,
                       sender: CommandSender, buffer: ArgumentBuffer): NamedParcelTarget {
        val matchResult = regex.matchEntire(buffer.next())
            ?: invalidInput(parameter, "must be a player, index, or player:index (/${regex.pattern}/)")

        val world = worlds.getTargetWorld(matchResult.groupValues[2], sender, parameter)

        matchResult.groupValues[3].takeUnless { it.isEmpty() }?.let {
            // first group was matched, it's a player or an int
            it.toIntOrNull()?.let {
                requirePlayer(sender, parameter)
                return NamedParcelTarget(world, sender as Player, it)
            }

            return NamedParcelTarget(world, getOfflinePlayer(it, parameter), 0)
        }

        val player = getOfflinePlayer(matchResult.groupValues[5], parameter)
        val index = matchResult.groupValues[6].toIntOrNull()
            ?: invalidInput(parameter, "couldn't parse int")

        return NamedParcelTarget(world, player, index)
    }

    override fun getDefaultValue(parameter: Parameter<NamedParcelTarget, NamedParcelDefaultValue>,
                                 sender: CommandSender, buffer: ArgumentBuffer): NamedParcelTarget? {
        if (parameter.paramInfo == NamedParcelDefaultValue.NULL) {
            return null
        }

        val world = worlds.getTargetWorld(null, sender, parameter)
        val player = requirePlayer(sender, parameter)
        return NamedParcelTarget(world, player, 0)
    }

}