Implement teleport cooldowns
This commit is contained in:
parent
b5e431c7f6
commit
6079ac4852
@ -4,6 +4,18 @@ import java.util.*
|
|||||||
import kotlin.Comparator
|
import kotlin.Comparator
|
||||||
|
|
||||||
typealias Comparison<V> = (V) -> Int
|
typealias Comparison<V> = (V) -> Int
|
||||||
|
typealias Cooldown = Pair<OfflinePlayer, Long>
|
||||||
|
|
||||||
|
fun Cooldown.isExpired(currentTime: Long) = second < currentTime
|
||||||
|
|
||||||
|
val COMPARATOR_COOLDOWN_PLAYER = Comparator<Cooldown> { a, b -> a.first.uniqueId.compareTo(b.first.uniqueId) }
|
||||||
|
val COMPARATOR_COOLDOWN_EXPIRY = Comparator<Cooldown> { a, b -> a.second.compareTo(b.second) }
|
||||||
|
|
||||||
|
val OfflinePlayer.COMPARISON_COOLDOWN: Comparison<Cooldown>
|
||||||
|
get() = { uniqueId.compareTo(it.first.uniqueId) }
|
||||||
|
|
||||||
|
val Long.COMPARISON_COOLDOWN: Comparison<Cooldown>
|
||||||
|
get() = { compareTo(it.second) }
|
||||||
|
|
||||||
val COMPARATOR_PLAYER = Comparator<OfflinePlayer> { a, b -> a.uniqueId.compareTo(b.uniqueId) }
|
val COMPARATOR_PLAYER = Comparator<OfflinePlayer> { a, b -> a.uniqueId.compareTo(b.uniqueId) }
|
||||||
val COMPARATOR_LOCATION = Comparator<Location> { a, b -> a.compareByOrder(b, { world!!.uid }, Location::getBlockX, Location::getBlockY, Location::getBlockZ) }
|
val COMPARATOR_LOCATION = Comparator<Location> { a, b -> a.compareByOrder(b, { world!!.uid }, Location::getBlockX, Location::getBlockY, Location::getBlockZ) }
|
||||||
|
@ -32,42 +32,41 @@ class MultiSortedList<E> constructor(
|
|||||||
SortedList(list, it)
|
SortedList(list, it)
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun add(element: E): Boolean {
|
override fun add(element: E) = add(element, true)
|
||||||
|
override fun add(element: E, searchForward: Boolean): Boolean {
|
||||||
extraLists.values.forEach {
|
extraLists.values.forEach {
|
||||||
it.add(element)
|
it.add(element, searchForward)
|
||||||
}
|
}
|
||||||
|
|
||||||
return super.add(element)
|
return super.add(element, searchForward)
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun add(index: Int, element: E) {
|
override fun add(index: Int, element: E) = add(index, element, true)
|
||||||
add(element)
|
override fun add(index: Int, element: E, searchForward: Boolean) {
|
||||||
|
add(element, searchForward)
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun addAll(elements: Collection<E>): Boolean {
|
override fun addAll(elements: Collection<E>) = addAll(elements, true)
|
||||||
|
override fun addAll(elements: Collection<E>, searchForward: Boolean): Boolean {
|
||||||
for (element in elements)
|
for (element in elements)
|
||||||
add(element)
|
add(element, searchForward)
|
||||||
|
|
||||||
return elements.isNotEmpty()
|
return elements.isNotEmpty()
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun addAll(index: Int, elements: Collection<E>) = addAll(elements)
|
override fun addAll(index: Int, elements: Collection<E>) = addAll(index, elements, true)
|
||||||
override fun containsAll(elements: Collection<E>): Boolean {
|
override fun addAll(index: Int, elements: Collection<E>, searchForward: Boolean) = addAll(elements, searchForward)
|
||||||
for (element in elements)
|
|
||||||
if (!contains(element))
|
|
||||||
return false
|
|
||||||
|
|
||||||
return true
|
fun contains(element: E, comparator: Comparator<E>, searchForward: Boolean = true) =
|
||||||
}
|
if (comparator == this.comparator) super.contains(element, searchForward)
|
||||||
|
else extraLists[comparator]!!.contains(element, searchForward)
|
||||||
|
|
||||||
fun contains(element: E, comparator: Comparator<E>) =
|
|
||||||
if (comparator == this.comparator) contains(element)
|
|
||||||
else extraLists[comparator]!!.contains(element)
|
|
||||||
|
|
||||||
override fun remove(element: E): Boolean {
|
override fun remove(element: E) = remove(element, true)
|
||||||
if (super.remove(element)) {
|
override fun remove(element: E, searchForward: Boolean): Boolean {
|
||||||
|
if (super.remove(element, searchForward)) {
|
||||||
extraLists.values.forEach {
|
extraLists.values.forEach {
|
||||||
it.remove(element)
|
it.remove(element, searchForward)
|
||||||
}
|
}
|
||||||
|
|
||||||
return true
|
return true
|
||||||
@ -76,31 +75,32 @@ class MultiSortedList<E> constructor(
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun removeAt(index: Int) = removeAt(index, comparator)
|
override fun removeAll(elements: Collection<E>) = removeAll(elements, true)
|
||||||
|
override fun removeAll(elements: Collection<E>, searchForward: Boolean): Boolean {
|
||||||
override fun removeAll(elements: Collection<E>): Boolean {
|
|
||||||
var result = false
|
var result = false
|
||||||
|
|
||||||
for (element in elements)
|
for (element in elements)
|
||||||
result = remove(element) || result
|
result = remove(element, searchForward) || result
|
||||||
|
|
||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
fun removeAt(index: Int, comparator: Comparator<in E>): E {
|
|
||||||
|
override fun removeAt(index: Int) = removeAt(index, comparator)
|
||||||
|
fun removeAt(index: Int, comparator: Comparator<in E>, searchForward: Boolean = true): E {
|
||||||
if (comparator == this.comparator) {
|
if (comparator == this.comparator) {
|
||||||
val result = super.removeAt(index)
|
val result = super.removeAt(index)
|
||||||
|
|
||||||
extraLists.values.forEach {
|
extraLists.values.forEach {
|
||||||
it.remove(result)
|
it.remove(result, searchForward)
|
||||||
}
|
}
|
||||||
|
|
||||||
return result
|
return result
|
||||||
} else {
|
} else {
|
||||||
val result = extraLists[comparator]!!.removeAt(index)
|
val result = extraLists[comparator]!!.removeAt(index)
|
||||||
|
|
||||||
extraLists.forEach { (comp, list) -> if (comparator != comp) list.remove(result) }
|
extraLists.forEach { (comp, list) -> if (comparator != comp) list.remove(result, searchForward) }
|
||||||
super.remove(result)
|
super.remove(result, searchForward)
|
||||||
|
|
||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
@ -115,7 +115,7 @@ class MultiSortedList<E> constructor(
|
|||||||
* @return Iterable object of all matching elements, or null if none exist
|
* @return Iterable object of all matching elements, or null if none exist
|
||||||
*/
|
*/
|
||||||
fun getAll(comparator: Comparator<in E>, comparison: (E) -> Int): Iterable<E>? {
|
fun getAll(comparator: Comparator<in E>, comparison: (E) -> Int): Iterable<E>? {
|
||||||
var index = binSearch(comparator, comparison)
|
var index = search(comparator, comparison)
|
||||||
if (index < 0) return null
|
if (index < 0) return null
|
||||||
|
|
||||||
val result = LinkedList<E>()
|
val result = LinkedList<E>()
|
||||||
@ -133,18 +133,18 @@ class MultiSortedList<E> constructor(
|
|||||||
}
|
}
|
||||||
|
|
||||||
fun findValueOrNull(comparator: Comparator<in E>, comparison: (E) -> Int): E? {
|
fun findValueOrNull(comparator: Comparator<in E>, comparison: (E) -> Int): E? {
|
||||||
val index = binSearch(comparator, comparison)
|
val index = search(comparator, comparison)
|
||||||
|
|
||||||
if (index < 0) return null
|
if (index < 0) return null
|
||||||
|
|
||||||
return get(index, comparator)
|
return get(index, comparator)
|
||||||
}
|
}
|
||||||
|
|
||||||
fun binSearch(element: E, comparator: Comparator<in E>) =
|
fun search(element: E, comparator: Comparator<in E>, searchStartToEnd: Boolean = true) =
|
||||||
if (comparator == this.comparator) binarySearch(element, comparator)
|
if (comparator == this.comparator) search(element, searchStartToEnd)
|
||||||
else extraLists[comparator]!!.binarySearch(element, comparator)
|
else extraLists[comparator]!!.search(element, searchStartToEnd)
|
||||||
|
|
||||||
fun binSearch(comparator: Comparator<in E>, comparison: (E) -> Int) =
|
fun search(comparator: Comparator<in E>, comparison: (E) -> Int) =
|
||||||
if (comparator == this.comparator) binarySearch(comparison = comparison)
|
if (comparator == this.comparator) search(comparison)
|
||||||
else extraLists[comparator]!!.binarySearch(comparison = comparison)
|
else extraLists[comparator]!!.search(comparison)
|
||||||
}
|
}
|
||||||
|
@ -27,8 +27,8 @@ class PortalManager(private val data: ConfigurationSection, private val config:
|
|||||||
private var portals = MultiSortedList(::ArrayList, COMPARATOR_PORTAL_LOCATION_OWNER, COMPARATOR_PORTAL_UID, COMPARATOR_PORTAL_OWNER_NAME, COMPARATOR_PORTAL_LINKS)
|
private var portals = MultiSortedList(::ArrayList, COMPARATOR_PORTAL_LOCATION_OWNER, COMPARATOR_PORTAL_UID, COMPARATOR_PORTAL_OWNER_NAME, COMPARATOR_PORTAL_LINKS)
|
||||||
private var invitations = MultiSortedList(::ArrayList, COMPARATOR_INVITE_RECIPIENT, COMPARATOR_INVITE_PORTAL)
|
private var invitations = MultiSortedList(::ArrayList, COMPARATOR_INVITE_RECIPIENT, COMPARATOR_INVITE_PORTAL)
|
||||||
|
|
||||||
private val cooldowns = LinkedList<Pair<OfflinePlayer, Long>>()
|
// Player-based list needs to handle random access efficiently, whereas expiry list will always be accessed sequentially
|
||||||
private val cooldownsLookup = SortedList<OfflinePlayer>(ArrayList(), COMPARATOR_PLAYER)
|
private val cooldowns = MultiSortedList(ArrayList(), ::LinkedList, COMPARATOR_COOLDOWN_PLAYER, COMPARATOR_COOLDOWN_EXPIRY)
|
||||||
|
|
||||||
private var cooldownTime = DEFAULT_COOLDOWN
|
private var cooldownTime = DEFAULT_COOLDOWN
|
||||||
|
|
||||||
@ -48,7 +48,7 @@ class PortalManager(private val data: ConfigurationSection, private val config:
|
|||||||
var msb = field.mostSignificantBits.toULong()
|
var msb = field.mostSignificantBits.toULong()
|
||||||
|
|
||||||
// Start sequential search at the resulting index if it is populated
|
// Start sequential search at the resulting index if it is populated
|
||||||
val index = portals.binSearch(COMPARATOR_PORTAL_UID) {
|
val index = portals.search(COMPARATOR_PORTAL_UID) {
|
||||||
compareValues(
|
compareValues(
|
||||||
{ msb } to { it.id.mostSignificantBits.toULong() },
|
{ msb } to { it.id.mostSignificantBits.toULong() },
|
||||||
{ lsb } to { it.id.leastSignificantBits.toULong() }
|
{ lsb } to { it.id.leastSignificantBits.toULong() }
|
||||||
@ -139,14 +139,14 @@ class PortalManager(private val data: ConfigurationSection, private val config:
|
|||||||
compareValues(
|
compareValues(
|
||||||
recipient::getUniqueId to it.recipient::getUniqueId,
|
recipient::getUniqueId to it.recipient::getUniqueId,
|
||||||
portals.get(
|
portals.get(
|
||||||
portals.binSearch(COMPARATOR_PORTAL_UID, it.portalID.COMPARISON_PORTAL_ID), COMPARATOR_PORTAL_UID
|
portals.search(COMPARATOR_PORTAL_UID, it.portalID.COMPARISON_PORTAL_ID), COMPARATOR_PORTAL_UID
|
||||||
).owner::getUniqueId to sender::getUniqueId
|
).owner::getUniqueId to sender::getUniqueId
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
fun invitePlayer(player: OfflinePlayer, portal: Portal): Boolean {
|
fun invitePlayer(player: OfflinePlayer, portal: Portal): Boolean {
|
||||||
// Player is already invited or already has a pending invitation
|
// Player is already invited or already has a pending invitation
|
||||||
if (player in portal.accessExclusions || invitations.binSearch(COMPARATOR_INVITE_RECIPIENT) {
|
if (player in portal.accessExclusions || invitations.search(COMPARATOR_INVITE_RECIPIENT) {
|
||||||
compareValues(player::getUniqueId to it.recipient::getUniqueId, portal::id to it::portalID)
|
compareValues(player::getUniqueId to it.recipient::getUniqueId, portal::id to it::portalID)
|
||||||
} >= 0)
|
} >= 0)
|
||||||
return false
|
return false
|
||||||
@ -157,7 +157,7 @@ class PortalManager(private val data: ConfigurationSection, private val config:
|
|||||||
}
|
}
|
||||||
|
|
||||||
fun cancelInvite(player: OfflinePlayer, portal: Portal): Boolean {
|
fun cancelInvite(player: OfflinePlayer, portal: Portal): Boolean {
|
||||||
val index = invitations.binSearch(COMPARATOR_INVITE_RECIPIENT) {
|
val index = invitations.search(COMPARATOR_INVITE_RECIPIENT) {
|
||||||
compareValues(
|
compareValues(
|
||||||
player::getUniqueId to it.recipient::getUniqueId,
|
player::getUniqueId to it.recipient::getUniqueId,
|
||||||
portal::id to it::portalID
|
portal::id to it::portalID
|
||||||
@ -192,7 +192,7 @@ class PortalManager(private val data: ConfigurationSection, private val config:
|
|||||||
invitations.getAll(comparator, comparison)?.forEach(invitations::remove)
|
invitations.getAll(comparator, comparison)?.forEach(invitations::remove)
|
||||||
|
|
||||||
fun removePortal(owner: OfflinePlayer, name: String): Boolean {
|
fun removePortal(owner: OfflinePlayer, name: String): Boolean {
|
||||||
val index = portals.binSearch(COMPARATOR_PORTAL_OWNER_NAME) {
|
val index = portals.search(COMPARATOR_PORTAL_OWNER_NAME) {
|
||||||
compareValues(owner::getUniqueId to it.owner::getUniqueId, { name } to it::name)
|
compareValues(owner::getUniqueId to it.owner::getUniqueId, { name } to it::name)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -209,7 +209,7 @@ class PortalManager(private val data: ConfigurationSection, private val config:
|
|||||||
}
|
}
|
||||||
|
|
||||||
fun getPortal(owner: OfflinePlayer, name: String): Portal? {
|
fun getPortal(owner: OfflinePlayer, name: String): Portal? {
|
||||||
val index = portals.binSearch(COMPARATOR_PORTAL_OWNER_NAME) {
|
val index = portals.search(COMPARATOR_PORTAL_OWNER_NAME) {
|
||||||
compareValues(owner::getUniqueId to it.owner::getUniqueId, { name } to it::name)
|
compareValues(owner::getUniqueId to it.owner::getUniqueId, { name } to it::name)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -223,24 +223,20 @@ class PortalManager(private val data: ConfigurationSection, private val config:
|
|||||||
|
|
||||||
private fun popCooldowns() {
|
private fun popCooldowns() {
|
||||||
val time = System.currentTimeMillis()
|
val time = System.currentTimeMillis()
|
||||||
while (true) {
|
while (cooldowns.isNotEmpty()) {
|
||||||
val front = cooldowns.first
|
val front = cooldowns.get(0, COMPARATOR_COOLDOWN_EXPIRY)
|
||||||
|
if (front.isExpired(time)) cooldowns.removeAt(0, COMPARATOR_COOLDOWN_EXPIRY)
|
||||||
if (front.second < time) {
|
|
||||||
cooldowns.removeFirst()
|
|
||||||
cooldownsLookup.remove(front.first)
|
|
||||||
}
|
|
||||||
else break
|
else break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun isOnCooldown(player: OfflinePlayer): Boolean {
|
private fun isOnCooldown(player: OfflinePlayer): Boolean {
|
||||||
popCooldowns()
|
popCooldowns()
|
||||||
return cooldownsLookup.contains(player)
|
return cooldowns.search(COMPARATOR_COOLDOWN_PLAYER, player.COMPARISON_COOLDOWN) >= 0
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun triggerCooldown(player: OfflinePlayer) {
|
private fun triggerCooldown(player: OfflinePlayer) {
|
||||||
cooldowns.addLast(Pair(player, System.currentTimeMillis() + cooldownTime))
|
cooldowns.add(Pair(player, System.currentTimeMillis() + cooldownTime), false)
|
||||||
}
|
}
|
||||||
|
|
||||||
@EventHandler
|
@EventHandler
|
||||||
|
@ -1,9 +1,10 @@
|
|||||||
import java.util.*
|
import java.util.*
|
||||||
import kotlin.collections.ArrayList
|
import kotlin.collections.ArrayList
|
||||||
|
import kotlin.collections.RandomAccess
|
||||||
|
|
||||||
open class SortedList<E> constructor(
|
open class SortedList<E> constructor(
|
||||||
private val underlying: MutableList<E> = ArrayList(),
|
private val underlying: MutableList<E> = ArrayList(),
|
||||||
protected val comparator: Comparator<in E>
|
val comparator: Comparator<in E>
|
||||||
): MutableList<E> by underlying {
|
): MutableList<E> by underlying {
|
||||||
companion object {
|
companion object {
|
||||||
fun <T: Comparable<T>> ofComparable(
|
fun <T: Comparable<T>> ofComparable(
|
||||||
@ -17,38 +18,46 @@ open class SortedList<E> constructor(
|
|||||||
underlying.sortWith(comparator)
|
underlying.sortWith(comparator)
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun add(element: E): Boolean {
|
open fun add(element: E, searchForward: Boolean): Boolean {
|
||||||
val index = underlying.binarySearch(element, comparator)
|
val index = search(element, searchForward)
|
||||||
|
|
||||||
underlying.add(if (index < 0) -(index + 1) else index, element)
|
underlying.add(if (index < 0) -(index + 1) else index, element)
|
||||||
|
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun add(index: Int, element: E) {
|
override fun add(element: E) = add(element, true)
|
||||||
add(element)
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun addAll(elements: Collection<E>): Boolean {
|
open fun add(index: Int, element: E, searchForward: Boolean) {
|
||||||
|
add(element, searchForward)
|
||||||
|
}
|
||||||
|
override fun add(index: Int, element: E) = add(index, element, true)
|
||||||
|
|
||||||
|
override fun addAll(elements: Collection<E>): Boolean = addAll(elements, true)
|
||||||
|
open fun addAll(elements: Collection<E>, searchForward: Boolean): Boolean {
|
||||||
for (element in elements)
|
for (element in elements)
|
||||||
add(element)
|
add(element, searchForward)
|
||||||
|
|
||||||
return elements.isNotEmpty()
|
return elements.isNotEmpty()
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun addAll(index: Int, elements: Collection<E>) = addAll(elements)
|
open fun addAll(index: Int, elements: Collection<E>, searchForward: Boolean) = addAll(elements, searchForward)
|
||||||
override fun contains(element: E) = underlying.binarySearch(element, comparator) >= 0
|
override fun addAll(index: Int, elements: Collection<E>) = addAll(index, elements, true)
|
||||||
|
open fun contains(element: E, searchForward: Boolean) = search(element, searchForward) >= 0
|
||||||
|
override fun contains(element: E) = contains(element, true)
|
||||||
|
|
||||||
override fun containsAll(elements: Collection<E>): Boolean {
|
override fun containsAll(elements: Collection<E>) = containsAll(elements, true)
|
||||||
|
open fun containsAll(elements: Collection<E>, searchForward: Boolean): Boolean {
|
||||||
for (element in elements)
|
for (element in elements)
|
||||||
if (!contains(element))
|
if (!contains(element, searchForward))
|
||||||
return false
|
return false
|
||||||
|
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun remove(element: E): Boolean {
|
override fun remove(element: E) = remove(element, true)
|
||||||
val index = underlying.binarySearch(element, comparator)
|
open fun remove(element: E, searchForward: Boolean): Boolean {
|
||||||
|
val index = search(element, searchForward)
|
||||||
|
|
||||||
if (index >= 0) {
|
if (index >= 0) {
|
||||||
underlying.removeAt(index)
|
underlying.removeAt(index)
|
||||||
@ -59,12 +68,41 @@ open class SortedList<E> constructor(
|
|||||||
}
|
}
|
||||||
|
|
||||||
override fun removeAt(index: Int) = underlying.removeAt(index)
|
override fun removeAt(index: Int) = underlying.removeAt(index)
|
||||||
override fun removeAll(elements: Collection<E>): Boolean {
|
override fun removeAll(elements: Collection<E>) = removeAll(elements, true)
|
||||||
|
open fun removeAll(elements: Collection<E>, searchForward: Boolean): Boolean {
|
||||||
var result = false
|
var result = false
|
||||||
|
|
||||||
for (element in elements)
|
for (element in elements)
|
||||||
result = remove(element) || result
|
result = remove(element, searchForward) || result
|
||||||
|
|
||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun isRandomAccess() = underlying is RandomAccess
|
||||||
|
|
||||||
|
fun search(element: E, searchStartToEnd: Boolean = true) =
|
||||||
|
if (isRandomAccess()) binarySearch(element, comparator)
|
||||||
|
else {
|
||||||
|
// Sequential search, because it's probably faster than binary search
|
||||||
|
val index = if (searchStartToEnd) indexOfFirst { comparator.compare(element, it) >= 0 }
|
||||||
|
else indexOfLast { comparator.compare(element, it) <= 0 }
|
||||||
|
|
||||||
|
if (index < 0 && searchStartToEnd) -size
|
||||||
|
else if (index < 0) -1
|
||||||
|
else if (comparator.compare(element, this[index]) == 0) index
|
||||||
|
else -(index + 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun search(comparison: Comparison<E>, searchStartToEnd: Boolean = true) =
|
||||||
|
if (isRandomAccess()) binarySearch(comparison = comparison)
|
||||||
|
else {
|
||||||
|
// Sequential search, because it's probably faster than binary search
|
||||||
|
val index = if (searchStartToEnd) indexOfFirst { comparison(it) >= 0 }
|
||||||
|
else indexOfLast { comparison(it) <= 0 }
|
||||||
|
|
||||||
|
if (index < 0 && searchStartToEnd) -size
|
||||||
|
else if (index < 0) -1
|
||||||
|
else if (comparison(this[index]) == 0) index
|
||||||
|
else -(index + 1)
|
||||||
|
}
|
||||||
}
|
}
|
Loading…
x
Reference in New Issue
Block a user