Initial commit
This commit is contained in:
commit
43c7fc093a
42
.gitignore
vendored
Normal file
42
.gitignore
vendored
Normal file
@ -0,0 +1,42 @@
|
||||
.gradle
|
||||
build/
|
||||
!gradle/wrapper/gradle-wrapper.jar
|
||||
!**/src/main/**/build/
|
||||
!**/src/test/**/build/
|
||||
|
||||
### IntelliJ IDEA ###
|
||||
.idea/modules.xml
|
||||
.idea/jarRepositories.xml
|
||||
.idea/compiler.xml
|
||||
.idea/libraries/
|
||||
*.iws
|
||||
*.iml
|
||||
*.ipr
|
||||
out/
|
||||
!**/src/main/**/out/
|
||||
!**/src/test/**/out/
|
||||
|
||||
### Eclipse ###
|
||||
.apt_generated
|
||||
.classpath
|
||||
.factorypath
|
||||
.project
|
||||
.settings
|
||||
.springBeans
|
||||
.sts4-cache
|
||||
bin/
|
||||
!**/src/main/**/bin/
|
||||
!**/src/test/**/bin/
|
||||
|
||||
### NetBeans ###
|
||||
/nbproject/private/
|
||||
/nbbuild/
|
||||
/dist/
|
||||
/nbdist/
|
||||
/.nb-gradle/
|
||||
|
||||
### VS Code ###
|
||||
.vscode/
|
||||
|
||||
### Mac OS ###
|
||||
.DS_Store
|
3
.idea/.gitignore
vendored
Normal file
3
.idea/.gitignore
vendored
Normal file
@ -0,0 +1,3 @@
|
||||
# Default ignored files
|
||||
/shelf/
|
||||
/workspace.xml
|
23
.idea/gradle.xml
Normal file
23
.idea/gradle.xml
Normal file
@ -0,0 +1,23 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="GradleSettings">
|
||||
<option name="linkedExternalProjectsSettings">
|
||||
<GradleProjectSettings>
|
||||
<option name="externalProjectPath" value="$PROJECT_DIR$" />
|
||||
<option name="gradleHome" value="" />
|
||||
<option name="modules">
|
||||
<set>
|
||||
<option value="$PROJECT_DIR$" />
|
||||
<option value="$PROJECT_DIR$/collect" />
|
||||
<option value="$PROJECT_DIR$/core" />
|
||||
<option value="$PROJECT_DIR$/guava" />
|
||||
<option value="$PROJECT_DIR$/io" />
|
||||
<option value="$PROJECT_DIR$/io-math" />
|
||||
<option value="$PROJECT_DIR$/math" />
|
||||
<option value="$PROJECT_DIR$/networking" />
|
||||
</set>
|
||||
</option>
|
||||
</GradleProjectSettings>
|
||||
</option>
|
||||
</component>
|
||||
</project>
|
6
.idea/kotlinc.xml
Normal file
6
.idea/kotlinc.xml
Normal file
@ -0,0 +1,6 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="KotlinJpsPluginSettings">
|
||||
<option name="version" value="1.8.0" />
|
||||
</component>
|
||||
</project>
|
7
.idea/misc.xml
Normal file
7
.idea/misc.xml
Normal file
@ -0,0 +1,7 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="ExternalStorageConfigurationManager" enabled="true" />
|
||||
<component name="ProjectRootManager" version="2" languageLevel="JDK_17" default="true" project-jdk-name="temurin-17" project-jdk-type="JavaSDK">
|
||||
<output url="file://$PROJECT_DIR$/out" />
|
||||
</component>
|
||||
</project>
|
6
.idea/vcs.xml
Normal file
6
.idea/vcs.xml
Normal file
@ -0,0 +1,6 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="VcsDirectoryMappings">
|
||||
<mapping directory="$PROJECT_DIR$" vcs="Git" />
|
||||
</component>
|
||||
</project>
|
46
build.gradle.kts
Normal file
46
build.gradle.kts
Normal file
@ -0,0 +1,46 @@
|
||||
|
||||
plugins {
|
||||
kotlin("jvm") version "1.8.0"
|
||||
`maven-publish`
|
||||
}
|
||||
|
||||
repositories {
|
||||
mavenCentral()
|
||||
}
|
||||
|
||||
tasks.compileKotlin {
|
||||
kotlinOptions {
|
||||
jvmTarget = "17"
|
||||
freeCompilerArgs += "-opt-in=kotlin.RequiresOptIn"
|
||||
freeCompilerArgs += "-Xjvm-default=all"
|
||||
}
|
||||
}
|
||||
|
||||
tasks.jar {
|
||||
from(project(":core").sourceSets.main.get().output)
|
||||
from(project(":io").sourceSets.main.get().output)
|
||||
from(project(":io-math").sourceSets.main.get().output)
|
||||
from(project(":math").sourceSets.main.get().output)
|
||||
from(project(":collect").sourceSets.main.get().output)
|
||||
from(project(":guava").sourceSets.main.get().output)
|
||||
from(project(":networking").sourceSets.main.get().output)
|
||||
}
|
||||
|
||||
subprojects {
|
||||
apply(plugin = "maven-publish")
|
||||
|
||||
publishing {
|
||||
repositories {
|
||||
maven {
|
||||
url = uri("sftp://maven@dbotthepony.ru:22/maven")
|
||||
|
||||
credentials {
|
||||
val mavenUser: String by project
|
||||
val mavenPassword: String by project
|
||||
username = mavenUser
|
||||
password = mavenPassword
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
40
collect/build.gradle.kts
Normal file
40
collect/build.gradle.kts
Normal file
@ -0,0 +1,40 @@
|
||||
plugins {
|
||||
kotlin("jvm")
|
||||
`maven-publish`
|
||||
}
|
||||
|
||||
val collectVersion: String by project
|
||||
val projectGroup: String by project
|
||||
|
||||
group = projectGroup
|
||||
version = collectVersion
|
||||
|
||||
repositories {
|
||||
mavenCentral()
|
||||
}
|
||||
|
||||
dependencies {
|
||||
testImplementation("org.jetbrains.kotlin:kotlin-test")
|
||||
}
|
||||
|
||||
tasks.test {
|
||||
useJUnitPlatform()
|
||||
}
|
||||
|
||||
kotlin {
|
||||
jvmToolchain(17)
|
||||
}
|
||||
|
||||
publishing {
|
||||
publications {
|
||||
create<MavenPublication>("mavenJava") {
|
||||
from(components["java"])
|
||||
|
||||
pom {
|
||||
dependencies {
|
||||
implementation(kotlin("stdlib"))
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,149 @@
|
||||
package ru.dbotthepony.kommons.collect
|
||||
|
||||
import java.lang.ref.Reference
|
||||
import java.util.*
|
||||
import java.util.stream.Stream
|
||||
import java.util.stream.StreamSupport
|
||||
|
||||
fun <T> Array<T>.stream(): Stream<T> = Arrays.stream(this)
|
||||
|
||||
@Suppress("unchecked_cast")
|
||||
fun <T> Stream<T?>.filterNotNull(): Stream<T> {
|
||||
return filter { it != null } as Stream<T>
|
||||
}
|
||||
|
||||
@Suppress("unchecked_cast")
|
||||
inline fun <reified T> Stream<*>.filterIsInstance(): Stream<T> {
|
||||
return filter { it is T } as Stream<T>
|
||||
}
|
||||
|
||||
inline fun <T> MutableList<out Reference<out T>>.forValidRefs(fn: (T) -> Unit) {
|
||||
val iterator = listIterator()
|
||||
|
||||
for (value in iterator) {
|
||||
val get = value.get()
|
||||
|
||||
if (get == null) {
|
||||
iterator.remove()
|
||||
} else {
|
||||
fn.invoke(get)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
inline fun <T> MutableList<out Reference<out T>>.forValidRefsBreak(fn: (T) -> Boolean) {
|
||||
val iterator = listIterator()
|
||||
|
||||
for (value in iterator) {
|
||||
val get = value.get()
|
||||
|
||||
if (get == null) {
|
||||
iterator.remove()
|
||||
} else {
|
||||
if (fn.invoke(get)) {
|
||||
break
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fun <T> Stream<T>.asIterable(): Iterable<T> {
|
||||
return object : Iterable<T> {
|
||||
override fun iterator(): Iterator<T> {
|
||||
return this@asIterable.iterator()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns applicable index to put [element] into [List] determined by [comparator], optionally specifying ranges as [fromIndex] and [toIndex]
|
||||
*
|
||||
* If [List] is not sorted, result of this function is undefined
|
||||
*/
|
||||
fun <E> List<E>.searchInsertionIndex(element: E, comparator: Comparator<in E>, fromIndex: Int = 0, toIndex: Int = size): Int {
|
||||
require(toIndex >= fromIndex) { "Invalid range: to $toIndex >= from $fromIndex" }
|
||||
require(fromIndex >= 0) { "Invalid from index: $fromIndex" }
|
||||
require(toIndex >= 0) { "Invalid to index: $toIndex" }
|
||||
require(fromIndex <= size) { "Invalid from index: $fromIndex (list size $size)" }
|
||||
require(toIndex <= size) { "Invalid to index: $toIndex (list size $size)" }
|
||||
|
||||
if (fromIndex == size || fromIndex == toIndex || comparator.compare(element, this[fromIndex]) <= 0) {
|
||||
return fromIndex
|
||||
}
|
||||
|
||||
// линейный поиск если границы маленькие
|
||||
if (toIndex - fromIndex <= 10) {
|
||||
for (i in fromIndex + 1 until toIndex) {
|
||||
val compare = comparator.compare(element, this[i])
|
||||
|
||||
if (compare <= 0) {
|
||||
return i
|
||||
}
|
||||
}
|
||||
|
||||
return size
|
||||
} else {
|
||||
// двоичный поиск
|
||||
var lower = fromIndex
|
||||
var upper = toIndex - 1
|
||||
|
||||
while (upper - lower >= 10) {
|
||||
val middle = (upper + lower) / 2
|
||||
val compare = comparator.compare(element, this[middle])
|
||||
|
||||
if (compare == 0) {
|
||||
return middle
|
||||
} else if (compare < 0) {
|
||||
upper = middle
|
||||
} else {
|
||||
lower = middle
|
||||
}
|
||||
}
|
||||
|
||||
return searchInsertionIndex(element, comparator, lower, upper + 1)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Inserts [element] into [MutableList] at index determined by [comparator]
|
||||
*
|
||||
* If [MutableList] is not sorted, result of this function is undefined
|
||||
*/
|
||||
fun <E> MutableList<E>.addSorted(element: E, comparator: Comparator<in E>) {
|
||||
add(searchInsertionIndex(element, comparator), element)
|
||||
}
|
||||
|
||||
private object NaturalComparator : Comparator<Comparable<Any>> {
|
||||
override fun compare(o1: Comparable<Any>, o2: Comparable<Any>): Int {
|
||||
return o1.compareTo(o2)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Inserts [element] into [MutableList] at index determined by comparing values themselves
|
||||
*
|
||||
* If [MutableList] is not sorted, result of this function is undefined
|
||||
*/
|
||||
fun <E : Comparable<E>> MutableList<E>.addSorted(element: E) {
|
||||
add(searchInsertionIndex(element, NaturalComparator as Comparator<in E>), element)
|
||||
}
|
||||
|
||||
fun <E> MutableCollection<E>.addAll(elements: Iterator<E>) {
|
||||
for (item in elements) {
|
||||
add(item)
|
||||
}
|
||||
}
|
||||
|
||||
fun <E> MutableCollection<E>.addAll(elements: Stream<out E>) {
|
||||
for (item in elements) {
|
||||
add(item)
|
||||
}
|
||||
}
|
||||
|
||||
fun <E> Iterable<E>.stream(): Stream<out E> {
|
||||
return StreamSupport.stream(this.spliterator(), false)
|
||||
}
|
||||
|
||||
fun <E> Iterator<E>.stream(): Stream<out E> {
|
||||
return StreamSupport.stream(Spliterators.spliteratorUnknownSize(this, 0), false)
|
||||
}
|
@ -0,0 +1,41 @@
|
||||
package ru.dbotthepony.kommons.collect
|
||||
|
||||
import java.util.EnumMap
|
||||
import java.util.function.BooleanSupplier
|
||||
import java.util.stream.Stream
|
||||
|
||||
class ConditionalEnumSet<T : Enum<T>>private constructor(private val backing: EnumMap<T, BooleanSupplier>, marker: Unit) : Set<T> {
|
||||
constructor(clazz: Class<T>) : this(EnumMap(clazz), Unit)
|
||||
constructor(map: Map<T, BooleanSupplier>) : this(EnumMap(map), Unit)
|
||||
|
||||
override val size: Int
|
||||
get() = backing.values.stream().filter { it.asBoolean }.count().toInt()
|
||||
|
||||
override fun contains(element: T): Boolean {
|
||||
return backing[element]?.asBoolean ?: false
|
||||
}
|
||||
|
||||
override fun containsAll(elements: Collection<T>): Boolean {
|
||||
return elements.all { contains(it) }
|
||||
}
|
||||
|
||||
override fun isEmpty(): Boolean {
|
||||
return backing.isEmpty() || backing.values.none { it.asBoolean }
|
||||
}
|
||||
|
||||
override fun stream(): Stream<T> {
|
||||
return backing.entries.stream().filter { it.value.asBoolean }.map { it.key }
|
||||
}
|
||||
|
||||
override fun iterator(): Iterator<T> {
|
||||
return backing.entries.iterator().filter { it.value.asBoolean }.map { it.key }
|
||||
}
|
||||
|
||||
fun add(value: T, condition: BooleanSupplier) {
|
||||
backing[value] = condition
|
||||
}
|
||||
|
||||
fun remove(value: T): Boolean {
|
||||
return backing.remove(value) != null
|
||||
}
|
||||
}
|
@ -0,0 +1,89 @@
|
||||
package ru.dbotthepony.kommons.collect
|
||||
|
||||
import java.util.function.BooleanSupplier
|
||||
import java.util.stream.Stream
|
||||
|
||||
/**
|
||||
* Set, with values appearing and disappearing by conditions
|
||||
*/
|
||||
class ConditionalSet<E : Any> : Set<E> {
|
||||
private class Entry<E>(val value: E, var condition: BooleanSupplier)
|
||||
private val backing = ArrayList<Entry<E>>()
|
||||
private val mapped = HashMap<E, Entry<E>>()
|
||||
|
||||
override val size: Int
|
||||
get() = backing.stream().filter { it.condition.asBoolean }.count().toInt()
|
||||
|
||||
override fun contains(element: E): Boolean {
|
||||
return mapped[element]?.condition?.asBoolean ?: false
|
||||
}
|
||||
|
||||
override fun containsAll(elements: Collection<E>): Boolean {
|
||||
return elements.all { contains(it) }
|
||||
}
|
||||
|
||||
override fun isEmpty(): Boolean {
|
||||
return mapped.isEmpty() || mapped.values.stream().noneMatch { it.condition.asBoolean }
|
||||
}
|
||||
|
||||
override fun stream(): Stream<E> {
|
||||
return backing.stream().filter { it.condition.asBoolean }.map { it.value }
|
||||
}
|
||||
|
||||
override fun iterator(): Iterator<E> {
|
||||
return backing.iterator().filter { it.condition.asBoolean }.map { it.value }
|
||||
}
|
||||
|
||||
fun actuallyContains(element: E): Boolean {
|
||||
return element in mapped
|
||||
}
|
||||
|
||||
fun add(element: E, condition: BooleanSupplier): Boolean {
|
||||
if (element in mapped) return false
|
||||
val entry = Entry(element, condition)
|
||||
mapped[element] = entry
|
||||
backing.add(entry)
|
||||
return true
|
||||
}
|
||||
|
||||
fun addFirst(element: E, condition: BooleanSupplier): Boolean {
|
||||
if (element in mapped) return false
|
||||
val entry = Entry(element, condition)
|
||||
mapped[element] = entry
|
||||
backing.add(0, entry)
|
||||
return true
|
||||
}
|
||||
|
||||
fun replace(element: E, condition: BooleanSupplier) {
|
||||
val entry = mapped[element]
|
||||
|
||||
if (entry != null) {
|
||||
entry.condition = condition
|
||||
} else {
|
||||
add(element, condition)
|
||||
}
|
||||
}
|
||||
|
||||
fun replaceFirst(element: E, condition: BooleanSupplier) {
|
||||
val entry = mapped[element]
|
||||
|
||||
if (entry != null) {
|
||||
entry.condition = condition
|
||||
backing.remove(entry)
|
||||
backing.add(0, entry)
|
||||
} else {
|
||||
addFirst(element, condition)
|
||||
}
|
||||
}
|
||||
|
||||
fun remove(element: E): Boolean {
|
||||
val entry = mapped.remove(element)
|
||||
|
||||
if (entry != null) {
|
||||
backing.remove(entry)
|
||||
return true
|
||||
}
|
||||
|
||||
return false
|
||||
}
|
||||
}
|
@ -0,0 +1,62 @@
|
||||
package ru.dbotthepony.kommons.collect
|
||||
|
||||
class ConditionalSupplierSet<T> : AbstractSet<T> {
|
||||
// method without boxing
|
||||
fun interface Condition {
|
||||
fun check(): Boolean
|
||||
}
|
||||
|
||||
private val getters: Array<Pair<Condition, () -> T>>
|
||||
|
||||
constructor(vararg getters: Pair<Condition, () -> T>) : super() {
|
||||
this.getters = Array(getters.size) { getters[it] }
|
||||
}
|
||||
|
||||
constructor(getters: List<Pair<Condition, () -> T>>) : super() {
|
||||
this.getters = Array(getters.size) { getters[it] }
|
||||
}
|
||||
|
||||
override val size: Int get() {
|
||||
var i = 0
|
||||
|
||||
for (pair in getters) {
|
||||
if (pair.first.check()) {
|
||||
i++
|
||||
}
|
||||
}
|
||||
|
||||
return i
|
||||
}
|
||||
|
||||
override fun iterator(): Iterator<T> {
|
||||
return object : Iterator<T> {
|
||||
val parent = getters.iterator()
|
||||
private var pair: Pair<Condition, () -> T>? = null
|
||||
|
||||
private fun search() {
|
||||
for (pair in parent) {
|
||||
if (pair.first.check()) {
|
||||
this.pair = pair
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
this.pair = null
|
||||
}
|
||||
|
||||
init {
|
||||
search()
|
||||
}
|
||||
|
||||
override fun hasNext(): Boolean {
|
||||
return pair != null
|
||||
}
|
||||
|
||||
override fun next(): T {
|
||||
val pair = pair ?: throw NoSuchElementException()
|
||||
search()
|
||||
return pair.second.invoke()
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,315 @@
|
||||
package ru.dbotthepony.kommons.collect
|
||||
|
||||
abstract class ProxiedMap<K, V>(protected val backingMap: MutableMap<K, V> = HashMap()) : MutableMap<K, V> {
|
||||
protected abstract fun onClear()
|
||||
protected abstract fun onValueAdded(key: K, value: V)
|
||||
protected abstract fun onValueRemoved(key: K, value: V)
|
||||
|
||||
final override val size: Int
|
||||
get() = backingMap.size
|
||||
|
||||
final override fun containsKey(key: K): Boolean {
|
||||
return backingMap.containsKey(key)
|
||||
}
|
||||
|
||||
final override fun containsValue(value: V): Boolean {
|
||||
return backingMap.containsValue(value)
|
||||
}
|
||||
|
||||
final override fun get(key: K): V? {
|
||||
return backingMap[key]
|
||||
}
|
||||
|
||||
final override fun isEmpty(): Boolean {
|
||||
return backingMap.isEmpty()
|
||||
}
|
||||
|
||||
final override val entries: MutableSet<MutableMap.MutableEntry<K, V>> by lazy {
|
||||
object : MutableSet<MutableMap.MutableEntry<K, V>> {
|
||||
override fun add(element: MutableMap.MutableEntry<K, V>): Boolean {
|
||||
this@ProxiedMap[element.key] = element.value
|
||||
return true
|
||||
}
|
||||
|
||||
override fun addAll(elements: Collection<MutableMap.MutableEntry<K, V>>): Boolean {
|
||||
for (value in elements) {
|
||||
add(value)
|
||||
}
|
||||
|
||||
return true
|
||||
}
|
||||
|
||||
override fun clear() {
|
||||
this@ProxiedMap.clear()
|
||||
}
|
||||
|
||||
private val setParent = this@ProxiedMap.backingMap.entries
|
||||
|
||||
override fun iterator(): MutableIterator<MutableMap.MutableEntry<K, V>> {
|
||||
return object : MutableIterator<MutableMap.MutableEntry<K, V>> {
|
||||
private val parent = setParent.iterator()
|
||||
private var last: MutableMap.MutableEntry<K, V>? = null
|
||||
|
||||
override fun hasNext(): Boolean {
|
||||
return parent.hasNext()
|
||||
}
|
||||
|
||||
override fun next(): MutableMap.MutableEntry<K, V> {
|
||||
return parent.next().also { last = it }
|
||||
}
|
||||
|
||||
override fun remove() {
|
||||
val last = last ?: throw IllegalStateException("Never called next()")
|
||||
parent.remove()
|
||||
onValueRemoved(last.key, last.value)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
override fun remove(element: MutableMap.MutableEntry<K, V>): Boolean {
|
||||
if (setParent.remove(element)) {
|
||||
onValueRemoved(element.key, element.value)
|
||||
return true
|
||||
}
|
||||
|
||||
return false
|
||||
}
|
||||
|
||||
override fun removeAll(elements: Collection<MutableMap.MutableEntry<K, V>>): Boolean {
|
||||
var any = false
|
||||
|
||||
for (element in elements) {
|
||||
any = remove(element) || any
|
||||
}
|
||||
|
||||
return any
|
||||
}
|
||||
|
||||
override fun retainAll(elements: Collection<MutableMap.MutableEntry<K, V>>): Boolean {
|
||||
val iterator = this.iterator()
|
||||
var any = false
|
||||
|
||||
for (value in iterator) {
|
||||
if (value !in elements) {
|
||||
iterator.remove()
|
||||
any = true
|
||||
}
|
||||
}
|
||||
|
||||
return any
|
||||
}
|
||||
|
||||
override val size: Int
|
||||
get() = setParent.size
|
||||
|
||||
override fun contains(element: MutableMap.MutableEntry<K, V>): Boolean {
|
||||
return setParent.contains(element)
|
||||
}
|
||||
|
||||
override fun containsAll(elements: Collection<MutableMap.MutableEntry<K, V>>): Boolean {
|
||||
return setParent.containsAll(elements)
|
||||
}
|
||||
|
||||
override fun isEmpty(): Boolean {
|
||||
return setParent.isEmpty()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
final override val keys: MutableSet<K> by lazy {
|
||||
object : MutableSet<K> {
|
||||
val parent = this@ProxiedMap.backingMap.keys
|
||||
|
||||
override fun add(element: K): Boolean {
|
||||
throw UnsupportedOperationException()
|
||||
}
|
||||
|
||||
override fun addAll(elements: Collection<K>): Boolean {
|
||||
throw UnsupportedOperationException()
|
||||
}
|
||||
|
||||
override fun clear() {
|
||||
this@ProxiedMap.clear()
|
||||
}
|
||||
|
||||
override fun iterator(): MutableIterator<K> {
|
||||
return object : MutableIterator<K> {
|
||||
private val parentIterator = parent.iterator()
|
||||
private var last: K? = null
|
||||
|
||||
override fun hasNext(): Boolean {
|
||||
return parentIterator.hasNext()
|
||||
}
|
||||
|
||||
override fun next(): K {
|
||||
return parentIterator.next().also { last = it }
|
||||
}
|
||||
|
||||
override fun remove() {
|
||||
val last = last ?: throw IllegalStateException("Never called next()")
|
||||
val value = this@ProxiedMap[last] ?: throw ConcurrentModificationException()
|
||||
parentIterator.remove()
|
||||
onValueRemoved(last, value)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
override fun remove(element: K): Boolean {
|
||||
return this@ProxiedMap.remove(element) != null
|
||||
}
|
||||
|
||||
override fun removeAll(elements: Collection<K>): Boolean {
|
||||
var changes = false
|
||||
|
||||
for (element in elements) {
|
||||
changes = remove(element) || changes
|
||||
}
|
||||
|
||||
return changes
|
||||
}
|
||||
|
||||
override fun retainAll(elements: Collection<K>): Boolean {
|
||||
val iterator = this.iterator()
|
||||
var any = false
|
||||
|
||||
for (element in iterator) {
|
||||
if (element !in elements) {
|
||||
iterator.remove()
|
||||
any = true
|
||||
}
|
||||
}
|
||||
|
||||
return any
|
||||
}
|
||||
|
||||
override val size: Int
|
||||
get() = parent.size
|
||||
|
||||
override fun contains(element: K): Boolean {
|
||||
return parent.contains(element)
|
||||
}
|
||||
|
||||
override fun containsAll(elements: Collection<K>): Boolean {
|
||||
return parent.containsAll(elements)
|
||||
}
|
||||
|
||||
override fun isEmpty(): Boolean {
|
||||
return parent.isEmpty()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
final override val values: MutableCollection<V> by lazy {
|
||||
object : MutableCollection<V> {
|
||||
private val parent = this@ProxiedMap.backingMap.values
|
||||
|
||||
override val size: Int
|
||||
get() = parent.size
|
||||
|
||||
override fun contains(element: V): Boolean {
|
||||
return parent.contains(element)
|
||||
}
|
||||
|
||||
override fun containsAll(elements: Collection<V>): Boolean {
|
||||
return parent.containsAll(elements)
|
||||
}
|
||||
|
||||
override fun isEmpty(): Boolean {
|
||||
return parent.isEmpty()
|
||||
}
|
||||
|
||||
override fun add(element: V): Boolean {
|
||||
throw UnsupportedOperationException()
|
||||
}
|
||||
|
||||
override fun addAll(elements: Collection<V>): Boolean {
|
||||
throw UnsupportedOperationException()
|
||||
}
|
||||
|
||||
override fun clear() {
|
||||
this@ProxiedMap.clear()
|
||||
}
|
||||
|
||||
override fun iterator(): MutableIterator<V> {
|
||||
return object : MutableIterator<V> {
|
||||
private val parentIterator = parent.iterator()
|
||||
|
||||
override fun hasNext(): Boolean {
|
||||
return parentIterator.hasNext()
|
||||
}
|
||||
|
||||
override fun next(): V {
|
||||
return parentIterator.next()
|
||||
}
|
||||
|
||||
override fun remove() {
|
||||
parentIterator.remove()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
override fun remove(element: V): Boolean {
|
||||
val indexOf = this@ProxiedMap.backingMap.firstNotNullOfOrNull { if (it.value == element) it.key else null } ?: return false
|
||||
this@ProxiedMap.remove(indexOf)
|
||||
return true
|
||||
}
|
||||
|
||||
override fun removeAll(elements: Collection<V>): Boolean {
|
||||
var changes = false
|
||||
|
||||
for (element in elements) {
|
||||
changes = remove(element) || changes
|
||||
}
|
||||
|
||||
return changes
|
||||
}
|
||||
|
||||
override fun retainAll(elements: Collection<V>): Boolean {
|
||||
val iterator = this.iterator()
|
||||
var changes = false
|
||||
|
||||
for (element in iterator) {
|
||||
if (element !in elements) {
|
||||
iterator.remove()
|
||||
changes = true
|
||||
}
|
||||
}
|
||||
|
||||
return changes
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
override fun clear() {
|
||||
onClear()
|
||||
return backingMap.clear()
|
||||
}
|
||||
|
||||
override fun put(key: K, value: V): V? {
|
||||
val existing = backingMap.put(key, value)
|
||||
|
||||
if (existing == value) {
|
||||
return existing
|
||||
}
|
||||
|
||||
onValueAdded(key, value)
|
||||
return existing
|
||||
}
|
||||
|
||||
final override fun putAll(from: Map<out K, V>) {
|
||||
for ((k, v) in from) {
|
||||
this[k] = v
|
||||
}
|
||||
}
|
||||
|
||||
override fun remove(key: K): V? {
|
||||
val removed = backingMap.remove(key)
|
||||
|
||||
if (removed != null) {
|
||||
onValueRemoved(key, removed)
|
||||
}
|
||||
|
||||
return removed
|
||||
}
|
||||
}
|
||||
|
@ -0,0 +1,421 @@
|
||||
package ru.dbotthepony.kommons.collect
|
||||
|
||||
import java.util.*
|
||||
import java.util.function.Predicate
|
||||
import java.util.stream.Collector
|
||||
import java.util.stream.Stream
|
||||
import java.util.stream.StreamSupport
|
||||
|
||||
// Purpose of Stream API over Iterators is that it is simple enough for JIT to inline most of it,
|
||||
// unlike actual Streams.
|
||||
|
||||
// We lose only one (actual) element of Stream API here: ability to parallelize work,
|
||||
// except it doesn't (properly) work in Minecraft (Forge) either, due to classloader bug:
|
||||
// https://github.com/MinecraftForge/EventBus/issues/44
|
||||
|
||||
// Aside parallel work, unimplemented Stream API elements can be easily implemented when required
|
||||
|
||||
private class FilteringIterator<T>(private val parent: Iterator<T>, private val predicate: Predicate<in T>, private var value: T) : MutableIterator<T> {
|
||||
private var hasValue = true
|
||||
private var returned = false
|
||||
|
||||
override fun hasNext(): Boolean {
|
||||
return hasValue
|
||||
}
|
||||
|
||||
override fun next(): T {
|
||||
if (!hasValue) throw NoSuchElementException()
|
||||
hasValue = false
|
||||
returned = true
|
||||
|
||||
val value = this.value
|
||||
|
||||
while (parent.hasNext()) {
|
||||
val next = parent.next()
|
||||
|
||||
if (predicate.test(next)) {
|
||||
hasValue = true
|
||||
this.value = next
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
return value
|
||||
}
|
||||
|
||||
override fun remove() {
|
||||
if (!returned) throw NoSuchElementException()
|
||||
returned = false
|
||||
(parent as MutableIterator<T>).remove()
|
||||
}
|
||||
}
|
||||
|
||||
private class MappingIterator<T, R>(private val parent: Iterator<T>, private val mapper: (T) -> R) : MutableIterator<R> {
|
||||
override fun hasNext(): Boolean {
|
||||
return parent.hasNext()
|
||||
}
|
||||
|
||||
override fun next(): R {
|
||||
return mapper.invoke(parent.next())
|
||||
}
|
||||
|
||||
override fun remove() {
|
||||
(parent as MutableIterator<T>).remove()
|
||||
}
|
||||
}
|
||||
|
||||
private class FlatMappingIterator<T, R>(private val parent: Iterator<T>, private val mapper: (T) -> Iterator<R>) : MutableIterator<R> {
|
||||
private var current: Iterator<R> = mapper.invoke(parent.next())
|
||||
private var last: Iterator<R>? = null
|
||||
|
||||
init {
|
||||
while (!current.hasNext() && parent.hasNext()) {
|
||||
current = mapper.invoke(parent.next())
|
||||
}
|
||||
}
|
||||
|
||||
override fun hasNext(): Boolean {
|
||||
return current.hasNext()
|
||||
}
|
||||
|
||||
override fun next(): R {
|
||||
if (!current.hasNext())
|
||||
throw NoSuchElementException()
|
||||
|
||||
val v = current.next()
|
||||
last = current
|
||||
|
||||
while (!current.hasNext() && parent.hasNext()) {
|
||||
current = mapper.invoke(parent.next())
|
||||
}
|
||||
|
||||
return v
|
||||
}
|
||||
|
||||
override fun remove() {
|
||||
(last as MutableIterator<R>? ?: throw NoSuchElementException()).remove()
|
||||
last = null
|
||||
}
|
||||
}
|
||||
|
||||
private class LimitingIterator<T>(private val parent: Iterator<T>, private val limit: Long) : Iterator<T> {
|
||||
init {
|
||||
require(limit > 0) { "Invalid limit $limit" }
|
||||
}
|
||||
|
||||
private var found = 0L
|
||||
|
||||
override fun hasNext(): Boolean {
|
||||
return found < limit && parent.hasNext()
|
||||
}
|
||||
|
||||
override fun next(): T {
|
||||
if (found >= limit)
|
||||
throw NoSuchElementException()
|
||||
|
||||
return parent.next()
|
||||
}
|
||||
}
|
||||
|
||||
private class SkippingIterator<T>(private val parent: Iterator<T>, skip: Long) : MutableIterator<T> {
|
||||
init {
|
||||
require(skip >= 0) { "Invalid skip amount $skip" }
|
||||
}
|
||||
|
||||
private var found = skip
|
||||
private var returned = false
|
||||
|
||||
override fun hasNext(): Boolean {
|
||||
while (parent.hasNext() && found > 0L) {
|
||||
found--
|
||||
parent.next()
|
||||
}
|
||||
|
||||
return parent.hasNext()
|
||||
}
|
||||
|
||||
override fun next(): T {
|
||||
if (!hasNext())
|
||||
throw NoSuchElementException()
|
||||
|
||||
val v = parent.next()
|
||||
returned = true
|
||||
return v
|
||||
}
|
||||
|
||||
override fun remove() {
|
||||
if (!returned) {
|
||||
throw NoSuchElementException()
|
||||
}
|
||||
|
||||
returned = false
|
||||
(parent as MutableIterator<T>).remove()
|
||||
}
|
||||
}
|
||||
|
||||
fun <T> concatIterators(): MutableIterator<T> {
|
||||
return listOf<T>().iterator() as MutableIterator<T>
|
||||
}
|
||||
|
||||
fun <T> concatIterators(a: Iterator<T>): MutableIterator<T> {
|
||||
return a as MutableIterator<T>
|
||||
}
|
||||
|
||||
fun <T> concatIterators(iterators: Iterable<Iterator<T>>): MutableIterator<T> {
|
||||
return iterators.iterator().flatMap { it }
|
||||
}
|
||||
|
||||
fun <T> concatIterators(vararg iterators: Iterator<T>): MutableIterator<T> {
|
||||
return iterators.iterator().flatMap { it }
|
||||
}
|
||||
|
||||
/**
|
||||
* Filters elements of [this] iterator
|
||||
*
|
||||
* Resulting [Iterator] is [MutableIterator] if [this] is
|
||||
*/
|
||||
fun <T> Iterator<T>.filter(condition: Predicate<in T>): MutableIterator<T> {
|
||||
while (hasNext()) {
|
||||
val v = next()
|
||||
|
||||
if (condition.test(v)) {
|
||||
return FilteringIterator(this, condition, v)
|
||||
}
|
||||
}
|
||||
|
||||
return emptyIterator()
|
||||
}
|
||||
|
||||
/**
|
||||
* Maps elements of [this] iterator from values of [T] to [R] using function [mapper]
|
||||
*
|
||||
* Resulting [Iterator] is [MutableIterator] if [this] is
|
||||
*/
|
||||
fun <T, R> Iterator<T>.map(mapper: (T) -> R): MutableIterator<R> {
|
||||
if (!hasNext()) {
|
||||
return emptyIterator()
|
||||
}
|
||||
|
||||
return MappingIterator(this, mapper)
|
||||
}
|
||||
|
||||
/**
|
||||
* Maps elements of [this] iterator from type [T] to other iterators of type [R] using function [mapper]
|
||||
*
|
||||
* Resulting [Iterator] is [MutableIterator] if [this] is
|
||||
*/
|
||||
fun <T, R> Iterator<T>.flatMap(mapper: (T) -> Iterator<R>): MutableIterator<R> {
|
||||
if (!hasNext()) {
|
||||
return emptyIterator()
|
||||
}
|
||||
|
||||
return FlatMappingIterator(this, mapper)
|
||||
}
|
||||
|
||||
inline fun <T> Iterator<T>.reduce(identity: T, reducer: (T, T) -> T): T {
|
||||
var result = identity
|
||||
while (hasNext()) result = reducer.invoke(result, next())
|
||||
return result
|
||||
}
|
||||
|
||||
fun <T> Iterator<T?>.filterNotNull(): MutableIterator<T> = filter { it != null } as MutableIterator<T>
|
||||
|
||||
inline fun <reified T> Iterator<*>.filterIsInstance(): MutableIterator<T> = filter { it is T } as MutableIterator<T>
|
||||
|
||||
fun <T> Iterator<T>.any(predicate: Predicate<in T>): Boolean {
|
||||
while (hasNext())
|
||||
if (predicate.test(next()))
|
||||
return true
|
||||
|
||||
return false
|
||||
}
|
||||
|
||||
inline fun <T> Iterator<T>.any(predicate: (T) -> Boolean): Boolean {
|
||||
while (hasNext())
|
||||
if (predicate.invoke(next()))
|
||||
return true
|
||||
|
||||
return false
|
||||
}
|
||||
|
||||
fun <T> Iterator<T>.all(predicate: Predicate<in T>): Boolean {
|
||||
while (hasNext())
|
||||
if (!predicate.test(next()))
|
||||
return false
|
||||
|
||||
return true
|
||||
}
|
||||
|
||||
inline fun <T> Iterator<T>.all(predicate: (T) -> Boolean): Boolean {
|
||||
while (hasNext())
|
||||
if (!predicate.invoke(next()))
|
||||
return false
|
||||
|
||||
return true
|
||||
}
|
||||
|
||||
fun <T> Iterator<T>.none(predicate: Predicate<in T>): Boolean {
|
||||
while (hasNext())
|
||||
if (predicate.test(next()))
|
||||
return false
|
||||
|
||||
return true
|
||||
}
|
||||
|
||||
inline fun <T> Iterator<T>.none(predicate: (T) -> Boolean): Boolean {
|
||||
while (hasNext())
|
||||
if (predicate.invoke(next()))
|
||||
return false
|
||||
|
||||
return true
|
||||
}
|
||||
|
||||
fun <T, A, R> Iterator<T>.collect(collector: Collector<T, A, R>): R {
|
||||
val accumulator = collector.accumulator()
|
||||
val instance = collector.supplier().get()
|
||||
|
||||
for (value in this) {
|
||||
accumulator.accept(instance, value)
|
||||
}
|
||||
|
||||
return collector.finisher().apply(instance)
|
||||
}
|
||||
|
||||
fun <T> Iterator<T>.toList(expectedSize: Int = 16): MutableList<T> {
|
||||
val result = ArrayList<T>(expectedSize)
|
||||
result.addAll(this)
|
||||
return result
|
||||
}
|
||||
|
||||
fun <T> Iterator<T>.toImmutableList(expectedSize: Int = 16): List<T> {
|
||||
if (!hasNext())
|
||||
return emptyList()
|
||||
|
||||
return toList(expectedSize)
|
||||
}
|
||||
|
||||
fun <T : Any> Iterator<T>.find(): Optional<T> {
|
||||
if (hasNext()) {
|
||||
return Optional.of(next())
|
||||
}
|
||||
|
||||
return Optional.empty<T>()
|
||||
}
|
||||
|
||||
fun <T> Iterator<T>.limit(limit: Long): Iterator<T> = LimitingIterator(this, limit)
|
||||
fun <T> Iterator<T>.skip(skip: Long): Iterator<T> = if (skip == 0L) this else SkippingIterator(this, skip)
|
||||
|
||||
inline fun <T> Iterator<T>.forEach(action: (T) -> Unit) {
|
||||
for (value in this) {
|
||||
action.invoke(value)
|
||||
}
|
||||
}
|
||||
|
||||
fun <T : Any> Iterator<T>.min(comparator: Comparator<T>): Optional<T> {
|
||||
if (!hasNext()) {
|
||||
return Optional.empty<T>()
|
||||
}
|
||||
|
||||
var min = next()
|
||||
|
||||
for (value in this) {
|
||||
if (comparator.compare(min, value) > 0) {
|
||||
min = value
|
||||
}
|
||||
}
|
||||
|
||||
return Optional.of(min)
|
||||
}
|
||||
|
||||
fun <T : Any> Iterator<T>.max(comparator: Comparator<T>): Optional<T> {
|
||||
if (!hasNext()) {
|
||||
return Optional.empty<T>()
|
||||
}
|
||||
|
||||
var max = next()
|
||||
|
||||
for (value in this) {
|
||||
if (comparator.compare(max, value) < 0) {
|
||||
max = value
|
||||
}
|
||||
}
|
||||
|
||||
return Optional.of(max)
|
||||
}
|
||||
|
||||
fun <T> Iterator<T>.peek(peeker: (T) -> Unit): MutableIterator<T> {
|
||||
return object : MutableIterator<T> {
|
||||
override fun hasNext(): Boolean {
|
||||
return this@peek.hasNext()
|
||||
}
|
||||
|
||||
override fun next(): T {
|
||||
return this@peek.next().also(peeker)
|
||||
}
|
||||
|
||||
override fun remove() {
|
||||
(this@peek as MutableIterator<T>).remove()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fun <T> Iterator<T>.maybe(): T? {
|
||||
return if (hasNext())
|
||||
next()
|
||||
else
|
||||
null
|
||||
}
|
||||
|
||||
fun <T> emptyIterator(): MutableIterator<T> {
|
||||
return emptyList<T>().iterator() as MutableIterator<T>
|
||||
}
|
||||
|
||||
fun <T> iteratorOf(value: T): Iterator<T> {
|
||||
return object : Iterator<T> {
|
||||
private var next = true
|
||||
private var v: T? = value
|
||||
|
||||
override fun hasNext(): Boolean {
|
||||
return next
|
||||
}
|
||||
|
||||
override fun next(): T {
|
||||
if (!next)
|
||||
throw NoSuchElementException()
|
||||
|
||||
val v = v
|
||||
this.v = null
|
||||
return v as T
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fun <T> iteratorOf(vararg value: T): Iterator<T> {
|
||||
return value.iterator()
|
||||
}
|
||||
|
||||
fun <T> Iterator<T>.toStream(): Stream<T> {
|
||||
return StreamSupport.stream(Spliterators.spliteratorUnknownSize(this, 0), false)
|
||||
}
|
||||
|
||||
fun <T> Iterator<T>.allEqual(): Boolean {
|
||||
if (hasNext()) {
|
||||
val v = next()
|
||||
|
||||
while (hasNext()) {
|
||||
if (v != next()) {
|
||||
return false
|
||||
}
|
||||
}
|
||||
|
||||
return true
|
||||
}
|
||||
|
||||
return false
|
||||
}
|
||||
|
||||
fun <T> Iterator<T>.count(): Long {
|
||||
var count = 0L
|
||||
while (hasNext()) count++
|
||||
return count
|
||||
}
|
@ -0,0 +1,43 @@
|
||||
package ru.dbotthepony.kommons.collect
|
||||
|
||||
import java.util.concurrent.Future
|
||||
import java.util.function.Supplier
|
||||
import java.util.stream.Stream
|
||||
|
||||
class SupplierList<T> : AbstractList<T> {
|
||||
private val getters: Array<Supplier<T>>
|
||||
|
||||
constructor(getters: Collection<Supplier<T>>) : super() {
|
||||
val iterator = getters.iterator()
|
||||
this.getters = Array(getters.size) { iterator.next() }
|
||||
}
|
||||
|
||||
constructor(getters: Stream<Supplier<T>>) : super() {
|
||||
this.getters = getters.toArray { arrayOfNulls<Supplier<T>>(it) }
|
||||
}
|
||||
|
||||
constructor(vararg getters: Supplier<T>) : super() {
|
||||
this.getters = Array(getters.size) { getters[it] }
|
||||
}
|
||||
|
||||
constructor(size: Int, provider: (Int) -> Supplier<T>) {
|
||||
this.getters = Array(size, provider)
|
||||
}
|
||||
|
||||
override val size: Int
|
||||
get() = getters.size
|
||||
|
||||
override fun get(index: Int): T {
|
||||
return getters[index].get()
|
||||
}
|
||||
|
||||
companion object {
|
||||
fun <T> ofLazy(vararg values: Lazy<T>): SupplierList<T> {
|
||||
return SupplierList(values.stream().map { Supplier { it.value } })
|
||||
}
|
||||
|
||||
fun <T> ofFuture(vararg values: Future<T>): SupplierList<T> {
|
||||
return SupplierList(values.stream().map { Supplier { it.get() } })
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,55 @@
|
||||
package ru.dbotthepony.kommons.collect
|
||||
|
||||
import java.util.Collections
|
||||
import java.util.function.Supplier
|
||||
import java.util.stream.Stream
|
||||
|
||||
class SupplierMap<K, T>(values: Stream<Pair<K, Supplier<T>>>) : Map<K, T> {
|
||||
private val backing = HashMap<K, Supplier<T>>()
|
||||
override val entries: Set<Map.Entry<K, T>>
|
||||
|
||||
override val keys: Set<K>
|
||||
get() = backing.keys
|
||||
|
||||
override val size: Int
|
||||
get() = entries.size
|
||||
|
||||
override val values: Collection<T>
|
||||
|
||||
init {
|
||||
values.forEach {
|
||||
backing[it.first] = it.second
|
||||
}
|
||||
|
||||
entries = Collections.unmodifiableSet(HashSet(backing.entries.map { Entry(it.key, it.value) }))
|
||||
this.values = SupplierList(backing.values)
|
||||
}
|
||||
|
||||
override fun containsKey(key: K): Boolean {
|
||||
return key in keys
|
||||
}
|
||||
|
||||
override fun containsValue(value: T): Boolean {
|
||||
return value in values
|
||||
}
|
||||
|
||||
override fun get(key: K): T? {
|
||||
return backing[key]?.get()
|
||||
}
|
||||
|
||||
override fun isEmpty(): Boolean {
|
||||
return entries.isEmpty()
|
||||
}
|
||||
|
||||
constructor(vararg mValues: Pair<K, Supplier<T>>) : this((mValues as Array<Pair<K, Supplier<T>>>).stream())
|
||||
constructor(mValues: Collection<Pair<K, Supplier<T>>>) : this(mValues.stream())
|
||||
constructor(mValues: Map<K, Supplier<T>>) : this(mValues.entries.stream().map { it.key to it.value })
|
||||
|
||||
private inner class Entry(
|
||||
override val key: K,
|
||||
val getter: Supplier<T>
|
||||
) : Map.Entry<K, T> {
|
||||
override val value: T
|
||||
get() = getter.get()
|
||||
}
|
||||
}
|
41
core/build.gradle.kts
Normal file
41
core/build.gradle.kts
Normal file
@ -0,0 +1,41 @@
|
||||
|
||||
plugins {
|
||||
kotlin("jvm")
|
||||
`maven-publish`
|
||||
}
|
||||
|
||||
val coreVersion: String by project
|
||||
val projectGroup: String by project
|
||||
|
||||
group = projectGroup
|
||||
version = coreVersion
|
||||
|
||||
repositories {
|
||||
mavenCentral()
|
||||
}
|
||||
|
||||
dependencies {
|
||||
testImplementation("org.jetbrains.kotlin:kotlin-test")
|
||||
}
|
||||
|
||||
tasks.test {
|
||||
useJUnitPlatform()
|
||||
}
|
||||
|
||||
kotlin {
|
||||
jvmToolchain(17)
|
||||
}
|
||||
|
||||
publishing {
|
||||
publications {
|
||||
create<MavenPublication>("mavenJava") {
|
||||
from(components["java"])
|
||||
|
||||
pom {
|
||||
dependencies {
|
||||
implementation(kotlin("stdlib"))
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
149
core/src/main/kotlin/ru/dbotthepony/kommons/core/GetterSetter.kt
Normal file
149
core/src/main/kotlin/ru/dbotthepony/kommons/core/GetterSetter.kt
Normal file
@ -0,0 +1,149 @@
|
||||
package ru.dbotthepony.kommons.core
|
||||
|
||||
import ru.dbotthepony.kommons.event.ISubscriptable
|
||||
import java.util.function.Consumer
|
||||
import java.util.function.Supplier
|
||||
import kotlin.properties.ReadWriteProperty
|
||||
import kotlin.reflect.KMutableProperty0
|
||||
import kotlin.reflect.KProperty
|
||||
|
||||
inline var <V> GetterSetter<V>.value: V
|
||||
get() = get()
|
||||
set(value) { accept(value) }
|
||||
|
||||
interface GetterSetter<V> : Supplier<V>, Consumer<V>, ReadWriteProperty<Any?, V> {
|
||||
override fun setValue(thisRef: Any?, property: KProperty<*>, value: V) {
|
||||
accept(value)
|
||||
}
|
||||
|
||||
override fun getValue(thisRef: Any?, property: KProperty<*>): V {
|
||||
return get()
|
||||
}
|
||||
|
||||
operator fun invoke(): V {
|
||||
return get()
|
||||
}
|
||||
|
||||
operator fun invoke(value: V) {
|
||||
accept(value)
|
||||
}
|
||||
|
||||
fun asGetterOnly(): GetterSetter<V> {
|
||||
val self = this
|
||||
|
||||
return object : GetterSetter<V> {
|
||||
override fun get(): V {
|
||||
return self.get()
|
||||
}
|
||||
|
||||
override fun accept(t: V) {
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fun watch(watch: (old: V, new: V) -> Unit): GetterSetter<V> {
|
||||
val self = this
|
||||
|
||||
return object : GetterSetter<V> by self {
|
||||
override fun accept(t: V) {
|
||||
val old = get()
|
||||
self.accept(t)
|
||||
watch.invoke(old, t)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
companion object {
|
||||
fun <V> of(getter: Supplier<V>, setter: Consumer<V>): GetterSetter<V> {
|
||||
return object : GetterSetter<V> {
|
||||
override fun get(): V {
|
||||
return getter.get()
|
||||
}
|
||||
|
||||
override fun accept(t: V) {
|
||||
setter.accept(t)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fun <V> of(getter: () -> V, setter: (V) -> Unit): GetterSetter<V> {
|
||||
return object : GetterSetter<V> {
|
||||
override fun get(): V {
|
||||
return getter.invoke()
|
||||
}
|
||||
|
||||
override fun accept(t: V) {
|
||||
setter.invoke(t)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fun <V> of(property: KMutableProperty0<V>): GetterSetter<V> {
|
||||
return object : GetterSetter<V> {
|
||||
override fun get(): V {
|
||||
return property.get()
|
||||
}
|
||||
|
||||
override fun accept(t: V) {
|
||||
property.set(t)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fun <V> box(value: V): SentientGetterSetter<V> {
|
||||
return object : SentientGetterSetter<V> {
|
||||
private val subs = ISubscriptable.Impl<V>()
|
||||
|
||||
override fun addListener(listener: Consumer<V>): ISubscriptable.L {
|
||||
return subs.addListener(listener)
|
||||
}
|
||||
|
||||
private var value = value
|
||||
|
||||
override fun get(): V {
|
||||
return value
|
||||
}
|
||||
|
||||
override fun accept(t: V) {
|
||||
this.value = t
|
||||
subs.accept(t)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
interface SentientGetterSetter<V> : GetterSetter<V>, ISubscriptable<V> {
|
||||
override fun watch(watch: (old: V, new: V) -> Unit): SentientGetterSetter<V> {
|
||||
val self = this
|
||||
|
||||
return object : SentientGetterSetter<V> by self {
|
||||
override fun accept(t: V) {
|
||||
val old = get()
|
||||
self.accept(t)
|
||||
watch.invoke(old, t)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
operator fun <T> Supplier<T>.getValue(thisRef: Any?, property: KProperty<*>): T {
|
||||
return get()
|
||||
}
|
||||
|
||||
operator fun <T> Consumer<T>.setValue(thisRef: Any?, property: KProperty<*>, value: T) {
|
||||
accept(value)
|
||||
}
|
||||
|
||||
fun <V> KMutableProperty0<V>.asGetterSetter(watch: ((old: V, new: V) -> Unit)? = null): GetterSetter<V> {
|
||||
return GetterSetter.of(this).let {
|
||||
if (watch != null) {
|
||||
it.watch(watch)
|
||||
} else {
|
||||
it
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fun <V> KMutableProperty0<V>.asGetterOnly() = GetterSetter.of(Supplier { this.get() }, Consumer { /* do nothing */ })
|
152
core/src/main/kotlin/ru/dbotthepony/kommons/core/KOptional.kt
Normal file
152
core/src/main/kotlin/ru/dbotthepony/kommons/core/KOptional.kt
Normal file
@ -0,0 +1,152 @@
|
||||
package ru.dbotthepony.kommons.core
|
||||
|
||||
import java.util.function.Supplier
|
||||
|
||||
fun <T> KOptional(value: T) = KOptional.of(value)
|
||||
fun KOptional(value: Boolean) = KOptional.of(value)
|
||||
fun KOptional(value: Boolean?) = KOptional.of(value)
|
||||
|
||||
/**
|
||||
* [java.util.Optional] supporting nulls
|
||||
*
|
||||
* This is done for structures, where value can be absent,
|
||||
* or can be present, including literal "null" as possible present value,
|
||||
* in more elegant solution than handling nullable Optionals
|
||||
*/
|
||||
class KOptional<T> private constructor(private val _value: T, val isPresent: Boolean) : Supplier<T> {
|
||||
/**
|
||||
* @throws NoSuchElementException
|
||||
*/
|
||||
val value: T get() {
|
||||
if (isPresent) {
|
||||
return _value
|
||||
}
|
||||
|
||||
throw NoSuchElementException("No value is present")
|
||||
}
|
||||
|
||||
inline fun ifPresent(block: (T) -> Unit): KOptional<T> {
|
||||
if (isPresent) {
|
||||
block.invoke(value)
|
||||
}
|
||||
|
||||
return this
|
||||
}
|
||||
|
||||
inline fun ifNotPresent(block: () -> Unit): KOptional<T> {
|
||||
if (!isPresent) {
|
||||
block.invoke()
|
||||
}
|
||||
|
||||
return this
|
||||
}
|
||||
|
||||
inline fun <R> map(block: (T) -> R): KOptional<R> {
|
||||
if (isPresent) {
|
||||
return of(block.invoke(value))
|
||||
} else {
|
||||
return empty()
|
||||
}
|
||||
}
|
||||
|
||||
@Suppress("NOTHING_TO_INLINE")
|
||||
inline fun orElse(value: T): T {
|
||||
if (isPresent) {
|
||||
return this.value
|
||||
} else {
|
||||
return value
|
||||
}
|
||||
}
|
||||
|
||||
inline fun orElse(value: () -> T): T {
|
||||
if (isPresent) {
|
||||
return this.value
|
||||
} else {
|
||||
return value.invoke()
|
||||
}
|
||||
}
|
||||
|
||||
infix fun or(value: KOptional<T>): KOptional<T> {
|
||||
if (isPresent) {
|
||||
return this
|
||||
} else {
|
||||
return value
|
||||
}
|
||||
}
|
||||
|
||||
override fun equals(other: Any?): Boolean {
|
||||
return this === other || other is KOptional<*> && isPresent == other.isPresent && _value == other._value
|
||||
}
|
||||
|
||||
override fun hashCode(): Int {
|
||||
return _value.hashCode() + 43839429
|
||||
}
|
||||
|
||||
override fun toString(): String {
|
||||
if (isPresent) {
|
||||
return "KOptional[value = $value]"
|
||||
} else {
|
||||
return "KOptional[empty]"
|
||||
}
|
||||
}
|
||||
|
||||
override fun get(): T {
|
||||
return value
|
||||
}
|
||||
|
||||
// gson hack since type token can't diff between nullable and non-null types
|
||||
@Deprecated("internal class")
|
||||
internal class Nullable<T : Any?> private constructor()
|
||||
@Deprecated("internal class")
|
||||
internal class NotNull<T : Any> private constructor()
|
||||
|
||||
companion object {
|
||||
private val EMPTY = KOptional(null, false)
|
||||
private val NULL = KOptional(null, true)
|
||||
private val TRUE = KOptional(true, true)
|
||||
private val FALSE = KOptional(false, true)
|
||||
|
||||
@JvmStatic
|
||||
fun <T> empty(): KOptional<T> {
|
||||
return EMPTY as KOptional<T>
|
||||
}
|
||||
|
||||
@JvmStatic
|
||||
fun <T> of(value: T): KOptional<T> {
|
||||
if (value == null) {
|
||||
return NULL as KOptional<T>
|
||||
} else {
|
||||
return KOptional(value, true)
|
||||
}
|
||||
}
|
||||
|
||||
@JvmStatic
|
||||
fun <T> ofNullable(value: T): KOptional<T> {
|
||||
if (value == null) {
|
||||
return EMPTY as KOptional<T>
|
||||
} else {
|
||||
return KOptional(value, true)
|
||||
}
|
||||
}
|
||||
|
||||
@JvmStatic
|
||||
fun of(value: Boolean): KOptional<Boolean> {
|
||||
if (value) {
|
||||
return TRUE
|
||||
} else {
|
||||
return FALSE
|
||||
}
|
||||
}
|
||||
|
||||
@JvmStatic
|
||||
fun of(value: Boolean?): KOptional<Boolean> {
|
||||
if (value == null) {
|
||||
return NULL as KOptional<Boolean>
|
||||
} else if (value) {
|
||||
return TRUE
|
||||
} else {
|
||||
return FALSE
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,43 @@
|
||||
package ru.dbotthepony.kommons.event
|
||||
|
||||
import ru.dbotthepony.kommons.util.BooleanConsumer
|
||||
import java.util.function.Consumer
|
||||
|
||||
interface IBooleanSubscriptable : ISubscriptable<Boolean> {
|
||||
@Deprecated("Use type specific listener")
|
||||
override fun addListener(listener: Consumer<Boolean>): ISubscriptable.L {
|
||||
return addListener(listener::accept)
|
||||
}
|
||||
|
||||
fun addListener(listener: BooleanConsumer): ISubscriptable.L
|
||||
|
||||
class Impl : IBooleanSubscriptable, Consumer<Boolean>, BooleanConsumer {
|
||||
private inner class L(val callback: BooleanConsumer) : ISubscriptable.L {
|
||||
private var isRemoved = false
|
||||
|
||||
init {
|
||||
subscribers.add(this)
|
||||
}
|
||||
|
||||
override fun remove() {
|
||||
if (!isRemoved) {
|
||||
isRemoved = true
|
||||
queue.add(this)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private val subscribers = LinkedHashSet<L>(0)
|
||||
private val queue = ArrayList<L>(0)
|
||||
|
||||
override fun addListener(listener: BooleanConsumer): ISubscriptable.L {
|
||||
return L(listener)
|
||||
}
|
||||
|
||||
override fun accept(t: Boolean) {
|
||||
queue.forEach { subscribers.remove(it) }
|
||||
queue.clear()
|
||||
subscribers.forEach { it.callback.accept(t) }
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,43 @@
|
||||
package ru.dbotthepony.kommons.event
|
||||
|
||||
import java.util.function.Consumer
|
||||
import java.util.function.DoubleConsumer
|
||||
|
||||
interface IDoubleSubcripable : ISubscriptable<Double> {
|
||||
@Deprecated("Use type specific listener")
|
||||
override fun addListener(listener: Consumer<Double>): ISubscriptable.L {
|
||||
return addListener(DoubleConsumer { listener.accept(it) })
|
||||
}
|
||||
|
||||
fun addListener(listener: DoubleConsumer): ISubscriptable.L
|
||||
|
||||
class Impl : IDoubleSubcripable, Consumer<Double>, DoubleConsumer {
|
||||
private inner class L(val callback: DoubleConsumer) : ISubscriptable.L {
|
||||
private var isRemoved = false
|
||||
|
||||
init {
|
||||
subscribers.add(this)
|
||||
}
|
||||
|
||||
override fun remove() {
|
||||
if (!isRemoved) {
|
||||
isRemoved = true
|
||||
queue.add(this)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private val subscribers = LinkedHashSet<L>(0)
|
||||
private val queue = ArrayList<L>(0)
|
||||
|
||||
override fun addListener(listener: DoubleConsumer): ISubscriptable.L {
|
||||
return L(listener)
|
||||
}
|
||||
|
||||
override fun accept(t: Double) {
|
||||
queue.forEach { subscribers.remove(it) }
|
||||
queue.clear()
|
||||
subscribers.forEach { it.callback.accept(t) }
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,43 @@
|
||||
package ru.dbotthepony.kommons.event
|
||||
|
||||
import ru.dbotthepony.kommons.util.FloatConsumer
|
||||
import java.util.function.Consumer
|
||||
|
||||
interface IFloatSubcripable : ISubscriptable<Float> {
|
||||
@Deprecated("Use type specific listener")
|
||||
override fun addListener(listener: Consumer<Float>): ISubscriptable.L {
|
||||
return addListener(listener::accept)
|
||||
}
|
||||
|
||||
fun addListener(listener: FloatConsumer): ISubscriptable.L
|
||||
|
||||
class Impl : IFloatSubcripable, Consumer<Float>, FloatConsumer {
|
||||
private inner class L(val callback: FloatConsumer) : ISubscriptable.L {
|
||||
private var isRemoved = false
|
||||
|
||||
init {
|
||||
subscribers.add(this)
|
||||
}
|
||||
|
||||
override fun remove() {
|
||||
if (!isRemoved) {
|
||||
isRemoved = true
|
||||
queue.add(this)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private val subscribers = LinkedHashSet<L>(0)
|
||||
private val queue = ArrayList<L>(0)
|
||||
|
||||
override fun addListener(listener: FloatConsumer): ISubscriptable.L {
|
||||
return L(listener)
|
||||
}
|
||||
|
||||
override fun accept(t: Float) {
|
||||
queue.forEach { subscribers.remove(it) }
|
||||
queue.clear()
|
||||
subscribers.forEach { it.callback.accept(t) }
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,43 @@
|
||||
package ru.dbotthepony.kommons.event
|
||||
|
||||
import java.util.function.Consumer
|
||||
import java.util.function.IntConsumer
|
||||
|
||||
interface IIntSubcripable : ISubscriptable<Int> {
|
||||
@Deprecated("Use type specific listener")
|
||||
override fun addListener(listener: Consumer<Int>): ISubscriptable.L {
|
||||
return addListener(IntConsumer { listener.accept(it) })
|
||||
}
|
||||
|
||||
fun addListener(listener: IntConsumer): ISubscriptable.L
|
||||
|
||||
class Impl : IIntSubcripable, Consumer<Int>, IntConsumer {
|
||||
private inner class L(val callback: IntConsumer) : ISubscriptable.L {
|
||||
private var isRemoved = false
|
||||
|
||||
init {
|
||||
subscribers.add(this)
|
||||
}
|
||||
|
||||
override fun remove() {
|
||||
if (!isRemoved) {
|
||||
isRemoved = true
|
||||
queue.add(this)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private val subscribers = LinkedHashSet<L>(0)
|
||||
private val queue = ArrayList<L>(0)
|
||||
|
||||
override fun addListener(listener: IntConsumer): ISubscriptable.L {
|
||||
return L(listener)
|
||||
}
|
||||
|
||||
override fun accept(t: Int) {
|
||||
queue.forEach { subscribers.remove(it) }
|
||||
queue.clear()
|
||||
subscribers.forEach { it.callback.accept(t) }
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,43 @@
|
||||
package ru.dbotthepony.kommons.event
|
||||
|
||||
import java.util.function.Consumer
|
||||
import java.util.function.LongConsumer
|
||||
|
||||
interface ILongSubcripable : ISubscriptable<Long> {
|
||||
@Deprecated("Use type specific listener")
|
||||
override fun addListener(listener: Consumer<Long>): ISubscriptable.L {
|
||||
return addListener(LongConsumer { listener.accept(it) })
|
||||
}
|
||||
|
||||
fun addListener(listener: LongConsumer): ISubscriptable.L
|
||||
|
||||
class Impl : ILongSubcripable, Consumer<Long>, LongConsumer {
|
||||
private inner class L(val callback: LongConsumer) : ISubscriptable.L {
|
||||
private var isRemoved = false
|
||||
|
||||
init {
|
||||
subscribers.add(this)
|
||||
}
|
||||
|
||||
override fun remove() {
|
||||
if (!isRemoved) {
|
||||
isRemoved = true
|
||||
queue.add(this)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private val subscribers = LinkedHashSet<L>(0)
|
||||
private val queue = ArrayList<L>(0)
|
||||
|
||||
override fun addListener(listener: LongConsumer): ISubscriptable.L {
|
||||
return L(listener)
|
||||
}
|
||||
|
||||
override fun accept(t: Long) {
|
||||
queue.forEach { subscribers.remove(it) }
|
||||
queue.clear()
|
||||
subscribers.forEach { it.callback.accept(t) }
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,60 @@
|
||||
package ru.dbotthepony.kommons.event
|
||||
|
||||
import java.util.function.Consumer
|
||||
|
||||
interface ISubscriptable<V> {
|
||||
/**
|
||||
* Listener token, allows to remove listener from subscriber list
|
||||
*/
|
||||
fun interface L {
|
||||
/**
|
||||
* Removes this listener
|
||||
*/
|
||||
fun remove()
|
||||
}
|
||||
|
||||
fun addListener(listener: Consumer<V>): L
|
||||
|
||||
class Impl<V> : ISubscriptable<V>, Consumer<V> {
|
||||
private inner class L(val callback: Consumer<V>) : ISubscriptable.L {
|
||||
private var isRemoved = false
|
||||
|
||||
init {
|
||||
subscribers.add(this)
|
||||
}
|
||||
|
||||
override fun remove() {
|
||||
if (!isRemoved) {
|
||||
isRemoved = true
|
||||
queue.add(this)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private val subscribers = LinkedHashSet<L>(0)
|
||||
private val queue = ArrayList<L>(0)
|
||||
|
||||
override fun addListener(listener: Consumer<V>): ISubscriptable.L {
|
||||
return L(listener)
|
||||
}
|
||||
|
||||
override fun accept(t: V) {
|
||||
queue.forEach { subscribers.remove(it) }
|
||||
queue.clear()
|
||||
subscribers.forEach { it.callback.accept(t) }
|
||||
}
|
||||
}
|
||||
|
||||
companion object : ISubscriptable<Nothing>, L {
|
||||
@Suppress("unchecked_cast")
|
||||
fun <T> empty(): ISubscriptable<T> {
|
||||
return this as ISubscriptable<T>
|
||||
}
|
||||
|
||||
override fun remove() {}
|
||||
|
||||
override fun addListener(listener: Consumer<Nothing>): L {
|
||||
return this
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,41 @@
|
||||
package ru.dbotthepony.kommons.event
|
||||
|
||||
import java.util.function.Consumer
|
||||
|
||||
interface IUnitSubscripable : ISubscriptable<Unit> {
|
||||
fun addListener(listener: Runnable): ISubscriptable.L
|
||||
|
||||
override fun addListener(listener: Consumer<Unit>): ISubscriptable.L {
|
||||
return addListener(Runnable { listener.accept(Unit) })
|
||||
}
|
||||
|
||||
class Impl : IUnitSubscripable, Runnable {
|
||||
private inner class L(val callback: Runnable) : ISubscriptable.L {
|
||||
private var isRemoved = false
|
||||
|
||||
init {
|
||||
subscribers.add(this)
|
||||
}
|
||||
|
||||
override fun remove() {
|
||||
if (!isRemoved) {
|
||||
isRemoved = true
|
||||
queue.add(this)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private val subscribers = LinkedHashSet<L>(0)
|
||||
private val queue = ArrayList<L>(0)
|
||||
|
||||
override fun addListener(listener: Runnable): ISubscriptable.L {
|
||||
return L(listener)
|
||||
}
|
||||
|
||||
override fun run() {
|
||||
queue.forEach { subscribers.remove(it) }
|
||||
queue.clear()
|
||||
subscribers.forEach { it.callback.run() }
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,7 @@
|
||||
package ru.dbotthepony.kommons.util
|
||||
|
||||
import java.util.function.Consumer
|
||||
|
||||
fun interface BooleanConsumer : Consumer<Boolean> {
|
||||
override fun accept(value: Boolean)
|
||||
}
|
@ -0,0 +1,38 @@
|
||||
package ru.dbotthepony.kommons.util
|
||||
|
||||
import java.util.function.Supplier
|
||||
|
||||
|
||||
fun <T> Comparator<T>.nullsFirst(): Comparator<T> {
|
||||
return Comparator.nullsFirst(this)
|
||||
}
|
||||
|
||||
fun <T> Comparator<T>.nullsLast(): Comparator<T> {
|
||||
return Comparator.nullsLast(this)
|
||||
}
|
||||
|
||||
class MappedComparator<T, O>(private val parent: Comparator<O>, private val mapper: (T) -> O) : Comparator<T> {
|
||||
override fun compare(o1: T, o2: T): Int {
|
||||
return parent.compare(mapper.invoke(o1), mapper.invoke(o2))
|
||||
}
|
||||
|
||||
override fun equals(other: Any?): Boolean {
|
||||
return other is MappedComparator<*, *> && parent == other.parent
|
||||
}
|
||||
|
||||
override fun hashCode(): Int {
|
||||
return parent.hashCode()
|
||||
}
|
||||
|
||||
override fun toString(): String {
|
||||
return "MappedComparator[$parent]"
|
||||
}
|
||||
}
|
||||
|
||||
fun <A, B> Comparator<A>.map(mapper: (B) -> A): Comparator<B> {
|
||||
return MappedComparator(this, mapper)
|
||||
}
|
||||
|
||||
fun <T> Comparator<T>.suppliers(): Comparator<Supplier<T>> {
|
||||
return MappedComparator(this) { it.get() }
|
||||
}
|
@ -0,0 +1,7 @@
|
||||
package ru.dbotthepony.kommons.util
|
||||
|
||||
import java.util.function.Consumer
|
||||
|
||||
fun interface FloatConsumer : Consumer<Float> {
|
||||
override fun accept(value: Float)
|
||||
}
|
@ -0,0 +1,12 @@
|
||||
package ru.dbotthepony.kommons.util
|
||||
|
||||
import java.util.function.Supplier
|
||||
|
||||
fun interface FloatSupplier : Supplier<Float> {
|
||||
fun getAsFloat(): Float
|
||||
|
||||
@Deprecated("Use type-specific method", replaceWith = ReplaceWith("this.getAsFloat()"))
|
||||
override fun get(): Float {
|
||||
return getAsFloat()
|
||||
}
|
||||
}
|
14
gradle.properties
Normal file
14
gradle.properties
Normal file
@ -0,0 +1,14 @@
|
||||
|
||||
kotlin.code.style=official
|
||||
|
||||
projectGroup=ru.dbotthepony.kommons
|
||||
|
||||
guavaDepVersion=33.0.0
|
||||
|
||||
kommonsVersion=1.0
|
||||
coreVersion=1.0
|
||||
ioVersion=1.0
|
||||
networkingVersion=1.0
|
||||
mathVersion=1.0
|
||||
collectVersion=1.0
|
||||
guavaVersion=1.0
|
BIN
gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
BIN
gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
Binary file not shown.
6
gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
6
gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
@ -0,0 +1,6 @@
|
||||
#Sat Feb 03 12:42:21 KRAT 2024
|
||||
distributionBase=GRADLE_USER_HOME
|
||||
distributionPath=wrapper/dists
|
||||
distributionUrl=https\://services.gradle.org/distributions/gradle-8.4-bin.zip
|
||||
zipStoreBase=GRADLE_USER_HOME
|
||||
zipStorePath=wrapper/dists
|
234
gradlew
vendored
Normal file
234
gradlew
vendored
Normal file
@ -0,0 +1,234 @@
|
||||
#!/bin/sh
|
||||
|
||||
#
|
||||
# Copyright © 2015-2021 the original authors.
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||
# you may not use this file except in compliance with the License.
|
||||
# You may obtain a copy of the License at
|
||||
#
|
||||
# https://www.apache.org/licenses/LICENSE-2.0
|
||||
#
|
||||
# Unless required by applicable law or agreed to in writing, software
|
||||
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
# See the License for the specific language governing permissions and
|
||||
# limitations under the License.
|
||||
#
|
||||
|
||||
##############################################################################
|
||||
#
|
||||
# Gradle start up script for POSIX generated by Gradle.
|
||||
#
|
||||
# Important for running:
|
||||
#
|
||||
# (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is
|
||||
# noncompliant, but you have some other compliant shell such as ksh or
|
||||
# bash, then to run this script, type that shell name before the whole
|
||||
# command line, like:
|
||||
#
|
||||
# ksh Gradle
|
||||
#
|
||||
# Busybox and similar reduced shells will NOT work, because this script
|
||||
# requires all of these POSIX shell features:
|
||||
# * functions;
|
||||
# * expansions «$var», «${var}», «${var:-default}», «${var+SET}»,
|
||||
# «${var#prefix}», «${var%suffix}», and «$( cmd )»;
|
||||
# * compound commands having a testable exit status, especially «case»;
|
||||
# * various built-in commands including «command», «set», and «ulimit».
|
||||
#
|
||||
# Important for patching:
|
||||
#
|
||||
# (2) This script targets any POSIX shell, so it avoids extensions provided
|
||||
# by Bash, Ksh, etc; in particular arrays are avoided.
|
||||
#
|
||||
# The "traditional" practice of packing multiple parameters into a
|
||||
# space-separated string is a well documented source of bugs and security
|
||||
# problems, so this is (mostly) avoided, by progressively accumulating
|
||||
# options in "$@", and eventually passing that to Java.
|
||||
#
|
||||
# Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS,
|
||||
# and GRADLE_OPTS) rely on word-splitting, this is performed explicitly;
|
||||
# see the in-line comments for details.
|
||||
#
|
||||
# There are tweaks for specific operating systems such as AIX, CygWin,
|
||||
# Darwin, MinGW, and NonStop.
|
||||
#
|
||||
# (3) This script is generated from the Groovy template
|
||||
# https://github.com/gradle/gradle/blob/master/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt
|
||||
# within the Gradle project.
|
||||
#
|
||||
# You can find Gradle at https://github.com/gradle/gradle/.
|
||||
#
|
||||
##############################################################################
|
||||
|
||||
# Attempt to set APP_HOME
|
||||
|
||||
# Resolve links: $0 may be a link
|
||||
app_path=$0
|
||||
|
||||
# Need this for daisy-chained symlinks.
|
||||
while
|
||||
APP_HOME=${app_path%"${app_path##*/}"} # leaves a trailing /; empty if no leading path
|
||||
[ -h "$app_path" ]
|
||||
do
|
||||
ls=$( ls -ld "$app_path" )
|
||||
link=${ls#*' -> '}
|
||||
case $link in #(
|
||||
/*) app_path=$link ;; #(
|
||||
*) app_path=$APP_HOME$link ;;
|
||||
esac
|
||||
done
|
||||
|
||||
APP_HOME=$( cd "${APP_HOME:-./}" && pwd -P ) || exit
|
||||
|
||||
APP_NAME="Gradle"
|
||||
APP_BASE_NAME=${0##*/}
|
||||
|
||||
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||
DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
|
||||
|
||||
# Use the maximum available, or set MAX_FD != -1 to use that value.
|
||||
MAX_FD=maximum
|
||||
|
||||
warn () {
|
||||
echo "$*"
|
||||
} >&2
|
||||
|
||||
die () {
|
||||
echo
|
||||
echo "$*"
|
||||
echo
|
||||
exit 1
|
||||
} >&2
|
||||
|
||||
# OS specific support (must be 'true' or 'false').
|
||||
cygwin=false
|
||||
msys=false
|
||||
darwin=false
|
||||
nonstop=false
|
||||
case "$( uname )" in #(
|
||||
CYGWIN* ) cygwin=true ;; #(
|
||||
Darwin* ) darwin=true ;; #(
|
||||
MSYS* | MINGW* ) msys=true ;; #(
|
||||
NONSTOP* ) nonstop=true ;;
|
||||
esac
|
||||
|
||||
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
|
||||
|
||||
|
||||
# Determine the Java command to use to start the JVM.
|
||||
if [ -n "$JAVA_HOME" ] ; then
|
||||
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
|
||||
# IBM's JDK on AIX uses strange locations for the executables
|
||||
JAVACMD=$JAVA_HOME/jre/sh/java
|
||||
else
|
||||
JAVACMD=$JAVA_HOME/bin/java
|
||||
fi
|
||||
if [ ! -x "$JAVACMD" ] ; then
|
||||
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
|
||||
|
||||
Please set the JAVA_HOME variable in your environment to match the
|
||||
location of your Java installation."
|
||||
fi
|
||||
else
|
||||
JAVACMD=java
|
||||
which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
||||
|
||||
Please set the JAVA_HOME variable in your environment to match the
|
||||
location of your Java installation."
|
||||
fi
|
||||
|
||||
# Increase the maximum file descriptors if we can.
|
||||
if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then
|
||||
case $MAX_FD in #(
|
||||
max*)
|
||||
MAX_FD=$( ulimit -H -n ) ||
|
||||
warn "Could not query maximum file descriptor limit"
|
||||
esac
|
||||
case $MAX_FD in #(
|
||||
'' | soft) :;; #(
|
||||
*)
|
||||
ulimit -n "$MAX_FD" ||
|
||||
warn "Could not set maximum file descriptor limit to $MAX_FD"
|
||||
esac
|
||||
fi
|
||||
|
||||
# Collect all arguments for the java command, stacking in reverse order:
|
||||
# * args from the command line
|
||||
# * the main class name
|
||||
# * -classpath
|
||||
# * -D...appname settings
|
||||
# * --module-path (only if needed)
|
||||
# * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables.
|
||||
|
||||
# For Cygwin or MSYS, switch paths to Windows format before running java
|
||||
if "$cygwin" || "$msys" ; then
|
||||
APP_HOME=$( cygpath --path --mixed "$APP_HOME" )
|
||||
CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" )
|
||||
|
||||
JAVACMD=$( cygpath --unix "$JAVACMD" )
|
||||
|
||||
# Now convert the arguments - kludge to limit ourselves to /bin/sh
|
||||
for arg do
|
||||
if
|
||||
case $arg in #(
|
||||
-*) false ;; # don't mess with options #(
|
||||
/?*) t=${arg#/} t=/${t%%/*} # looks like a POSIX filepath
|
||||
[ -e "$t" ] ;; #(
|
||||
*) false ;;
|
||||
esac
|
||||
then
|
||||
arg=$( cygpath --path --ignore --mixed "$arg" )
|
||||
fi
|
||||
# Roll the args list around exactly as many times as the number of
|
||||
# args, so each arg winds up back in the position where it started, but
|
||||
# possibly modified.
|
||||
#
|
||||
# NB: a `for` loop captures its iteration list before it begins, so
|
||||
# changing the positional parameters here affects neither the number of
|
||||
# iterations, nor the values presented in `arg`.
|
||||
shift # remove old arg
|
||||
set -- "$@" "$arg" # push replacement arg
|
||||
done
|
||||
fi
|
||||
|
||||
# Collect all arguments for the java command;
|
||||
# * $DEFAULT_JVM_OPTS, $JAVA_OPTS, and $GRADLE_OPTS can contain fragments of
|
||||
# shell script including quotes and variable substitutions, so put them in
|
||||
# double quotes to make sure that they get re-expanded; and
|
||||
# * put everything else in single quotes, so that it's not re-expanded.
|
||||
|
||||
set -- \
|
||||
"-Dorg.gradle.appname=$APP_BASE_NAME" \
|
||||
-classpath "$CLASSPATH" \
|
||||
org.gradle.wrapper.GradleWrapperMain \
|
||||
"$@"
|
||||
|
||||
# Use "xargs" to parse quoted args.
|
||||
#
|
||||
# With -n1 it outputs one arg per line, with the quotes and backslashes removed.
|
||||
#
|
||||
# In Bash we could simply go:
|
||||
#
|
||||
# readarray ARGS < <( xargs -n1 <<<"$var" ) &&
|
||||
# set -- "${ARGS[@]}" "$@"
|
||||
#
|
||||
# but POSIX shell has neither arrays nor command substitution, so instead we
|
||||
# post-process each arg (as a line of input to sed) to backslash-escape any
|
||||
# character that might be a shell metacharacter, then use eval to reverse
|
||||
# that process (while maintaining the separation between arguments), and wrap
|
||||
# the whole thing up as a single "set" statement.
|
||||
#
|
||||
# This will of course break if any of these variables contains a newline or
|
||||
# an unmatched quote.
|
||||
#
|
||||
|
||||
eval "set -- $(
|
||||
printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" |
|
||||
xargs -n1 |
|
||||
sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' |
|
||||
tr '\n' ' '
|
||||
)" '"$@"'
|
||||
|
||||
exec "$JAVACMD" "$@"
|
89
gradlew.bat
vendored
Normal file
89
gradlew.bat
vendored
Normal file
@ -0,0 +1,89 @@
|
||||
@rem
|
||||
@rem Copyright 2015 the original author or authors.
|
||||
@rem
|
||||
@rem Licensed under the Apache License, Version 2.0 (the "License");
|
||||
@rem you may not use this file except in compliance with the License.
|
||||
@rem You may obtain a copy of the License at
|
||||
@rem
|
||||
@rem https://www.apache.org/licenses/LICENSE-2.0
|
||||
@rem
|
||||
@rem Unless required by applicable law or agreed to in writing, software
|
||||
@rem distributed under the License is distributed on an "AS IS" BASIS,
|
||||
@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
@rem See the License for the specific language governing permissions and
|
||||
@rem limitations under the License.
|
||||
@rem
|
||||
|
||||
@if "%DEBUG%" == "" @echo off
|
||||
@rem ##########################################################################
|
||||
@rem
|
||||
@rem Gradle startup script for Windows
|
||||
@rem
|
||||
@rem ##########################################################################
|
||||
|
||||
@rem Set local scope for the variables with windows NT shell
|
||||
if "%OS%"=="Windows_NT" setlocal
|
||||
|
||||
set DIRNAME=%~dp0
|
||||
if "%DIRNAME%" == "" set DIRNAME=.
|
||||
set APP_BASE_NAME=%~n0
|
||||
set APP_HOME=%DIRNAME%
|
||||
|
||||
@rem Resolve any "." and ".." in APP_HOME to make it shorter.
|
||||
for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
|
||||
|
||||
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||
set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m"
|
||||
|
||||
@rem Find java.exe
|
||||
if defined JAVA_HOME goto findJavaFromJavaHome
|
||||
|
||||
set JAVA_EXE=java.exe
|
||||
%JAVA_EXE% -version >NUL 2>&1
|
||||
if "%ERRORLEVEL%" == "0" goto execute
|
||||
|
||||
echo.
|
||||
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
||||
echo.
|
||||
echo Please set the JAVA_HOME variable in your environment to match the
|
||||
echo location of your Java installation.
|
||||
|
||||
goto fail
|
||||
|
||||
:findJavaFromJavaHome
|
||||
set JAVA_HOME=%JAVA_HOME:"=%
|
||||
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
|
||||
|
||||
if exist "%JAVA_EXE%" goto execute
|
||||
|
||||
echo.
|
||||
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
|
||||
echo.
|
||||
echo Please set the JAVA_HOME variable in your environment to match the
|
||||
echo location of your Java installation.
|
||||
|
||||
goto fail
|
||||
|
||||
:execute
|
||||
@rem Setup the command line
|
||||
|
||||
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
|
||||
|
||||
|
||||
@rem Execute Gradle
|
||||
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %*
|
||||
|
||||
:end
|
||||
@rem End local scope for the variables with windows NT shell
|
||||
if "%ERRORLEVEL%"=="0" goto mainEnd
|
||||
|
||||
:fail
|
||||
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
|
||||
rem the _cmd.exe /c_ return code!
|
||||
if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
|
||||
exit /b 1
|
||||
|
||||
:mainEnd
|
||||
if "%OS%"=="Windows_NT" endlocal
|
||||
|
||||
:omega
|
43
guava/build.gradle.kts
Normal file
43
guava/build.gradle.kts
Normal file
@ -0,0 +1,43 @@
|
||||
plugins {
|
||||
kotlin("jvm")
|
||||
`maven-publish`
|
||||
}
|
||||
|
||||
val guavaVersion: String by project
|
||||
val guavaDepVersion: String by project
|
||||
val projectGroup: String by project
|
||||
|
||||
group = projectGroup
|
||||
version = guavaVersion
|
||||
|
||||
repositories {
|
||||
mavenCentral()
|
||||
}
|
||||
|
||||
dependencies {
|
||||
testImplementation("org.jetbrains.kotlin:kotlin-test")
|
||||
|
||||
implementation("com.google.guava:guava:$guavaDepVersion-jre")
|
||||
}
|
||||
|
||||
tasks.test {
|
||||
useJUnitPlatform()
|
||||
}
|
||||
|
||||
kotlin {
|
||||
jvmToolchain(17)
|
||||
}
|
||||
|
||||
publishing {
|
||||
publications {
|
||||
create<MavenPublication>("mavenJava") {
|
||||
from(components["java"])
|
||||
|
||||
pom {
|
||||
dependencies {
|
||||
implementation(kotlin("stdlib"))
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,54 @@
|
||||
package ru.dbotthepony.kommons.collect
|
||||
|
||||
import com.google.common.collect.ImmutableList
|
||||
import com.google.common.collect.ImmutableMap
|
||||
import com.google.common.collect.ImmutableMultimap
|
||||
import com.google.common.collect.ImmutableSet
|
||||
import java.util.function.Consumer
|
||||
|
||||
inline fun <T : Any> immutableList(size: Int, initializer: (index: Int) -> T): ImmutableList<T> {
|
||||
require(size >= 0) { "Invalid list size $size" }
|
||||
|
||||
return when (size) {
|
||||
0 -> ImmutableList.of()
|
||||
1 -> ImmutableList.of(initializer(0))
|
||||
else -> ImmutableList.Builder<T>().let {
|
||||
for (i in 0 until size) {
|
||||
it.add(initializer(i))
|
||||
}
|
||||
|
||||
it.build()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
inline fun <K : Any, V : Any> immutableMap(initializer: ImmutableMap.Builder<K, V>.() -> Unit): ImmutableMap<K, V> {
|
||||
val builder = ImmutableMap.Builder<K, V>()
|
||||
initializer.invoke(builder)
|
||||
return builder.build()
|
||||
}
|
||||
|
||||
inline fun <K : Any, V : Any> immutableMultimap(initializer: ImmutableMultimap.Builder<K, V>.() -> Unit): ImmutableMultimap<K, V> {
|
||||
val builder = ImmutableMultimap.Builder<K, V>()
|
||||
initializer.invoke(builder)
|
||||
return builder.build()
|
||||
}
|
||||
|
||||
inline fun <V : Any> immutableSet(initializer: Consumer<V>.() -> Unit): ImmutableSet<V> {
|
||||
val builder = ImmutableSet.Builder<V>()
|
||||
initializer.invoke(builder::add)
|
||||
return builder.build()
|
||||
}
|
||||
|
||||
inline fun <V : Any> immutableList(initializer: Consumer<V>.() -> Unit): ImmutableList<V> {
|
||||
val builder = ImmutableList.Builder<V>()
|
||||
initializer.invoke(builder::add)
|
||||
return builder.build()
|
||||
}
|
||||
|
||||
fun <V : Any> immutableList(a: V, vararg values: V): ImmutableList<V> {
|
||||
val builder = ImmutableList.Builder<V>()
|
||||
builder.add(a)
|
||||
builder.addAll(values.iterator())
|
||||
return builder.build()
|
||||
}
|
44
io-math/build.gradle.kts
Normal file
44
io-math/build.gradle.kts
Normal file
@ -0,0 +1,44 @@
|
||||
plugins {
|
||||
kotlin("jvm")
|
||||
`maven-publish`
|
||||
}
|
||||
|
||||
val kommonsVersion: String by project
|
||||
val projectGroup: String by project
|
||||
|
||||
group = projectGroup
|
||||
version = kommonsVersion
|
||||
|
||||
repositories {
|
||||
mavenCentral()
|
||||
}
|
||||
|
||||
dependencies {
|
||||
testImplementation("org.jetbrains.kotlin:kotlin-test")
|
||||
implementation(project(":io"))
|
||||
implementation(project(":math"))
|
||||
}
|
||||
|
||||
tasks.test {
|
||||
useJUnitPlatform()
|
||||
}
|
||||
|
||||
kotlin {
|
||||
jvmToolchain(17)
|
||||
}
|
||||
|
||||
publishing {
|
||||
publications {
|
||||
create<MavenPublication>("mavenJava") {
|
||||
from(components["java"])
|
||||
|
||||
pom {
|
||||
dependencies {
|
||||
implementation(kotlin("stdlib"))
|
||||
implementation(project(":io"))
|
||||
implementation(project(":math"))
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,23 @@
|
||||
package ru.dbotthepony.kommons.io
|
||||
|
||||
import ru.dbotthepony.kommons.math.Decimal
|
||||
import java.io.DataInputStream
|
||||
import java.io.DataOutputStream
|
||||
import java.io.InputStream
|
||||
import java.io.OutputStream
|
||||
|
||||
fun InputStream.readDecimal(): Decimal {
|
||||
val size = readVarInt()
|
||||
require(size >= 0) { "Negative payload size: $size" }
|
||||
val bytes = ByteArray(size)
|
||||
read(bytes)
|
||||
return Decimal.fromByteArray(bytes)
|
||||
}
|
||||
|
||||
fun OutputStream.writeDecimal(value: Decimal) {
|
||||
val bytes = value.toByteArray()
|
||||
writeVarInt(bytes.size)
|
||||
write(bytes)
|
||||
}
|
||||
|
||||
val DecimalValueCodec = StreamCodec(DataInputStream::readDecimal, DataOutputStream::writeDecimal)
|
42
io/build.gradle.kts
Normal file
42
io/build.gradle.kts
Normal file
@ -0,0 +1,42 @@
|
||||
plugins {
|
||||
kotlin("jvm")
|
||||
`maven-publish`
|
||||
}
|
||||
|
||||
val ioVersion: String by project
|
||||
val projectGroup: String by project
|
||||
|
||||
group = projectGroup
|
||||
version = ioVersion
|
||||
|
||||
repositories {
|
||||
mavenCentral()
|
||||
}
|
||||
|
||||
dependencies {
|
||||
testImplementation("org.jetbrains.kotlin:kotlin-test")
|
||||
implementation(project(":core"))
|
||||
}
|
||||
|
||||
tasks.test {
|
||||
useJUnitPlatform()
|
||||
}
|
||||
|
||||
kotlin {
|
||||
jvmToolchain(17)
|
||||
}
|
||||
|
||||
publishing {
|
||||
publications {
|
||||
create<MavenPublication>("mavenJava") {
|
||||
from(components["java"])
|
||||
|
||||
pom {
|
||||
dependencies {
|
||||
implementation(kotlin("stdlib"))
|
||||
implementation(project(":core"))
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,34 @@
|
||||
package ru.dbotthepony.kommons.io
|
||||
|
||||
import java.io.DataInputStream
|
||||
import java.io.DataOutputStream
|
||||
|
||||
class CollectionStreamCodec<E, C : MutableCollection<E>>(val elementCodec: IStreamCodec<E>, val collectionFactory: (Int) -> C) :
|
||||
IStreamCodec<C> {
|
||||
override fun read(stream: DataInputStream): C {
|
||||
val size = stream.readVarInt()
|
||||
|
||||
if (size <= 0) {
|
||||
return collectionFactory.invoke(0)
|
||||
}
|
||||
|
||||
val collection = collectionFactory.invoke(size)
|
||||
|
||||
for (i in 0 until size) {
|
||||
collection.add(elementCodec.read(stream))
|
||||
}
|
||||
|
||||
return collection
|
||||
}
|
||||
|
||||
override fun write(stream: DataOutputStream, value: C) {
|
||||
stream.writeVarInt(value.size)
|
||||
value.forEach { elementCodec.write(stream, it) }
|
||||
}
|
||||
|
||||
override fun copy(value: C): C {
|
||||
val new = collectionFactory.invoke(value.size)
|
||||
value.forEach { new.add(elementCodec.copy(it)) }
|
||||
return new
|
||||
}
|
||||
}
|
@ -0,0 +1,48 @@
|
||||
package ru.dbotthepony.kommons.io
|
||||
|
||||
import java.io.DataInputStream
|
||||
import java.io.DataOutputStream
|
||||
|
||||
class EnumValueCodec<V : Enum<V>>(clazz: Class<out V>) : IStreamCodec<V> {
|
||||
val clazz = searchClass(clazz)
|
||||
val values: List<V> = listOf(*this.clazz.enumConstants!!)
|
||||
val valuesMap = values.associateBy { it.name }
|
||||
|
||||
override fun read(stream: DataInputStream): V {
|
||||
val id = stream.readVarInt()
|
||||
return values.getOrNull(id) ?: throw NoSuchElementException("No such enum with index $id")
|
||||
}
|
||||
|
||||
override fun write(stream: DataOutputStream, value: V) {
|
||||
stream.writeVarInt(value.ordinal)
|
||||
}
|
||||
|
||||
override fun copy(value: V): V {
|
||||
return value
|
||||
}
|
||||
|
||||
override fun compare(a: V, b: V): Boolean {
|
||||
return a === b
|
||||
}
|
||||
|
||||
companion object {
|
||||
/**
|
||||
* FIXME: enums with abstract methods which get compiled to subclasses, whose DO NOT expose "parent's" enum constants array
|
||||
*
|
||||
* is there an already existing solution?
|
||||
*/
|
||||
fun <V : Enum<V>> searchClass(clazz: Class<out V>): Class<out V> {
|
||||
var search: Class<*> = clazz
|
||||
|
||||
while (search.enumConstants == null && search.superclass != null) {
|
||||
search = search.superclass
|
||||
}
|
||||
|
||||
if (search.enumConstants == null) {
|
||||
throw ClassCastException("$clazz does not represent an enum or enum subclass")
|
||||
}
|
||||
|
||||
return search as Class<out V>
|
||||
}
|
||||
}
|
||||
}
|
28
io/src/main/kotlin/ru/dbotthepony/kommons/io/IStreamCodec.kt
Normal file
28
io/src/main/kotlin/ru/dbotthepony/kommons/io/IStreamCodec.kt
Normal file
@ -0,0 +1,28 @@
|
||||
package ru.dbotthepony.kommons.io
|
||||
|
||||
import java.io.DataInputStream
|
||||
import java.io.DataOutputStream
|
||||
|
||||
/**
|
||||
* Represents value which can be encoded onto or decoded from stream.
|
||||
*
|
||||
* Also provides [copy] and [compare] methods
|
||||
*/
|
||||
interface IStreamCodec<V> {
|
||||
fun read(stream: DataInputStream): V
|
||||
fun write(stream: DataOutputStream, value: V)
|
||||
|
||||
/**
|
||||
* if value is immutable, return it right away
|
||||
*/
|
||||
fun copy(value: V): V
|
||||
|
||||
/**
|
||||
* Optional equality check override. Utilized to determine whenever e.g. network value is different from new value
|
||||
*
|
||||
* By default uses [Any.equals]
|
||||
*/
|
||||
fun compare(a: V, b: V): Boolean {
|
||||
return a == b
|
||||
}
|
||||
}
|
151
io/src/main/kotlin/ru/dbotthepony/kommons/io/InputStreamUtils.kt
Normal file
151
io/src/main/kotlin/ru/dbotthepony/kommons/io/InputStreamUtils.kt
Normal file
@ -0,0 +1,151 @@
|
||||
package ru.dbotthepony.kommons.io
|
||||
|
||||
import java.io.DataInput
|
||||
import java.io.InputStream
|
||||
import java.io.RandomAccessFile
|
||||
import java.math.BigDecimal
|
||||
import java.math.BigInteger
|
||||
import java.util.function.IntSupplier
|
||||
|
||||
fun InputStream.readBigDecimal(): BigDecimal {
|
||||
val scale = readInt()
|
||||
val size = readVarInt()
|
||||
require(size >= 0) { "Negative payload size: $size" }
|
||||
val bytes = ByteArray(size)
|
||||
read(bytes)
|
||||
return BigDecimal(BigInteger(bytes), scale)
|
||||
}
|
||||
|
||||
fun <S : InputStream, V, C : MutableCollection<V>> S.readCollection(reader: S.() -> V, factory: (Int) -> C): C {
|
||||
val size = readVarInt()
|
||||
val collection = factory.invoke(size)
|
||||
|
||||
for (i in 0 until size) {
|
||||
collection.add(reader(this))
|
||||
}
|
||||
|
||||
return collection
|
||||
}
|
||||
|
||||
fun <S : InputStream, V> S.readCollection(reader: S.() -> V) = readCollection(reader, ::ArrayList)
|
||||
|
||||
fun InputStream.readInt(): Int {
|
||||
if (this is DataInput) {
|
||||
return readInt()
|
||||
}
|
||||
|
||||
return (read() shl 24) or (read() shl 16) or (read() shl 8) or read()
|
||||
}
|
||||
|
||||
fun InputStream.readLong(): Long {
|
||||
if (this is DataInput) {
|
||||
return readLong()
|
||||
}
|
||||
|
||||
return (read().toLong() shl 48) or
|
||||
(read().toLong() shl 40) or
|
||||
(read().toLong() shl 32) or
|
||||
(read().toLong() shl 24) or
|
||||
(read().toLong() shl 16) or
|
||||
(read().toLong() shl 8) or
|
||||
read().toLong()
|
||||
}
|
||||
|
||||
fun InputStream.readFloat() = Float.fromBits(readInt())
|
||||
fun InputStream.readDouble() = Double.fromBits(readLong())
|
||||
|
||||
data class VarIntReadResult(val value: Int, val cells: Int)
|
||||
data class VarLongReadResult(val value: Long, val cells: Int)
|
||||
|
||||
private fun readVarLongInfo(supplier: IntSupplier): VarLongReadResult {
|
||||
var result = 0L
|
||||
var read = supplier.asInt
|
||||
var i = 1
|
||||
|
||||
while (true) {
|
||||
result = (result shl 7) or (read.toLong() and 0x7F)
|
||||
|
||||
if (read and 0x80 == 0)
|
||||
break
|
||||
|
||||
read = supplier.asInt
|
||||
i++
|
||||
}
|
||||
|
||||
return VarLongReadResult(result, i)
|
||||
}
|
||||
|
||||
private fun readVarIntInfo(supplier: IntSupplier): VarIntReadResult {
|
||||
val read = readVarLongInfo(supplier)
|
||||
return VarIntReadResult(read.value.toInt(), read.cells)
|
||||
}
|
||||
|
||||
private fun readVarInt(supplier: IntSupplier): Int {
|
||||
return readVarIntInfo(supplier).value
|
||||
}
|
||||
|
||||
private fun readVarLong(supplier: IntSupplier): Long {
|
||||
return readVarLongInfo(supplier).value
|
||||
}
|
||||
|
||||
private fun Int.fromSignedVar(): Int {
|
||||
val sign = this and 1
|
||||
|
||||
if (sign == 0)
|
||||
return this ushr 1
|
||||
else
|
||||
return -(this ushr 1) - 1
|
||||
}
|
||||
|
||||
private fun VarIntReadResult.fromSignedVar(): VarIntReadResult {
|
||||
val sign = this.value and 1
|
||||
|
||||
if (sign == 0)
|
||||
return VarIntReadResult(this.value ushr 1, this.cells)
|
||||
else
|
||||
return VarIntReadResult(-(this.value ushr 1) - 1, this.cells)
|
||||
}
|
||||
|
||||
private fun Long.fromSignedVar(): Long {
|
||||
val sign = this and 1
|
||||
|
||||
if (sign == 0L)
|
||||
return this ushr 1
|
||||
else
|
||||
return -(this ushr 1) - 1L
|
||||
}
|
||||
|
||||
private fun VarLongReadResult.fromSignedVar(): VarLongReadResult {
|
||||
val sign = this.value and 1
|
||||
|
||||
if (sign == 0L)
|
||||
return VarLongReadResult(this.value ushr 1, this.cells)
|
||||
else
|
||||
return VarLongReadResult(-(this.value ushr 1) - 1, this.cells)
|
||||
}
|
||||
|
||||
fun RandomAccessFile.readVarLong() = readVarLong(::readUnsignedByte)
|
||||
fun RandomAccessFile.readVarInt() = readVarInt(::readUnsignedByte)
|
||||
fun RandomAccessFile.readVarLongInfo() = readVarLongInfo(::readUnsignedByte)
|
||||
fun RandomAccessFile.readVarIntInfo() = readVarIntInfo(::readUnsignedByte)
|
||||
fun InputStream.readVarLong() = readVarLong(::read)
|
||||
fun InputStream.readVarInt() = readVarInt(::read)
|
||||
fun InputStream.readVarLongInfo() = readVarLongInfo(::read)
|
||||
fun InputStream.readVarIntInfo() = readVarIntInfo(::read)
|
||||
|
||||
fun RandomAccessFile.readSignedVarLong() = readVarLong(::readUnsignedByte).fromSignedVar()
|
||||
fun RandomAccessFile.readSignedVarInt() = readVarInt(::readUnsignedByte).fromSignedVar()
|
||||
fun RandomAccessFile.readSignedVarLongInfo() = readVarLongInfo(::readUnsignedByte).fromSignedVar()
|
||||
fun RandomAccessFile.readSignedVarIntInfo() = readVarIntInfo(::readUnsignedByte).fromSignedVar()
|
||||
fun InputStream.readSignedVarLong() = readVarLong(::read).fromSignedVar()
|
||||
fun InputStream.readSignedVarInt() = readVarInt(::read).fromSignedVar()
|
||||
fun InputStream.readSignedVarLongInfo() = readVarLongInfo(::read).fromSignedVar()
|
||||
fun InputStream.readSignedVarIntInfo() = readVarIntInfo(::read).fromSignedVar()
|
||||
|
||||
fun InputStream.readBinaryString(): String {
|
||||
val size = readVarInt()
|
||||
require(size >= 0) { "Negative payload size: $size" }
|
||||
val bytes = ByteArray(size)
|
||||
read(bytes)
|
||||
return bytes.decodeToString()
|
||||
}
|
@ -0,0 +1,105 @@
|
||||
package ru.dbotthepony.kommons.io
|
||||
|
||||
import java.io.DataOutput
|
||||
import java.io.OutputStream
|
||||
import java.io.RandomAccessFile
|
||||
import java.math.BigDecimal
|
||||
import java.util.function.IntConsumer
|
||||
|
||||
fun OutputStream.writeBigDecimal(value: BigDecimal) {
|
||||
writeInt(value.scale())
|
||||
val bytes = value.unscaledValue().toByteArray()
|
||||
writeVarInt(bytes.size)
|
||||
write(bytes)
|
||||
}
|
||||
|
||||
fun <S : OutputStream, V> S.writeCollection(collection: Collection<V>, writer: S.(V) -> Unit) {
|
||||
writeVarInt(collection.size)
|
||||
|
||||
for (value in collection) {
|
||||
writer(this, value)
|
||||
}
|
||||
}
|
||||
|
||||
fun OutputStream.writeInt(value: Int) {
|
||||
if (this is DataOutput) {
|
||||
writeInt(value)
|
||||
return
|
||||
}
|
||||
|
||||
write(value ushr 24)
|
||||
write(value ushr 16)
|
||||
write(value ushr 8)
|
||||
write(value)
|
||||
}
|
||||
|
||||
fun OutputStream.writeLong(value: Long) {
|
||||
if (this is DataOutput) {
|
||||
writeLong(value)
|
||||
return
|
||||
}
|
||||
|
||||
write((value ushr 48).toInt())
|
||||
write((value ushr 40).toInt())
|
||||
write((value ushr 32).toInt())
|
||||
write((value ushr 24).toInt())
|
||||
write((value ushr 16).toInt())
|
||||
write((value ushr 8).toInt())
|
||||
write(value.toInt())
|
||||
}
|
||||
|
||||
fun OutputStream.writeFloat(value: Float) = writeInt(value.toBits())
|
||||
fun OutputStream.writeDouble(value: Double) = writeLong(value.toBits())
|
||||
|
||||
private fun writeVarLong(write: IntConsumer, value: Long): Int {
|
||||
var value = value
|
||||
var i = 0
|
||||
|
||||
do {
|
||||
val toWrite = (value and 0x7F).toInt()
|
||||
value = value ushr 7
|
||||
|
||||
if (value == 0L)
|
||||
write.accept(toWrite)
|
||||
else
|
||||
write.accept(toWrite or 0x80)
|
||||
|
||||
i++
|
||||
} while (value != 0L)
|
||||
|
||||
return i
|
||||
}
|
||||
|
||||
private fun writeVarInt(write: IntConsumer, value: Int): Int {
|
||||
return writeVarLong(write, value.toLong())
|
||||
}
|
||||
|
||||
private fun Int.toSignedVar(): Int {
|
||||
if (this >= 0)
|
||||
return this shl 1
|
||||
else
|
||||
return (this shl 1) or 1
|
||||
}
|
||||
|
||||
private fun Long.toSignedVar(): Long {
|
||||
if (this >= 0)
|
||||
return this shl 1
|
||||
else
|
||||
return ((-this - 1) shl 1) or 1L
|
||||
}
|
||||
|
||||
fun RandomAccessFile.writeVarLong(value: Long) = writeVarLong(::write, value)
|
||||
fun RandomAccessFile.writeVarInt(value: Int) = writeVarInt(::write, value)
|
||||
fun OutputStream.writeVarLong(value: Long) = writeVarLong(::write, value)
|
||||
fun OutputStream.writeVarInt(value: Int) = writeVarInt(::write, value)
|
||||
|
||||
fun RandomAccessFile.writeSignedVarLong(value: Long) = writeVarLong(::write, value.toSignedVar())
|
||||
fun RandomAccessFile.writeSignedVarInt(value: Int) = writeVarInt(::write, value.toSignedVar())
|
||||
fun OutputStream.writeSignedVarLong(value: Long) = writeVarLong(::write, value.toSignedVar())
|
||||
fun OutputStream.writeSignedVarInt(value: Int) = writeVarInt(::write, value.toSignedVar())
|
||||
|
||||
fun OutputStream.writeBinaryString(input: String) {
|
||||
val bytes = input.encodeToByteArray()
|
||||
writeVarInt(bytes.size)
|
||||
write(bytes)
|
||||
}
|
77
io/src/main/kotlin/ru/dbotthepony/kommons/io/StreamCodecs.kt
Normal file
77
io/src/main/kotlin/ru/dbotthepony/kommons/io/StreamCodecs.kt
Normal file
@ -0,0 +1,77 @@
|
||||
package ru.dbotthepony.kommons.io
|
||||
|
||||
import java.io.DataInputStream
|
||||
import java.io.DataOutputStream
|
||||
import java.util.*
|
||||
import kotlin.reflect.KClass
|
||||
|
||||
class StreamCodec<V>(
|
||||
private val reader: (stream: DataInputStream) -> V,
|
||||
private val writer: (stream: DataOutputStream, value: V) -> Unit,
|
||||
private val copier: ((value: V) -> V) = { it },
|
||||
private val comparator: ((a: V, b: V) -> Boolean) = { a, b -> a == b }
|
||||
) : IStreamCodec<V> {
|
||||
constructor(
|
||||
reader: (stream: DataInputStream) -> V,
|
||||
payloadSize: Long,
|
||||
writer: (stream: DataOutputStream, value: V) -> Unit,
|
||||
copier: ((value: V) -> V) = { it },
|
||||
comparator: ((a: V, b: V) -> Boolean) = { a, b -> a == b }
|
||||
) : this({ stream -> reader.invoke(stream) }, writer, copier, comparator)
|
||||
|
||||
val nullable = object : IStreamCodec<V?> {
|
||||
override fun read(stream: DataInputStream): V? {
|
||||
return if (stream.read() == 0) null else reader.invoke(stream)
|
||||
}
|
||||
|
||||
override fun write(stream: DataOutputStream, value: V?) {
|
||||
if (value === null) stream.write(0) else {
|
||||
stream.write(1)
|
||||
writer.invoke(stream, value)
|
||||
}
|
||||
}
|
||||
|
||||
override fun copy(value: V?): V? {
|
||||
return if (value === null) null else copier.invoke(value)
|
||||
}
|
||||
|
||||
override fun compare(a: V?, b: V?): Boolean {
|
||||
if (a === null && b === null) return true
|
||||
if (a === null || b === null) return false
|
||||
return comparator.invoke(a, b)
|
||||
}
|
||||
}
|
||||
|
||||
override fun read(stream: DataInputStream): V {
|
||||
return reader.invoke(stream)
|
||||
}
|
||||
|
||||
override fun write(stream: DataOutputStream, value: V) {
|
||||
writer.invoke(stream, value)
|
||||
}
|
||||
|
||||
override fun copy(value: V): V {
|
||||
return copier.invoke(value)
|
||||
}
|
||||
|
||||
override fun compare(a: V, b: V): Boolean {
|
||||
return comparator.invoke(a, b)
|
||||
}
|
||||
}
|
||||
|
||||
val NullValueCodec = StreamCodec({ _ -> null }, { _, _ -> })
|
||||
val BooleanValueCodec = StreamCodec(DataInputStream::readBoolean, 1L, DataOutputStream::writeBoolean) { a, b -> a == b }
|
||||
val ByteValueCodec = StreamCodec(DataInputStream::readByte, 1L, { s, v -> s.writeByte(v.toInt()) }) { a, b -> a == b }
|
||||
val ShortValueCodec = StreamCodec(DataInputStream::readShort, 2L, { s, v -> s.writeShort(v.toInt()) }) { a, b -> a == b }
|
||||
val IntValueCodec = StreamCodec(DataInputStream::readInt, 4L, DataOutputStream::writeInt) { a, b -> a == b }
|
||||
val LongValueCodec = StreamCodec(DataInputStream::readLong, 8L, DataOutputStream::writeLong) { a, b -> a == b }
|
||||
val FloatValueCodec = StreamCodec(DataInputStream::readFloat, 4L, DataOutputStream::writeFloat) { a, b -> a == b }
|
||||
val DoubleValueCodec = StreamCodec(DataInputStream::readDouble, 8L, DataOutputStream::writeDouble) { a, b -> a == b }
|
||||
val BigDecimalValueCodec = StreamCodec(DataInputStream::readBigDecimal, DataOutputStream::writeBigDecimal)
|
||||
val UUIDValueCodec = StreamCodec({ s -> UUID(s.readLong(), s.readLong()) }, { s, v -> s.writeLong(v.mostSignificantBits); s.writeLong(v.leastSignificantBits) })
|
||||
val VarIntValueCodec = StreamCodec(DataInputStream::readSignedVarInt, DataOutputStream::writeSignedVarInt) { a, b -> a == b }
|
||||
val VarLongValueCodec = StreamCodec(DataInputStream::readSignedVarLong, DataOutputStream::writeSignedVarLong) { a, b -> a == b }
|
||||
val BinaryStringCodec = StreamCodec(DataInputStream::readBinaryString, DataOutputStream::writeBinaryString)
|
||||
|
||||
fun <E : Enum<E>> Class<E>.codec() = EnumValueCodec(this)
|
||||
fun <E : Enum<E>> KClass<E>.codec() = EnumValueCodec(this.java)
|
40
math/build.gradle.kts
Normal file
40
math/build.gradle.kts
Normal file
@ -0,0 +1,40 @@
|
||||
plugins {
|
||||
kotlin("jvm")
|
||||
`maven-publish`
|
||||
}
|
||||
|
||||
val mathVersion: String by project
|
||||
val projectGroup: String by project
|
||||
|
||||
group = projectGroup
|
||||
version = mathVersion
|
||||
|
||||
repositories {
|
||||
mavenCentral()
|
||||
}
|
||||
|
||||
dependencies {
|
||||
testImplementation("org.jetbrains.kotlin:kotlin-test")
|
||||
}
|
||||
|
||||
tasks.test {
|
||||
useJUnitPlatform()
|
||||
}
|
||||
|
||||
kotlin {
|
||||
jvmToolchain(17)
|
||||
}
|
||||
|
||||
publishing {
|
||||
publications {
|
||||
create<MavenPublication>("mavenJava") {
|
||||
from(components["java"])
|
||||
|
||||
pom {
|
||||
dependencies {
|
||||
implementation(kotlin("stdlib"))
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
1553
math/src/main/kotlin/ru/dbotthepony/kommons/math/Decimal.kt
Normal file
1553
math/src/main/kotlin/ru/dbotthepony/kommons/math/Decimal.kt
Normal file
File diff suppressed because it is too large
Load Diff
53
math/src/main/kotlin/ru/dbotthepony/kommons/math/HSVColor.kt
Normal file
53
math/src/main/kotlin/ru/dbotthepony/kommons/math/HSVColor.kt
Normal file
@ -0,0 +1,53 @@
|
||||
package ru.dbotthepony.kommons.math
|
||||
|
||||
class HSVColor(hue: Float, saturation: Float, value: Float) : Comparable<HSVColor> {
|
||||
val hue = (hue % 360f).let { if (it < 0f) it + 360f else it }
|
||||
val saturation = saturation.coerceIn(0f, 1f)
|
||||
val value = value.coerceIn(0f, 1f)
|
||||
|
||||
override fun equals(other: Any?): Boolean {
|
||||
return other === this || other is HSVColor && other.hue == hue && other.saturation == saturation && other.value == value
|
||||
}
|
||||
|
||||
override fun hashCode(): Int {
|
||||
return hue.hashCode() + saturation.hashCode() * 31 + value.hashCode() * 31 * 31
|
||||
}
|
||||
|
||||
fun copy(hue: Float = this.hue, saturation: Float = this.saturation, value: Float = this.value): HSVColor {
|
||||
return HSVColor(hue, saturation, value)
|
||||
}
|
||||
|
||||
operator fun component1() = hue
|
||||
operator fun component2() = saturation
|
||||
operator fun component3() = value
|
||||
|
||||
fun toRGBA(alpha: Float = 1f): RGBAColor {
|
||||
val valueMin = (1f - saturation) * value
|
||||
val delta = (value - valueMin) * (hue % 60f) / 60f
|
||||
val valueInc = valueMin + delta
|
||||
val valueDec = value - delta
|
||||
|
||||
return when ((hue / 60f).toInt()) {
|
||||
0 -> RGBAColor(value, valueInc, valueMin, alpha)
|
||||
1 -> RGBAColor(valueDec, value, valueMin, alpha)
|
||||
2 -> RGBAColor(valueMin, value, valueInc, alpha)
|
||||
3 -> RGBAColor(valueMin, valueDec, value, alpha)
|
||||
4 -> RGBAColor(valueInc, valueMin, value, alpha)
|
||||
5 -> RGBAColor(value, valueMin, valueDec, alpha)
|
||||
else -> throw IllegalStateException("whut")
|
||||
}
|
||||
}
|
||||
|
||||
override fun compareTo(other: HSVColor): Int {
|
||||
return comparator.compare(this, other)
|
||||
}
|
||||
|
||||
companion object {
|
||||
@JvmField val WHITE = HSVColor(0f, 1f, 1f)
|
||||
|
||||
private val comparator = Comparator
|
||||
.comparing(HSVColor::hue)
|
||||
.thenComparing(HSVColor::saturation)
|
||||
.thenComparing(HSVColor::value)
|
||||
}
|
||||
}
|
299
math/src/main/kotlin/ru/dbotthepony/kommons/math/RGBAColor.kt
Normal file
299
math/src/main/kotlin/ru/dbotthepony/kommons/math/RGBAColor.kt
Normal file
@ -0,0 +1,299 @@
|
||||
package ru.dbotthepony.kommons.math
|
||||
|
||||
import kotlin.math.roundToInt
|
||||
|
||||
private fun hex(value: Int): String {
|
||||
require(value in 0 .. 255)
|
||||
val v = value.toString(16)
|
||||
|
||||
if (v.length == 1)
|
||||
return "0$v"
|
||||
else
|
||||
return v
|
||||
}
|
||||
|
||||
class RGBAColor(red: Float, green: Float, blue: Float, alpha: Float = 1f) : Comparable<RGBAColor> {
|
||||
constructor(r: Int, g: Int, b: Int) : this((r / 255f), (g / 255f), (b / 255f), 1f)
|
||||
constructor(r: Int, g: Int, b: Int, a: Int) : this((r / 255f), (g / 255f), (b / 255f), (a / 255f))
|
||||
constructor(r: Int, g: Int, b: Int, a: Float) : this((r / 255f), (g / 255f), (b / 255f), a)
|
||||
|
||||
val red = red.coerceIn(0f, 1f)
|
||||
val green = green.coerceIn(0f, 1f)
|
||||
val blue = blue.coerceIn(0f, 1f)
|
||||
val alpha = alpha.coerceIn(0f, 1f)
|
||||
|
||||
val redInt get() = (red * 255f).roundToInt()
|
||||
val greenInt get() = (green * 255f).roundToInt()
|
||||
val blueInt get() = (blue * 255f).roundToInt()
|
||||
val alphaInt get() = (alpha * 255f).roundToInt()
|
||||
|
||||
fun toRGBA(): Int {
|
||||
return (redInt shl 24) or (greenInt shl 16) or (blueInt shl 8) or alphaInt
|
||||
}
|
||||
|
||||
fun toARGB(): Int {
|
||||
return (alphaInt shl 24) or (redInt shl 16) or (greenInt shl 8) or blueInt
|
||||
}
|
||||
|
||||
fun toBGRA(): Int {
|
||||
return (blueInt shl 24) or (greenInt shl 16) or (redInt shl 8) or alphaInt
|
||||
}
|
||||
|
||||
val isFullyTransparent get() = alpha <= 0f
|
||||
val isWhite: Boolean get() = red >= 1f && green >= 1f && blue >= 1f && alpha >= 1f
|
||||
|
||||
fun canRepresentHue(): Boolean {
|
||||
val min = red.coerceAtMost(green).coerceAtMost(blue)
|
||||
val max = red.coerceAtLeast(green).coerceAtLeast(blue)
|
||||
return min != max
|
||||
}
|
||||
|
||||
fun hue(ifNoHue: Float = 0f): Float {
|
||||
val min = red.coerceAtMost(green).coerceAtMost(blue)
|
||||
val max = red.coerceAtLeast(green).coerceAtLeast(blue)
|
||||
|
||||
if (min == max) {
|
||||
return ifNoHue
|
||||
}
|
||||
|
||||
val diff = max - min
|
||||
|
||||
return if (max == red && green >= blue) {
|
||||
60f * (green - blue) / diff
|
||||
} else if (max == red) {
|
||||
60f * (green - blue) / diff + 360f
|
||||
} else if (max == green) {
|
||||
60f * (blue - red) / diff + 120f
|
||||
} else if (max == blue) {
|
||||
60f * (red - green) / diff + 240f
|
||||
} else {
|
||||
throw IllegalStateException("Whut $red $green $blue ($min / $max)")
|
||||
}
|
||||
}
|
||||
|
||||
fun toHSV(): HSVColor {
|
||||
val min = red.coerceAtMost(green).coerceAtMost(blue)
|
||||
val max = red.coerceAtLeast(green).coerceAtLeast(blue)
|
||||
|
||||
if (min == max) {
|
||||
return HSVColor(0f, if (max == 0f) 0f else 1f - min / max, max)
|
||||
}
|
||||
|
||||
val diff = max - min
|
||||
|
||||
val hue = if (max == red && green >= blue) {
|
||||
60f * (green - blue) / diff
|
||||
} else if (max == red) {
|
||||
60f * (green - blue) / diff + 360f
|
||||
} else if (max == green) {
|
||||
60f * (blue - red) / diff + 120f
|
||||
} else if (max == blue) {
|
||||
60f * (red - green) / diff + 240f
|
||||
} else {
|
||||
throw IllegalStateException("Whut $red $green $blue ($min / $max)")
|
||||
}
|
||||
|
||||
return HSVColor(hue, 1f - min / max, max)
|
||||
}
|
||||
|
||||
fun toHexStringRGB(): String {
|
||||
return "#${hex(redInt)}${hex(greenInt)}${hex(blueInt)}"
|
||||
}
|
||||
|
||||
fun toHexStringRGBA(): String {
|
||||
return "#${hex(redInt)}${hex(greenInt)}${hex(blueInt)}${hex(alphaInt)}"
|
||||
}
|
||||
|
||||
fun toHexStringARGB(): String {
|
||||
return "#${hex(alphaInt)}${hex(redInt)}${hex(greenInt)}${hex(blueInt)}"
|
||||
}
|
||||
|
||||
override fun toString(): String {
|
||||
return "RGBAColor[$red $green $blue $alpha]"
|
||||
}
|
||||
|
||||
operator fun component1() = red
|
||||
operator fun component2() = green
|
||||
operator fun component3() = blue
|
||||
operator fun component4() = alpha
|
||||
|
||||
fun toIntInv(): Int {
|
||||
return (blueInt shl 16) or (greenInt shl 8) or redInt
|
||||
}
|
||||
|
||||
fun toRGB(): Int {
|
||||
return (redInt shl 16) or (greenInt shl 8) or blueInt
|
||||
}
|
||||
|
||||
fun copy(red: Float = this.red, green: Float = this.green, blue: Float = this.blue, alpha: Float = this.alpha): RGBAColor {
|
||||
return RGBAColor(red, green, blue, alpha)
|
||||
}
|
||||
|
||||
override fun compareTo(other: RGBAColor): Int {
|
||||
if (canRepresentHue() && other.canRepresentHue())
|
||||
return hue().compareTo(other.hue()).let {
|
||||
if (it != 0)
|
||||
it
|
||||
else
|
||||
toHSV().compareTo(other.toHSV())
|
||||
}
|
||||
|
||||
return comparator.compare(this, other)
|
||||
}
|
||||
|
||||
override fun equals(other: Any?): Boolean {
|
||||
return other === this || other is RGBAColor && comparator.compare(this, other) == 0
|
||||
}
|
||||
|
||||
override fun hashCode(): Int {
|
||||
return red.hashCode() + green.hashCode() * 31 + blue.hashCode() * 31 * 31 + alpha.hashCode() * 31 * 31 * 31
|
||||
}
|
||||
|
||||
/*fun linearInterpolation(t: Float, other: RGBAColor, interpolateAlpha: Boolean = true): RGBAColor {
|
||||
return RGBAColor(
|
||||
linearInterpolation(t, red, other.red),
|
||||
linearInterpolation(t, green, other.green),
|
||||
linearInterpolation(t, blue, other.blue),
|
||||
if (interpolateAlpha) linearInterpolation(t, alpha, other.alpha) else alpha,
|
||||
)
|
||||
}*/
|
||||
|
||||
operator fun times(other: RGBAColor): RGBAColor {
|
||||
if (isWhite)
|
||||
return other
|
||||
else if (other.isWhite)
|
||||
return this
|
||||
|
||||
return RGBAColor(red * other.red, green * other.green, blue * other.blue, alpha * other.alpha)
|
||||
}
|
||||
|
||||
@Suppress("unused")
|
||||
companion object {
|
||||
private val comparator = Comparator
|
||||
.comparing(RGBAColor::red)
|
||||
.thenComparing(RGBAColor::green)
|
||||
.thenComparing(RGBAColor::blue)
|
||||
.thenComparing(RGBAColor::alpha)
|
||||
|
||||
@JvmField val TRANSPARENT_BLACK = RGBAColor(0f, 0f, 0f, 0f)
|
||||
@JvmField val TRANSPARENT_WHITE = RGBAColor(1f, 1f, 1f, 0f)
|
||||
|
||||
@JvmField val BLACK = RGBAColor(0f, 0f, 0f)
|
||||
@JvmField val WHITE = RGBAColor(1f, 1f, 1f)
|
||||
@JvmField val RED = RGBAColor(1f, 0f, 0f)
|
||||
@JvmField val GREEN = RGBAColor(0f, 1f, 0f)
|
||||
@JvmField val LIGHT_GREEN = RGBAColor(136, 255, 124)
|
||||
@JvmField val SLATE_GRAY = RGBAColor(64, 64, 64)
|
||||
@JvmField val GRAY = rgb(0x2C2C2CL)
|
||||
@JvmField val LIGHT_GRAY = rgb(0x7F7F7FL)
|
||||
@JvmField val HEADING_TEXT = rgb(0x404040L)
|
||||
|
||||
@JvmField val LOW_POWER = RGBAColor(173, 41, 41)
|
||||
@JvmField val FULL_POWER = RGBAColor(255, 242, 40)
|
||||
@JvmField val LOW_MATTER = RGBAColor(0, 24, 148)
|
||||
@JvmField val FULL_MATTER = RGBAColor(72, 90, 255)
|
||||
@JvmField val LOW_PATTERNS = RGBAColor(44, 104, 57)
|
||||
@JvmField val FULL_PATTERNS = RGBAColor(65, 255, 87)
|
||||
|
||||
@JvmField val HALF_TRANSPARENT = RGBAColor(1f, 1f, 1f, 0.5f)
|
||||
@JvmField val REDDISH = RGBAColor(1f, 0.4f, 0.4f)
|
||||
|
||||
fun rgb(color: Int): RGBAColor {
|
||||
val r = (color and 0xFF0000 ushr 16) / 255f
|
||||
val g = (color and 0xFF00 ushr 8) / 255f
|
||||
val b = (color and 0xFF) / 255f
|
||||
return RGBAColor(r, g, b)
|
||||
}
|
||||
|
||||
fun rgb(color: Long): RGBAColor {
|
||||
val r = (color and 0xFF0000 ushr 16) / 255f
|
||||
val g = (color and 0xFF00 ushr 8) / 255f
|
||||
val b = (color and 0xFF) / 255f
|
||||
return RGBAColor(r, g, b)
|
||||
}
|
||||
|
||||
fun bgr(color: Int): RGBAColor {
|
||||
val r = (color and 0xFF0000 ushr 16) / 255f
|
||||
val g = (color and 0xFF00 ushr 8) / 255f
|
||||
val b = (color and 0xFF) / 255f
|
||||
return RGBAColor(b, g, r)
|
||||
}
|
||||
|
||||
fun bgr(color: Long): RGBAColor {
|
||||
val r = (color and 0xFF0000 ushr 16) / 255f
|
||||
val g = (color and 0xFF00 ushr 8) / 255f
|
||||
val b = (color and 0xFF) / 255f
|
||||
return RGBAColor(b, g, r)
|
||||
}
|
||||
|
||||
fun abgr(color: Int): RGBAColor {
|
||||
val r = (color and -0x1000000 ushr 24) / 255f
|
||||
val g = (color and 0xFF0000 ushr 16) / 255f
|
||||
val b = (color and 0xFF00 ushr 8) / 255f
|
||||
val a = (color and 0xFF) / 255f
|
||||
return RGBAColor(a, b, g, r)
|
||||
}
|
||||
|
||||
fun argb(color: Int): RGBAColor {
|
||||
val a = (color and -0x1000000 ushr 24) / 255f
|
||||
val r = (color and 0xFF0000 ushr 16) / 255f
|
||||
val g = (color and 0xFF00 ushr 8) / 255f
|
||||
val b = (color and 0xFF) / 255f
|
||||
return RGBAColor(r, g, b, a)
|
||||
}
|
||||
|
||||
private val hexChars = HashSet<Char>()
|
||||
|
||||
init {
|
||||
"#0123456789abcdefABCDEF".forEach { hexChars.add(it) }
|
||||
}
|
||||
|
||||
fun isHexCharacter(char: Char): Boolean {
|
||||
return char in hexChars
|
||||
}
|
||||
|
||||
private val shorthandRGBHex = Regex("#?([0-9abcdef])([0-9abcdef])([0-9abcdef])", RegexOption.IGNORE_CASE)
|
||||
private val longhandRGBHex = Regex("#?([0-9abcdef]{2})([0-9abcdef]{2})([0-9abcdef]{2})", RegexOption.IGNORE_CASE)
|
||||
|
||||
private val shorthandRGBAHex = Regex("#?([0-9abcdef])([0-9abcdef])([0-9abcdef])([0-9abcdef])", RegexOption.IGNORE_CASE)
|
||||
private val longhandRGBAHex = Regex("#?([0-9abcdef]{2})([0-9abcdef]{2})([0-9abcdef]{2})([0-9abcdef]{2})", RegexOption.IGNORE_CASE)
|
||||
|
||||
fun fromHexStringRGB(value: String): RGBAColor? {
|
||||
if (value.length == 3 || value.length == 4) {
|
||||
val match = shorthandRGBHex.find(value) ?: return null
|
||||
val red = match.groupValues[1].toIntOrNull(16) ?: return null
|
||||
val green = match.groupValues[2].toIntOrNull(16) ?: return null
|
||||
val blue = match.groupValues[3].toIntOrNull(16) ?: return null
|
||||
return RGBAColor(red * 16, green * 16, blue * 16)
|
||||
} else if (value.length == 6 || value.length == 7) {
|
||||
val match = longhandRGBHex.find(value) ?: return null
|
||||
val red = match.groupValues[1].toIntOrNull(16) ?: return null
|
||||
val green = match.groupValues[2].toIntOrNull(16) ?: return null
|
||||
val blue = match.groupValues[3].toIntOrNull(16) ?: return null
|
||||
return RGBAColor(red, green, blue)
|
||||
} else {
|
||||
return null
|
||||
}
|
||||
}
|
||||
|
||||
fun fromHexStringRGBA(value: String): RGBAColor? {
|
||||
if (value.length == 4 || value.length == 5) {
|
||||
val match = shorthandRGBAHex.find(value) ?: return null
|
||||
val red = match.groupValues[1].toIntOrNull(16) ?: return null
|
||||
val green = match.groupValues[2].toIntOrNull(16) ?: return null
|
||||
val blue = match.groupValues[3].toIntOrNull(16) ?: return null
|
||||
val alpha = match.groupValues[4].toIntOrNull(16) ?: return null
|
||||
return RGBAColor(red * 16, green * 16, blue * 16, alpha * 16)
|
||||
} else if (value.length == 8 || value.length == 9) {
|
||||
val match = longhandRGBAHex.find(value) ?: return null
|
||||
val red = match.groupValues[1].toIntOrNull(16) ?: return null
|
||||
val green = match.groupValues[2].toIntOrNull(16) ?: return null
|
||||
val blue = match.groupValues[3].toIntOrNull(16) ?: return null
|
||||
val alpha = match.groupValues[4].toIntOrNull(16) ?: return null
|
||||
return RGBAColor(red, green, blue, alpha)
|
||||
} else {
|
||||
return null
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
47
networking/build.gradle.kts
Normal file
47
networking/build.gradle.kts
Normal file
@ -0,0 +1,47 @@
|
||||
|
||||
plugins {
|
||||
kotlin("jvm")
|
||||
`maven-publish`
|
||||
}
|
||||
|
||||
val networkingVersion: String by project
|
||||
val projectGroup: String by project
|
||||
|
||||
group = projectGroup
|
||||
version = networkingVersion
|
||||
|
||||
repositories {
|
||||
mavenCentral()
|
||||
}
|
||||
|
||||
dependencies {
|
||||
testImplementation("org.jetbrains.kotlin:kotlin-test")
|
||||
implementation(project(":core"))
|
||||
implementation(project(":io"))
|
||||
implementation(project(":collect"))
|
||||
}
|
||||
|
||||
tasks.test {
|
||||
useJUnitPlatform()
|
||||
}
|
||||
|
||||
kotlin {
|
||||
jvmToolchain(17)
|
||||
}
|
||||
|
||||
publishing {
|
||||
publications {
|
||||
create<MavenPublication>("mavenJava") {
|
||||
from(components["java"])
|
||||
|
||||
pom {
|
||||
dependencies {
|
||||
implementation(kotlin("stdlib"))
|
||||
implementation(project(":core"))
|
||||
implementation(project(":io"))
|
||||
implementation(project(":collect"))
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,5 @@
|
||||
package ru.dbotthepony.kommons.networking
|
||||
|
||||
enum class ChangesetAction {
|
||||
CLEAR, ADD, REMOVE
|
||||
}
|
@ -0,0 +1,41 @@
|
||||
package ru.dbotthepony.kommons.networking
|
||||
|
||||
interface FieldAccess<V> {
|
||||
fun read(): V
|
||||
fun write(value: V)
|
||||
}
|
||||
|
||||
interface FloatFieldAccess : FieldAccess<Float> {
|
||||
override fun write(value: Float)
|
||||
@Deprecated("Use type specific method", replaceWith = ReplaceWith("this.readFloat()"))
|
||||
override fun read() = readFloat()
|
||||
fun readFloat(): Float
|
||||
}
|
||||
|
||||
interface DoubleFieldAccess : FieldAccess<Double> {
|
||||
override fun write(value: Double)
|
||||
@Deprecated("Use type specific method", replaceWith = ReplaceWith("this.readDouble()"))
|
||||
override fun read() = readDouble()
|
||||
fun readDouble(): Double
|
||||
}
|
||||
|
||||
interface IntFieldAccess : FieldAccess<Int> {
|
||||
override fun write(value: Int)
|
||||
@Deprecated("Use type specific method", replaceWith = ReplaceWith("this.readInt()"))
|
||||
override fun read() = readInt()
|
||||
fun readInt(): Int
|
||||
}
|
||||
|
||||
interface LongFieldAccess : FieldAccess<Long> {
|
||||
override fun write(value: Long)
|
||||
@Deprecated("Use type specific method", replaceWith = ReplaceWith("this.readLong()"))
|
||||
override fun read() = readLong()
|
||||
fun readLong(): Long
|
||||
}
|
||||
|
||||
interface BooleanFieldAccess : FieldAccess<Boolean> {
|
||||
override fun write(value: Boolean)
|
||||
@Deprecated("Use type specific method", replaceWith = ReplaceWith("this.readBoolean()"))
|
||||
override fun read() = readBoolean()
|
||||
fun readBoolean(): Boolean
|
||||
}
|
@ -0,0 +1,50 @@
|
||||
package ru.dbotthepony.kommons.networking
|
||||
|
||||
fun interface FieldGetter<V> {
|
||||
fun invoke(field: FieldAccess<V>): V
|
||||
}
|
||||
|
||||
fun interface FloatFieldGetter : FieldGetter<Float> {
|
||||
fun invoke(field: FloatFieldAccess): Float
|
||||
|
||||
@Deprecated("Use type specific invoke")
|
||||
override fun invoke(field: FieldAccess<Float>): Float {
|
||||
return invoke(field as FloatFieldAccess)
|
||||
}
|
||||
}
|
||||
|
||||
fun interface DoubleFieldGetter : FieldGetter<Double> {
|
||||
fun invoke(field: DoubleFieldAccess): Double
|
||||
|
||||
@Deprecated("Use type specific invoke")
|
||||
override fun invoke(field: FieldAccess<Double>): Double {
|
||||
return invoke(field as DoubleFieldAccess)
|
||||
}
|
||||
}
|
||||
|
||||
fun interface IntFieldGetter : FieldGetter<Int> {
|
||||
fun invoke(field: IntFieldAccess): Int
|
||||
|
||||
@Deprecated("Use type specific invoke")
|
||||
override fun invoke(field: FieldAccess<Int>): Int {
|
||||
return invoke(field as IntFieldAccess)
|
||||
}
|
||||
}
|
||||
|
||||
fun interface LongFieldGetter : FieldGetter<Long> {
|
||||
fun invoke(field: LongFieldAccess): Long
|
||||
|
||||
@Deprecated("Use type specific invoke")
|
||||
override fun invoke(field: FieldAccess<Long>): Long {
|
||||
return invoke(field as LongFieldAccess)
|
||||
}
|
||||
}
|
||||
|
||||
fun interface BooleanFieldGetter : FieldGetter<Boolean> {
|
||||
fun invoke(field: BooleanFieldAccess): Boolean
|
||||
|
||||
@Deprecated("Use type specific invoke")
|
||||
override fun invoke(field: FieldAccess<Boolean>): Boolean {
|
||||
return invoke(field as BooleanFieldAccess)
|
||||
}
|
||||
}
|
@ -0,0 +1,50 @@
|
||||
package ru.dbotthepony.kommons.networking
|
||||
|
||||
fun interface FieldSetter<V> {
|
||||
fun invoke(value: V, access: FieldAccess<V>, setByRemote: Boolean)
|
||||
}
|
||||
|
||||
fun interface FloatFieldSetter : FieldSetter<Float> {
|
||||
fun invoke(value: Float, access: FloatFieldAccess, setByRemote: Boolean)
|
||||
|
||||
@Deprecated("Use type specific invoke")
|
||||
override fun invoke(value: Float, access: FieldAccess<Float>, setByRemote: Boolean) {
|
||||
invoke(value, access as FloatFieldAccess, setByRemote)
|
||||
}
|
||||
}
|
||||
|
||||
fun interface DoubleFieldSetter : FieldSetter<Double> {
|
||||
fun invoke(value: Double, access: DoubleFieldAccess, setByRemote: Boolean)
|
||||
|
||||
@Deprecated("Use type specific invoke")
|
||||
override fun invoke(value: Double, access: FieldAccess<Double>, setByRemote: Boolean) {
|
||||
invoke(value, access as DoubleFieldAccess, setByRemote)
|
||||
}
|
||||
}
|
||||
|
||||
fun interface IntFieldSetter : FieldSetter<Int> {
|
||||
fun invoke(value: Int, access: IntFieldAccess, setByRemote: Boolean)
|
||||
|
||||
@Deprecated("Use type specific invoke")
|
||||
override fun invoke(value: Int, access: FieldAccess<Int>, setByRemote: Boolean) {
|
||||
invoke(value, access as IntFieldAccess, setByRemote)
|
||||
}
|
||||
}
|
||||
|
||||
fun interface LongFieldSetter : FieldSetter<Long> {
|
||||
fun invoke(value: Long, access: LongFieldAccess, setByRemote: Boolean)
|
||||
|
||||
@Deprecated("Use type specific invoke")
|
||||
override fun invoke(value: Long, access: FieldAccess<Long>, setByRemote: Boolean) {
|
||||
invoke(value, access as LongFieldAccess, setByRemote)
|
||||
}
|
||||
}
|
||||
|
||||
fun interface BooleanFieldSetter : FieldSetter<Boolean> {
|
||||
fun invoke(value: Boolean, access: BooleanFieldAccess, setByRemote: Boolean)
|
||||
|
||||
@Deprecated("Use type specific invoke")
|
||||
override fun invoke(value: Boolean, access: FieldAccess<Boolean>, setByRemote: Boolean) {
|
||||
invoke(value, access as BooleanFieldAccess, setByRemote)
|
||||
}
|
||||
}
|
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,173 @@
|
||||
package ru.dbotthepony.kommons.networking
|
||||
|
||||
import ru.dbotthepony.kommons.event.IBooleanSubscriptable
|
||||
import ru.dbotthepony.kommons.event.IDoubleSubcripable
|
||||
import ru.dbotthepony.kommons.event.IFloatSubcripable
|
||||
import ru.dbotthepony.kommons.event.IIntSubcripable
|
||||
import ru.dbotthepony.kommons.event.ILongSubcripable
|
||||
import ru.dbotthepony.kommons.event.ISubscriptable
|
||||
import ru.dbotthepony.kommons.util.FloatSupplier
|
||||
import java.io.DataInputStream
|
||||
import java.io.DataOutputStream
|
||||
import java.util.function.BooleanSupplier
|
||||
import java.util.function.DoubleSupplier
|
||||
import java.util.function.IntSupplier
|
||||
import java.util.function.LongSupplier
|
||||
import java.util.function.Supplier
|
||||
import kotlin.properties.ReadOnlyProperty
|
||||
import kotlin.reflect.KProperty
|
||||
|
||||
sealed interface IField<V> : ReadOnlyProperty<Any?, V>, Supplier<V>, () -> V, ISubscriptable<V> {
|
||||
fun observe(): Boolean
|
||||
fun markDirty()
|
||||
fun markDirty(endpoint: FieldSynchronizer.Endpoint)
|
||||
val value: V
|
||||
val isRemoved: Boolean
|
||||
|
||||
fun remove()
|
||||
|
||||
fun write(stream: DataOutputStream, endpoint: FieldSynchronizer.Endpoint)
|
||||
fun read(stream: DataInputStream)
|
||||
|
||||
override fun getValue(thisRef: Any?, property: KProperty<*>): V {
|
||||
return this.value
|
||||
}
|
||||
|
||||
override fun get(): V {
|
||||
return value
|
||||
}
|
||||
|
||||
override fun invoke(): V {
|
||||
return value
|
||||
}
|
||||
}
|
||||
|
||||
interface IFloatProperty {
|
||||
operator fun getValue(thisRef: Any?, property: KProperty<*>): Float
|
||||
}
|
||||
|
||||
sealed interface IFloatField : IField<Float>, FloatSupplier, IFloatSubcripable {
|
||||
val float: Float
|
||||
val property: IFloatProperty
|
||||
|
||||
override val value: Float
|
||||
get() = float
|
||||
|
||||
@Deprecated("Use type specific Supplier interface")
|
||||
override fun get(): Float {
|
||||
return float
|
||||
}
|
||||
|
||||
override fun getAsFloat(): Float {
|
||||
return float
|
||||
}
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.property"))
|
||||
override fun getValue(thisRef: Any?, property: KProperty<*>): Float {
|
||||
return float
|
||||
}
|
||||
}
|
||||
|
||||
interface IDoubleProperty {
|
||||
operator fun getValue(thisRef: Any?, property: KProperty<*>): Double
|
||||
}
|
||||
|
||||
sealed interface IDoubleField : IField<Double>, DoubleSupplier, IDoubleSubcripable {
|
||||
val double: Double
|
||||
val property: IDoubleProperty
|
||||
|
||||
@Deprecated("Use type specific Supplier interface")
|
||||
override fun get(): Double {
|
||||
return double
|
||||
}
|
||||
|
||||
override val value: Double
|
||||
get() = double
|
||||
|
||||
override fun getAsDouble(): Double {
|
||||
return double
|
||||
}
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.property"))
|
||||
override fun getValue(thisRef: Any?, property: KProperty<*>): Double {
|
||||
return double
|
||||
}
|
||||
}
|
||||
|
||||
interface IIntProperty {
|
||||
operator fun getValue(thisRef: Any?, property: KProperty<*>): Int
|
||||
}
|
||||
|
||||
sealed interface IIntField : IField<Int>, IntSupplier, IIntSubcripable {
|
||||
val int: Int
|
||||
val property: IIntProperty
|
||||
|
||||
@Deprecated("Use type specific Supplier interface")
|
||||
override fun get(): Int {
|
||||
return int
|
||||
}
|
||||
|
||||
override val value: Int
|
||||
get() = int
|
||||
|
||||
override fun getAsInt(): Int {
|
||||
return int
|
||||
}
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.property"))
|
||||
override fun getValue(thisRef: Any?, property: KProperty<*>): Int {
|
||||
return int
|
||||
}
|
||||
}
|
||||
|
||||
interface ILongProperty {
|
||||
operator fun getValue(thisRef: Any?, property: KProperty<*>): Long
|
||||
}
|
||||
|
||||
sealed interface ILongField : IField<Long>, LongSupplier, ILongSubcripable {
|
||||
val long: Long
|
||||
val property: ILongProperty
|
||||
|
||||
@Deprecated("Use type specific Supplier interface")
|
||||
override fun get(): Long {
|
||||
return long
|
||||
}
|
||||
|
||||
override val value: Long
|
||||
get() = long
|
||||
|
||||
override fun getAsLong(): Long {
|
||||
return long
|
||||
}
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.property"))
|
||||
override fun getValue(thisRef: Any?, property: KProperty<*>): Long {
|
||||
return long
|
||||
}
|
||||
}
|
||||
|
||||
interface IBooleanProperty {
|
||||
operator fun getValue(thisRef: Any?, property: KProperty<*>): Boolean
|
||||
}
|
||||
|
||||
sealed interface IBooleanField : IField<Boolean>, BooleanSupplier, IBooleanSubscriptable {
|
||||
val boolean: Boolean
|
||||
val property: IBooleanProperty
|
||||
|
||||
@Deprecated("Use type specific Supplier interface")
|
||||
override fun get(): Boolean {
|
||||
return boolean
|
||||
}
|
||||
|
||||
override val value: Boolean
|
||||
get() = boolean
|
||||
|
||||
override fun getAsBoolean(): Boolean {
|
||||
return boolean
|
||||
}
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.property"))
|
||||
override fun getValue(thisRef: Any?, property: KProperty<*>): Boolean {
|
||||
return boolean
|
||||
}
|
||||
}
|
@ -0,0 +1,23 @@
|
||||
package ru.dbotthepony.kommons.networking
|
||||
|
||||
data class MapChangeset<out K, out V>(
|
||||
val action: ChangesetAction,
|
||||
val key: K?,
|
||||
val value: V?
|
||||
) {
|
||||
inline fun map(add: (K, V) -> Unit, remove: (K) -> Unit) {
|
||||
when (action) {
|
||||
ChangesetAction.ADD -> add.invoke(key!!, value!!)
|
||||
ChangesetAction.REMOVE -> remove.invoke(key!!)
|
||||
else -> {}
|
||||
}
|
||||
}
|
||||
|
||||
inline fun map(add: (K, V) -> Unit, remove: (K) -> Unit, clear: () -> Unit) {
|
||||
when (action) {
|
||||
ChangesetAction.CLEAR -> clear.invoke()
|
||||
ChangesetAction.ADD -> add.invoke(key!!, value!!)
|
||||
ChangesetAction.REMOVE -> remove.invoke(key!!)
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,165 @@
|
||||
package ru.dbotthepony.kommons.networking
|
||||
|
||||
import ru.dbotthepony.kommons.util.FloatConsumer
|
||||
import ru.dbotthepony.kommons.core.SentientGetterSetter
|
||||
import ru.dbotthepony.kommons.util.BooleanConsumer
|
||||
import java.util.function.DoubleConsumer
|
||||
import java.util.function.IntConsumer
|
||||
import java.util.function.LongConsumer
|
||||
import kotlin.reflect.KProperty
|
||||
|
||||
sealed interface IMutableField<V> : IField<V>, SentientGetterSetter<V> {
|
||||
override fun getValue(thisRef: Any?, property: KProperty<*>): V {
|
||||
return this.value
|
||||
}
|
||||
|
||||
override var value: V
|
||||
|
||||
override fun accept(t: V) {
|
||||
value = t
|
||||
}
|
||||
|
||||
override fun invoke(): V {
|
||||
return this.value
|
||||
}
|
||||
}
|
||||
|
||||
interface IMutableFloatProperty : IFloatProperty {
|
||||
operator fun setValue(thisRef: Any?, property: KProperty<*>, value: Float)
|
||||
}
|
||||
|
||||
sealed interface IMutableFloatField : IMutableField<Float>, IFloatField, FloatConsumer {
|
||||
override var float: Float
|
||||
override val property: IMutableFloatProperty
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.float"))
|
||||
override var value: Float
|
||||
get() = float
|
||||
set(value) { float = value }
|
||||
|
||||
override fun accept(t: Float) {
|
||||
float = t
|
||||
}
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.property"))
|
||||
override fun getValue(thisRef: Any?, property: KProperty<*>): Float {
|
||||
return float
|
||||
}
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.property"))
|
||||
override fun setValue(thisRef: Any?, property: KProperty<*>, value: Float) {
|
||||
float = value
|
||||
}
|
||||
}
|
||||
|
||||
interface IMutableDoubleProperty : IDoubleProperty {
|
||||
operator fun setValue(thisRef: Any?, property: KProperty<*>, value: Double)
|
||||
}
|
||||
|
||||
sealed interface IMutableDoubleField : IMutableField<Double>, IDoubleField, DoubleConsumer {
|
||||
override var double: Double
|
||||
override val property: IMutableDoubleProperty
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.double"))
|
||||
override var value: Double
|
||||
get() = double
|
||||
set(value) { double = value }
|
||||
|
||||
override fun accept(t: Double) {
|
||||
double = t
|
||||
}
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.property"))
|
||||
override fun getValue(thisRef: Any?, property: KProperty<*>): Double {
|
||||
return double
|
||||
}
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.property"))
|
||||
override fun setValue(thisRef: Any?, property: KProperty<*>, value: Double) {
|
||||
double = value
|
||||
}
|
||||
}
|
||||
|
||||
interface IMutableIntProperty : IIntProperty {
|
||||
operator fun setValue(thisRef: Any?, property: KProperty<*>, value: Int)
|
||||
}
|
||||
|
||||
sealed interface IMutableIntField : IMutableField<Int>, IIntField, IntConsumer {
|
||||
override var int: Int
|
||||
override val property: IMutableIntProperty
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.int"))
|
||||
override var value: Int
|
||||
get() = int
|
||||
set(value) { int = value }
|
||||
|
||||
override fun accept(t: Int) {
|
||||
int = t
|
||||
}
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.property"))
|
||||
override fun getValue(thisRef: Any?, property: KProperty<*>): Int {
|
||||
return int
|
||||
}
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.property"))
|
||||
override fun setValue(thisRef: Any?, property: KProperty<*>, value: Int) {
|
||||
int = value
|
||||
}
|
||||
}
|
||||
|
||||
interface IMutableLongProperty : ILongProperty {
|
||||
operator fun setValue(thisRef: Any?, property: KProperty<*>, value: Long)
|
||||
}
|
||||
|
||||
sealed interface IMutableLongField : IMutableField<Long>, ILongField, LongConsumer {
|
||||
override var long: Long
|
||||
override val property: IMutableLongProperty
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.long"))
|
||||
override var value: Long
|
||||
get() = long
|
||||
set(value) { long = value }
|
||||
|
||||
override fun accept(t: Long) {
|
||||
long = t
|
||||
}
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.property"))
|
||||
override fun getValue(thisRef: Any?, property: KProperty<*>): Long {
|
||||
return long
|
||||
}
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.property"))
|
||||
override fun setValue(thisRef: Any?, property: KProperty<*>, value: Long) {
|
||||
long = value
|
||||
}
|
||||
}
|
||||
|
||||
interface IMutableBooleanProperty : IBooleanProperty {
|
||||
operator fun setValue(thisRef: Any?, property: KProperty<*>, value: Boolean)
|
||||
}
|
||||
|
||||
sealed interface IMutableBooleanField : IMutableField<Boolean>, IBooleanField, BooleanConsumer {
|
||||
override var boolean: Boolean
|
||||
override val property: IMutableBooleanProperty
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.boolean"))
|
||||
override var value: Boolean
|
||||
get() = boolean
|
||||
set(value) { boolean = value }
|
||||
|
||||
override fun accept(t: Boolean) {
|
||||
boolean = t
|
||||
}
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.property"))
|
||||
override fun getValue(thisRef: Any?, property: KProperty<*>): Boolean {
|
||||
return boolean
|
||||
}
|
||||
|
||||
@Deprecated("Use type specific property", replaceWith = ReplaceWith("this.property"))
|
||||
override fun setValue(thisRef: Any?, property: KProperty<*>, value: Boolean) {
|
||||
boolean = value
|
||||
}
|
||||
}
|
@ -0,0 +1,22 @@
|
||||
package ru.dbotthepony.kommons.networking
|
||||
|
||||
data class SetChangeset<out V>(
|
||||
val action: ChangesetAction,
|
||||
val value: V?
|
||||
) {
|
||||
inline fun map(add: (V) -> Unit, remove: (V) -> Unit) {
|
||||
when (action) {
|
||||
ChangesetAction.ADD -> add.invoke(value!!)
|
||||
ChangesetAction.REMOVE -> remove.invoke(value!!)
|
||||
else -> {}
|
||||
}
|
||||
}
|
||||
|
||||
inline fun map(add: (V) -> Unit, remove: (V) -> Unit, clear: () -> Unit) {
|
||||
when (action) {
|
||||
ChangesetAction.CLEAR -> clear.invoke()
|
||||
ChangesetAction.ADD -> add.invoke(value!!)
|
||||
ChangesetAction.REMOVE -> remove.invoke(value!!)
|
||||
}
|
||||
}
|
||||
}
|
14
settings.gradle.kts
Normal file
14
settings.gradle.kts
Normal file
@ -0,0 +1,14 @@
|
||||
|
||||
plugins {
|
||||
//kotlin("jvm") version "1.9.21"
|
||||
id("org.gradle.toolchains.foojay-resolver-convention") version "0.5.0"
|
||||
}
|
||||
|
||||
rootProject.name = "kommons"
|
||||
include("core")
|
||||
include("networking")
|
||||
include("io")
|
||||
include("math")
|
||||
include("io-math")
|
||||
include("collect")
|
||||
include("guava")
|
Loading…
Reference in New Issue
Block a user