Compare commits

..

11 Commits

7 changed files with 146 additions and 21 deletions

View File

@@ -1,5 +1,19 @@
# Changelog # Changelog
## 0.20.33
* `Colors`
* `Common`:
* Add opportunity to use `HEXAColor` with `ahex` colors
## 0.20.32
* `Versions`:
* `Okio`: `3.7.0` -> `3.8.0`
* `Resources`:
* Make `StringResource` serializable
* Add several variants of builder usages
## 0.20.31 ## 0.20.31
* `Versions`: * `Versions`:

View File

@@ -12,16 +12,30 @@ import kotlin.math.floor
* * Red (0.5 capacity): `0xff000088u` * * Red (0.5 capacity): `0xff000088u`
* *
* Anyway it is recommended to use * Anyway it is recommended to use
*
* @param hexaUInt rgba [UInt] in format `0xFFEEBBAA` where FF - red, EE - green, BB - blue` and AA - alpha
*/ */
@Serializable @Serializable
@JvmInline @JvmInline
value class HEXAColor ( value class HEXAColor (
val uint: UInt val hexaUInt: UInt
) : Comparable<HEXAColor> { ) : Comparable<HEXAColor> {
/**
* @returns [hexaUInt] as a string with format `#FFEEBBAA` where FF - red, EE - green, BB - blue and AA - alpha
*/
val hexa: String val hexa: String
get() = "#${uint.toString(16).padStart(8, '0')}" get() = "#${hexaUInt.toString(16).padStart(8, '0')}"
/**
* @returns [hexaUInt] as a string with format `#FFEEBB` where FF - red, EE - green and BB - blue
*/
val hex: String val hex: String
get() = hexa.take(7) get() = hexa.take(7)
/**
* @returns [hexaUInt] as a string with format `#AAFFEEBB` where AA - alpha, FF - red, EE - green and BB - blue
*/
val ahex: String
get() = "#${a.toString(16).padStart(2, '2')}${hex.drop(1)}"
val rgba: String val rgba: String
get() = "rgba($r,$g,$b,${aOfOne.toString().take(5)})" get() = "rgba($r,$g,$b,${aOfOne.toString().take(5)})"
val rgb: String val rgb: String
@@ -30,21 +44,25 @@ value class HEXAColor (
get() = "#${r.shortPart()}${g.shortPart()}${b.shortPart()}" get() = "#${r.shortPart()}${g.shortPart()}${b.shortPart()}"
val shortHexa: String val shortHexa: String
get() = "$shortHex${a.shortPart()}" get() = "$shortHex${a.shortPart()}"
val rgbUInt: UInt
get() = (hexaUInt / 256u)
val rgbInt: Int val rgbInt: Int
get() = (uint shr 2).toInt() get() = rgbUInt.toInt()
val ahexUInt
get() = (a * 0x1000000).toUInt() + rgbUInt
val r: Int val r: Int
get() = ((uint and 0xff000000u) / 0x1000000u).toInt() get() = ((hexaUInt and 0xff000000u) / 0x1000000u).toInt()
val g: Int val g: Int
get() = ((uint and 0x00ff0000u) / 0x10000u).toInt() get() = ((hexaUInt and 0x00ff0000u) / 0x10000u).toInt()
val b: Int val b: Int
get() = ((uint and 0x0000ff00u) / 0x100u).toInt() get() = ((hexaUInt and 0x0000ff00u) / 0x100u).toInt()
val a: Int val a: Int
get() = ((uint and 0x000000ffu)).toInt() get() = ((hexaUInt and 0x000000ffu)).toInt()
val aOfOne: Float val aOfOne: Float
get() = a.toFloat() / (0xff) get() = a.toFloat() / (0xff)
init { init {
require(uint in 0u ..0xffffffffu) require(hexaUInt in 0u ..0xffffffffu)
} }
constructor(r: Int, g: Int, b: Int, a: Int) : this( constructor(r: Int, g: Int, b: Int, a: Int) : this(
@@ -64,7 +82,7 @@ value class HEXAColor (
return hexa return hexa
} }
override fun compareTo(other: HEXAColor): Int = (uint - other.uint).coerceIn(Int.MIN_VALUE.toUInt(), Int.MAX_VALUE.toLong().toUInt()).toInt() override fun compareTo(other: HEXAColor): Int = (hexaUInt - other.hexaUInt).coerceIn(Int.MIN_VALUE.toUInt(), Int.MAX_VALUE.toLong().toUInt()).toInt()
fun copy( fun copy(
r: Int = this.r, r: Int = this.r,
@@ -121,6 +139,21 @@ value class HEXAColor (
else -> color else -> color
}.lowercase().toUInt(16).let(::HEXAColor) }.lowercase().toUInt(16).let(::HEXAColor)
/**
* Creates [HEXAColor] from [uint] presume it is in format `0xFFEEBBAA` where FF - red, EE - green, BB - blue` and AA - alpha
*/
fun fromHexa(uint: UInt) = HEXAColor(uint)
/**
* Creates [HEXAColor] from [uint] presume it is in format `0xAAFFEEBB` where AA - alpha, FF - red, EE - green and BB - blue`
*/
fun fromAhex(uint: UInt) = HEXAColor(
a = ((uint and 0xff000000u) / 0x1000000u).toInt(),
r = ((uint and 0x00ff0000u) / 0x10000u).toInt(),
g = ((uint and 0x0000ff00u) / 0x100u).toInt(),
b = ((uint and 0x000000ffu)).toInt()
)
/** /**
* Parsing color from [color] * Parsing color from [color]
* *

View File

@@ -13,6 +13,9 @@ class HexColorTests {
val shortHexa: String, val shortHexa: String,
val hex: String, val hex: String,
val hexa: String, val hexa: String,
val ahex: String,
val ahexUInt: UInt,
val rgbUInt: UInt,
val rgb: String, val rgb: String,
val rgba: String, val rgba: String,
val r: Int, val r: Int,
@@ -24,11 +27,14 @@ class HexColorTests {
val testColors: List<TestColor> val testColors: List<TestColor>
get() = listOf( get() = listOf(
TestColor( TestColor(
color = HEXAColor(uint = 0xff0000ffu), color = HEXAColor(hexaUInt = 0xff0000ffu),
shortHex = "#f00", shortHex = "#f00",
shortHexa = "#f00f", shortHexa = "#f00f",
hex = "#ff0000", hex = "#ff0000",
hexa = "#ff0000ff", hexa = "#ff0000ff",
ahex = "#ffff0000",
ahexUInt = 0xffff0000u,
rgbUInt = 0xff0000u,
rgb = "rgb(255,0,0)", rgb = "rgb(255,0,0)",
rgba = "rgba(255,0,0,1.0)", rgba = "rgba(255,0,0,1.0)",
r = 0xff, r = 0xff,
@@ -38,11 +44,14 @@ class HexColorTests {
"rgba(255,0,0,1)", "rgba(255,0,0,1)",
), ),
TestColor( TestColor(
color = HEXAColor(uint = 0x00ff00ffu), color = HEXAColor(hexaUInt = 0x00ff00ffu),
shortHex = "#0f0", shortHex = "#0f0",
shortHexa = "#0f0f", shortHexa = "#0f0f",
hex = "#00ff00", hex = "#00ff00",
hexa = "#00ff00ff", hexa = "#00ff00ff",
ahex = "#ff00ff00",
ahexUInt = 0xff00ff00u,
rgbUInt = 0x00ff00u,
rgb = "rgb(0,255,0)", rgb = "rgb(0,255,0)",
rgba = "rgba(0,255,0,1.0)", rgba = "rgba(0,255,0,1.0)",
r = 0x00, r = 0x00,
@@ -57,6 +66,9 @@ class HexColorTests {
shortHexa = "#00ff", shortHexa = "#00ff",
hex = "#0000ff", hex = "#0000ff",
hexa = "#0000ffff", hexa = "#0000ffff",
ahex = "#ff0000ff",
ahexUInt = 0xff0000ffu,
rgbUInt = 0x0000ffu,
rgb = "rgb(0,0,255)", rgb = "rgb(0,0,255)",
rgba = "rgba(0,0,255,1.0)", rgba = "rgba(0,0,255,1.0)",
r = 0x00, r = 0x00,
@@ -71,6 +83,9 @@ class HexColorTests {
shortHexa = "#f008", shortHexa = "#f008",
hex = "#ff0000", hex = "#ff0000",
hexa = "#ff000088", hexa = "#ff000088",
ahex = "#88ff0000",
ahexUInt = 0x88ff0000u,
rgbUInt = 0xff0000u,
rgb = "rgb(255,0,0)", rgb = "rgb(255,0,0)",
rgba = "rgba(255,0,0,0.533)", rgba = "rgba(255,0,0,0.533)",
r = 0xff, r = 0xff,
@@ -84,6 +99,9 @@ class HexColorTests {
shortHexa = "#0f08", shortHexa = "#0f08",
hex = "#00ff00", hex = "#00ff00",
hexa = "#00ff0088", hexa = "#00ff0088",
ahex = "#8800ff00",
ahexUInt = 0x8800ff00u,
rgbUInt = 0x00ff00u,
rgb = "rgb(0,255,0)", rgb = "rgb(0,255,0)",
rgba = "rgba(0,255,0,0.533)", rgba = "rgba(0,255,0,0.533)",
r = 0x00, r = 0x00,
@@ -97,6 +115,9 @@ class HexColorTests {
shortHexa = "#00f8", shortHexa = "#00f8",
hex = "#0000ff", hex = "#0000ff",
hexa = "#0000ff88", hexa = "#0000ff88",
ahex = "#880000ff",
ahexUInt = 0x880000ffu,
rgbUInt = 0x0000ffu,
rgb = "rgb(0,0,255)", rgb = "rgb(0,0,255)",
rgba = "rgba(0,0,255,0.533)", rgba = "rgba(0,0,255,0.533)",
r = 0x00, r = 0x00,
@@ -110,6 +131,9 @@ class HexColorTests {
shortHexa = "#f002", shortHexa = "#f002",
hex = "#ff0000", hex = "#ff0000",
hexa = "#ff000022", hexa = "#ff000022",
ahex = "#22ff0000",
ahexUInt = 0x22ff0000u,
rgbUInt = 0xff0000u,
rgb = "rgb(255,0,0)", rgb = "rgb(255,0,0)",
rgba = "rgba(255,0,0,0.133)", rgba = "rgba(255,0,0,0.133)",
r = 0xff, r = 0xff,
@@ -123,6 +147,9 @@ class HexColorTests {
shortHexa = "#0f02", shortHexa = "#0f02",
hex = "#00ff00", hex = "#00ff00",
hexa = "#00ff0022", hexa = "#00ff0022",
ahex = "#2200ff00",
ahexUInt = 0x2200ff00u,
rgbUInt = 0x00ff00u,
rgb = "rgb(0,255,0)", rgb = "rgb(0,255,0)",
rgba = "rgba(0,255,0,0.133)", rgba = "rgba(0,255,0,0.133)",
r = 0x00, r = 0x00,
@@ -136,6 +163,9 @@ class HexColorTests {
shortHexa = "#00f2", shortHexa = "#00f2",
hex = "#0000ff", hex = "#0000ff",
hexa = "#0000ff22", hexa = "#0000ff22",
ahex = "#220000ff",
ahexUInt = 0x220000ffu,
rgbUInt = 0x0000ffu,
rgb = "rgb(0,0,255)", rgb = "rgb(0,0,255)",
rgba = "rgba(0,0,255,0.133)", rgba = "rgba(0,0,255,0.133)",
r = 0x00, r = 0x00,
@@ -150,6 +180,9 @@ class HexColorTests {
testColors.forEach { testColors.forEach {
assertEquals(it.hex, it.color.hex) assertEquals(it.hex, it.color.hex)
assertEquals(it.hexa, it.color.hexa) assertEquals(it.hexa, it.color.hexa)
assertEquals(it.ahex, it.color.ahex)
assertEquals(it.rgbUInt, it.color.rgbUInt)
assertEquals(it.ahexUInt, it.color.ahexUInt)
assertEquals(it.shortHex, it.color.shortHex) assertEquals(it.shortHex, it.color.shortHex)
assertEquals(it.shortHexa, it.color.shortHexa) assertEquals(it.shortHexa, it.color.shortHexa)
assertEquals(it.rgb, it.color.rgb) assertEquals(it.rgb, it.color.rgb)
@@ -158,6 +191,7 @@ class HexColorTests {
assertEquals(it.g, it.color.g) assertEquals(it.g, it.color.g)
assertEquals(it.b, it.color.b) assertEquals(it.b, it.color.b)
assertEquals(it.a, it.color.a) assertEquals(it.a, it.color.a)
assertEquals(it.color, HEXAColor.fromAhex(it.ahexUInt))
} }
} }
@@ -167,7 +201,7 @@ class HexColorTests {
assertEquals(it.color.copy(aOfOne = 1f), HEXAColor.parseStringColor(it.hex)) assertEquals(it.color.copy(aOfOne = 1f), HEXAColor.parseStringColor(it.hex))
assertEquals(it.color, HEXAColor.parseStringColor(it.hexa)) assertEquals(it.color, HEXAColor.parseStringColor(it.hexa))
assertEquals(it.color.copy(aOfOne = 1f), HEXAColor.parseStringColor(it.rgb)) assertEquals(it.color.copy(aOfOne = 1f), HEXAColor.parseStringColor(it.rgb))
assertTrue(it.color.uint.toInt() - HEXAColor.parseStringColor(it.rgba).uint.toInt() in -0x1 .. 0x1, ) assertTrue(it.color.hexaUInt.toInt() - HEXAColor.parseStringColor(it.rgba).hexaUInt.toInt() in -0x1 .. 0x1, )
assertEquals(it.color.copy(aOfOne = 1f), HEXAColor.parseStringColor(it.shortHex)) assertEquals(it.color.copy(aOfOne = 1f), HEXAColor.parseStringColor(it.shortHex))
assertEquals(it.color.copy(a = floor(it.color.a.toFloat() / 16).toInt() * 0x10), HEXAColor.parseStringColor(it.shortHexa)) assertEquals(it.color.copy(a = floor(it.color.a.toFloat() / 16).toInt() * 0x10), HEXAColor.parseStringColor(it.shortHexa))
} }

View File

@@ -15,5 +15,5 @@ crypto_js_version=4.1.1
# Project data # Project data
group=dev.inmo group=dev.inmo
version=0.20.31 version=0.20.33
android_code_version=237 android_code_version=239

View File

@@ -19,7 +19,7 @@ gh-release = "2.5.2"
koin = "3.5.3" koin = "3.5.3"
okio = "3.7.0" okio = "3.8.0"
ksp = "1.9.22-1.0.17" ksp = "1.9.22-1.0.17"
kotlin-poet = "1.16.0" kotlin-poet = "1.16.0"

View File

@@ -1,5 +1,5 @@
distributionBase=GRADLE_USER_HOME distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists distributionPath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-8.5-bin.zip distributionUrl=https\://services.gradle.org/distributions/gradle-8.6-bin.zip
zipStoreBase=GRADLE_USER_HOME zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists zipStorePath=wrapper/dists

View File

@@ -1,6 +1,11 @@
package dev.inmo.micro_utils.strings package dev.inmo.micro_utils.strings
import dev.inmo.micro_utils.language_codes.IetfLang import dev.inmo.micro_utils.language_codes.IetfLang
import kotlinx.serialization.KSerializer
import kotlinx.serialization.Serializable
import kotlinx.serialization.descriptors.SerialDescriptor
import kotlinx.serialization.encoding.Decoder
import kotlinx.serialization.encoding.Encoder
/** /**
* Use this class as a type of your strings object fields. For example: * Use this class as a type of your strings object fields. For example:
@@ -15,9 +20,10 @@ import dev.inmo.micro_utils.language_codes.IetfLang
* *
* @see buildStringResource * @see buildStringResource
*/ */
class StringResource( @Serializable(StringResource.Companion::class)
data class StringResource(
val default: String, val default: String,
val map: Map<IetfLang, Lazy<String>> val translations: Map<IetfLang, Lazy<String>>
) { ) {
class Builder( class Builder(
var default: String var default: String
@@ -31,11 +37,18 @@ class StringResource(
infix fun IetfLang.variant(value: () -> String) = this variant lazy(value) infix fun IetfLang.variant(value: () -> String) = this variant lazy(value)
infix fun IetfLang.variant(value: String) = this variant lazyOf(value) infix fun IetfLang.variant(value: String) = this variant lazyOf(value)
operator fun IetfLang.invoke(value: () -> String) = this variant value
operator fun IetfLang.invoke(value: String) = this variant value
infix fun String.variant(value: Lazy<String>) = IetfLang(this) variant value infix fun String.variant(value: Lazy<String>) = IetfLang(this) variant value
infix fun String.variant(value: () -> String) = IetfLang(this) variant lazy(value) infix fun String.variant(value: () -> String) = IetfLang(this) variant lazy(value)
infix fun String.variant(value: String) = this variant lazyOf(value) infix fun String.variant(value: String) = this variant lazyOf(value)
operator fun String.invoke(value: () -> String) = this variant value
operator fun String.invoke(value: String) = this variant value
fun build() = StringResource(default, map.toMap()) fun build() = StringResource(default, map.toMap())
} }
@@ -43,17 +56,48 @@ class StringResource(
if (languageCode == null) { if (languageCode == null) {
return default return default
} }
map[languageCode] ?.let { return it.value } translations[languageCode] ?.let { return it.value }
return languageCode.parentLang ?.let { return languageCode.parentLang ?.let {
map[it] ?.value translations[it] ?.value
} ?: default } ?: default
} }
companion object : KSerializer<StringResource> {
@Serializable
private class Surrogate(
val default: String,
val translations: Map<String, String>
)
override val descriptor: SerialDescriptor
get() = Surrogate.serializer().descriptor
override fun deserialize(decoder: Decoder): StringResource {
val surrogate = Surrogate.serializer().deserialize(decoder)
return StringResource(
surrogate.default,
surrogate.translations.map { IetfLang(it.key) to lazyOf(it.value) }.toMap()
)
}
override fun serialize(encoder: Encoder, value: StringResource) {
Surrogate.serializer().serialize(
encoder,
Surrogate(
value.default,
value.translations.map {
it.key.code to it.value.value
}.toMap()
)
)
}
}
} }
inline fun buildStringResource( inline fun buildStringResource(
default: String, default: String,
builder: StringResource.Builder.() -> Unit builder: StringResource.Builder.() -> Unit = {}
): StringResource { ): StringResource {
return StringResource.Builder(default).apply(builder).build() return StringResource.Builder(default).apply(builder).build()
} }