From 791e57cb0f105dbb586fd8f2ec1fac63f2cd6899 Mon Sep 17 00:00:00 2001 From: DBotThePony Date: Mon, 13 Feb 2023 21:50:19 +0700 Subject: [PATCH] =?UTF-8?q?Image=20Reference=20=D0=BA=D0=B0=D0=BA=20=D0=B8?= =?UTF-8?q?=20=D0=B4=D0=BB=D1=8F=20=D0=B8=D0=B7=D0=BE=D0=B1=D1=80=D0=B0?= =?UTF-8?q?=D0=B6=D0=B5=D0=BD=D0=B8=D0=B9,=20=D1=82=D0=B0=D0=BA=20=D0=B8?= =?UTF-8?q?=20=D0=B4=D0=BB=D1=8F=20=D1=81=D0=BF=D1=80=D0=B0=D0=B9=D1=82?= =?UTF-8?q?=D0=BE=D0=B2?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../ru/dbotthepony/kstarbound/Starbound.kt | 12 +-- .../kstarbound/client/render/BoundSprite.kt | 10 +- .../kstarbound/client/render/TileRenderer.kt | 2 +- .../ru/dbotthepony/kstarbound/defs/Species.kt | 6 +- .../kstarbound/defs/StatusEffectDefinition.kt | 4 +- .../defs/animation/AnimationDefinition.kt | 4 +- .../defs/image/AtlasConfiguration.kt | 4 +- .../kstarbound/defs/image/ImageReference.kt | 94 ++++++++++++++++--- .../kstarbound/defs/image/SpriteReference.kt | 53 ----------- .../kstarbound/defs/item/IInventoryIcon.kt | 4 +- .../kstarbound/defs/item/InventoryIcon.kt | 11 ++- .../defs/particle/ParticleConfig.kt | 4 +- .../kstarbound/defs/tile/RenderParameters.kt | 6 +- .../dbotthepony/kstarbound/util/PathStack.kt | 4 +- .../dbotthepony/kstarbound/util/SBPattern.kt | 38 ++++++-- 15 files changed, 149 insertions(+), 107 deletions(-) delete mode 100644 src/main/kotlin/ru/dbotthepony/kstarbound/defs/image/SpriteReference.kt diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/Starbound.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/Starbound.kt index 9827602e..a547e2ea 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/Starbound.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/Starbound.kt @@ -4,9 +4,6 @@ import com.google.common.collect.Interner import com.google.common.collect.Interners import com.google.gson.* import com.google.gson.internal.bind.JsonTreeReader -import com.google.gson.internal.bind.TypeAdapters -import com.google.gson.stream.JsonReader -import com.google.gson.stream.JsonWriter import it.unimi.dsi.fastutil.objects.Object2ObjectFunction import it.unimi.dsi.fastutil.objects.Object2ObjectOpenHashMap import org.apache.logging.log4j.LogManager @@ -18,7 +15,6 @@ import ru.dbotthepony.kstarbound.api.explore import ru.dbotthepony.kstarbound.defs.* import ru.dbotthepony.kstarbound.defs.image.AtlasConfiguration import ru.dbotthepony.kstarbound.defs.image.ImageReference -import ru.dbotthepony.kstarbound.defs.image.SpriteReference import ru.dbotthepony.kstarbound.defs.item.BackArmorItemPrototype import ru.dbotthepony.kstarbound.defs.item.ChestArmorItemPrototype import ru.dbotthepony.kstarbound.defs.item.CurrencyItemPrototype @@ -100,8 +96,6 @@ class Starbound : ISBFileLocator { private val _items = ObjectRegistry("items", IItemDefinition::itemName) val items = _items.view - val spriteRegistry: SpriteReference.Adapter - val gson: Gson = with(GsonBuilder()) { serializeNulls() setDateFormat(DateFormat.LONG) @@ -160,13 +154,11 @@ class Starbound : ISBFileLocator { registerTypeAdapter(EnumAdapter(DamageType::class, default = DamageType.NORMAL)) - spriteRegistry = SpriteReference.Adapter(pathStack, this@Starbound::atlasRegistry) - registerTypeAdapter(spriteRegistry) - registerTypeAdapterFactory(InventoryIcon.Factory(pathStack, spriteRegistry)) + registerTypeAdapterFactory(InventoryIcon.Factory(pathStack)) registerTypeAdapterFactory(IArmorItemDefinition.Frames.Factory) registerTypeAdapterFactory(DirectAssetReferenceFactory(pathStack)) - registerTypeAdapterFactory(ImageReference.Adapter(this@Starbound::atlasRegistry)) + registerTypeAdapterFactory(ImageReference.Factory({ atlasRegistry.get(it) }, pathStack)) registerTypeAdapterFactory(AssetReferenceFactory(pathStack, this@Starbound)) diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/client/render/BoundSprite.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/client/render/BoundSprite.kt index 5132bef9..37eb0a37 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/client/render/BoundSprite.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/client/render/BoundSprite.kt @@ -5,7 +5,7 @@ import ru.dbotthepony.kstarbound.client.gl.GLTexture2D import ru.dbotthepony.kstarbound.client.gl.vertex.QuadTransformers import ru.dbotthepony.kstarbound.defs.image.AtlasConfiguration import ru.dbotthepony.kstarbound.defs.image.IUVCoordinates -import ru.dbotthepony.kstarbound.defs.image.SpriteReference +import ru.dbotthepony.kstarbound.defs.image.ImageReference /** * Связка текстуры-атласа + координат спрайта на ней @@ -45,15 +45,15 @@ class BoundSprite( /** * Создаёт связку текстуры-атласа + координгат спрайта на ней */ -fun SpriteReference.bind(texture: GLTexture2D): BoundSprite { - return BoundSprite(stateless, texture) +fun ImageReference.bind(texture: GLTexture2D): BoundSprite { + return BoundSprite(sprite!!, texture) } /** * Создаёт связку текстуры-атласа, которая загружается через [GLStateTracker.loadNamedTextureSafe] + координгат спрайта на ней */ -fun SpriteReference.bind(state: GLStateTracker): BoundSprite { - return BoundSprite(stateless, state.loadNamedTextureSafe(image)) +fun ImageReference.bind(state: GLStateTracker): BoundSprite { + return BoundSprite(sprite!!, state.loadNamedTextureSafe(imagePath.value!!)) } /** diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/client/render/TileRenderer.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/client/render/TileRenderer.kt index 13b5adf0..f55939da 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/client/render/TileRenderer.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/client/render/TileRenderer.kt @@ -185,7 +185,7 @@ private class ModifierEqualityTester(val definition: MaterialModifier) : Equalit class TileRenderer(val renderers: TileRenderers, val def: IRenderableTile) { val state get() = renderers.state - val texture = state.loadNamedTexture(def.renderParameters.texture.image.fullPath).also { + val texture = state.loadNamedTexture(def.renderParameters.texture.imagePath.value!!).also { it.textureMagFilter = GL_NEAREST } diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/Species.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/Species.kt index ab2a7242..0b95113a 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/Species.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/Species.kt @@ -2,7 +2,7 @@ package ru.dbotthepony.kstarbound.defs import com.google.common.collect.ImmutableList import com.google.common.collect.ImmutableSet -import ru.dbotthepony.kstarbound.defs.image.SpriteReference +import ru.dbotthepony.kstarbound.defs.image.ImageReference import ru.dbotthepony.kstarbound.defs.item.IItemDefinition import ru.dbotthepony.kstarbound.defs.player.BlueprintLearnList import ru.dbotthepony.kstarbound.io.json.builder.JsonFactory @@ -33,8 +33,8 @@ data class Species( @JsonFactory data class Gender( val name: String, - val image: SpriteReference, - val characterImage: SpriteReference, + val image: ImageReference, + val characterImage: ImageReference, val hairGroup: String? = null, val hair: ImmutableSet, val shirt: ImmutableSet>, diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/StatusEffectDefinition.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/StatusEffectDefinition.kt index de43f6d1..6d35cead 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/StatusEffectDefinition.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/StatusEffectDefinition.kt @@ -2,7 +2,7 @@ package ru.dbotthepony.kstarbound.defs import com.google.common.collect.ImmutableList import ru.dbotthepony.kstarbound.defs.animation.AnimationDefinition -import ru.dbotthepony.kstarbound.defs.image.SpriteReference +import ru.dbotthepony.kstarbound.defs.image.ImageReference import ru.dbotthepony.kstarbound.io.json.builder.JsonFactory @JsonFactory @@ -11,7 +11,7 @@ data class StatusEffectDefinition( val defaultDuration: Double, val blockingStat: String? = null, val label: String? = null, - val icon: SpriteReference? = null, + val icon: ImageReference? = null, override val scripts: ImmutableList = ImmutableList.of(), override val scriptDelta: Int = 1, val animationConfig: AssetReference? = null, diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/animation/AnimationDefinition.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/animation/AnimationDefinition.kt index 6dcf0b24..30dc76ec 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/animation/AnimationDefinition.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/animation/AnimationDefinition.kt @@ -3,10 +3,8 @@ package ru.dbotthepony.kstarbound.defs.animation import com.google.common.collect.ImmutableList import com.google.common.collect.ImmutableMap import ru.dbotthepony.kstarbound.defs.image.ImageReference -import ru.dbotthepony.kstarbound.defs.image.SpriteReference import ru.dbotthepony.kstarbound.defs.particle.ParticleEmitter import ru.dbotthepony.kstarbound.io.json.builder.JsonFactory -import ru.dbotthepony.kstarbound.util.Either import ru.dbotthepony.kvector.vector.ndouble.Vector2d @JsonFactory @@ -74,7 +72,7 @@ data class AnimationDefinition( val centered: Boolean? = null, val transformationGroups: ImmutableList? = null, val offset: Vector2d? = null, - val image: SpriteReference? = null + val image: ImageReference? = null ) { companion object { val EMPTY = Properties() diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/image/AtlasConfiguration.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/image/AtlasConfiguration.kt index 0c6a5df8..2bc161ab 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/image/AtlasConfiguration.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/image/AtlasConfiguration.kt @@ -70,8 +70,10 @@ class AtlasConfiguration private constructor( return get(name) ?: first } + private val any by lazy { get("root") ?: get("0") ?: get("default") ?: first } + fun any(): Sprite { - return get("root") ?: get("0") ?: get("default") ?: first + return any } class Sprite( diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/image/ImageReference.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/image/ImageReference.kt index 7ea6b608..193e1851 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/image/ImageReference.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/image/ImageReference.kt @@ -10,36 +10,102 @@ import com.google.gson.stream.JsonToken import com.google.gson.stream.JsonWriter import ru.dbotthepony.kstarbound.defs.DirectAssetReference import ru.dbotthepony.kstarbound.util.PathStack +import ru.dbotthepony.kstarbound.util.SBPattern /** - * Хранит данные (пару) вида "/example/animated.png" у которого, вероятнее всего, есть "/example/animated.frames" - * * @see [AtlasConfiguration.Companion.get] */ -data class ImageReference( - val image: DirectAssetReference, - val config: AtlasConfiguration, +class ImageReference private constructor( + val raw: DirectAssetReference, + val imagePath: SBPattern, + val spritePath: SBPattern?, + val atlas: AtlasConfiguration?, + private val atlasLocator: (String) -> AtlasConfiguration? ) { - class Adapter(private val atlasRegistry: () -> AtlasConfiguration.Registry) : TypeAdapterFactory { + /** + * Спрайт, на которое ссылается данный референс, или `null` если: + * * [atlas] равен `null` + * * [spritePath] является шаблоном и определены не все значения + */ + val sprite by lazy { + if (atlas == null) + null + else if (spritePath == null) + atlas.any() + else + atlas.get(spritePath.value ?: return@lazy null) + } + + fun with(values: (String) -> String?): ImageReference { + val imagePath = this.imagePath.with(values) + val spritePath = this.spritePath?.with(values) + + if (imagePath != this.imagePath || spritePath != this.spritePath) { + if (imagePath != this.imagePath) { + val resolved = imagePath.value + + if (resolved == null) + return ImageReference(raw, imagePath, spritePath, null, atlasLocator) + else + return ImageReference(raw, imagePath, spritePath, atlasLocator.invoke(resolved), atlasLocator) + } else { + return ImageReference(raw, imagePath, spritePath, atlas, atlasLocator) + } + } + + return this + } + + fun with(values: Map): ImageReference { + return with(values::get) + } + + override fun equals(other: Any?): Boolean { + return other is ImageReference && other.imagePath == imagePath && other.spritePath == spritePath && other.atlas == atlas + } + + override fun hashCode(): Int { + return imagePath.hashCode() * 31 + spritePath.hashCode() + } + + override fun toString(): String { + return "ImageReference[$imagePath:$spritePath]" + } + + class Factory(private val atlasLocator: (String) -> AtlasConfiguration, private val remapper: PathStack) : TypeAdapterFactory { override fun create(gson: Gson, type: TypeToken): TypeAdapter? { if (type.rawType == ImageReference::class.java) { return object : TypeAdapter() { - private val assets = gson.getAdapter(DirectAssetReference::class.java) + private val strings = gson.getAdapter(String::class.java) override fun write(out: JsonWriter, value: ImageReference?) { - out.value(value?.image?.fullPath) + out.value(value?.raw?.fullPath) } override fun read(`in`: JsonReader): ImageReference? { if (`in`.peek() == JsonToken.NULL) return null - val image = assets.read(`in`) + val path = strings.read(`in`) - if (image.path.contains(':')) - throw JsonSyntaxException("Expected atlas/image reference, but got sprite reference: $image") + if (path == "") + return NEVER - return ImageReference(image, atlasRegistry.invoke().get(image.fullPath)) + val split = path.split(':') + + if (split.size > 2) { + throw JsonSyntaxException("Ambiguous image reference: $path") + } + + val imagePath = if (split.size == 2) SBPattern.of(split[0]) else SBPattern.of(path) + val spritePath = if (split.size == 2) SBPattern.of(split[1]) else null + + if (imagePath.isPlainString) { + val remapped = remapper.remap(split[0]) + return ImageReference(DirectAssetReference(path, remapper.remap(path)), SBPattern.raw(remapped), spritePath, atlasLocator.invoke(remapped), atlasLocator) + } else { + return ImageReference(DirectAssetReference(path, path), imagePath, spritePath, null, atlasLocator) + } } } as TypeAdapter } @@ -47,4 +113,8 @@ data class ImageReference( return null } } + + companion object { + val NEVER = ImageReference(DirectAssetReference("", ""), SBPattern.EMPTY, null, null) { null } + } } diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/image/SpriteReference.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/image/SpriteReference.kt deleted file mode 100644 index 3590847e..00000000 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/image/SpriteReference.kt +++ /dev/null @@ -1,53 +0,0 @@ -package ru.dbotthepony.kstarbound.defs.image - -import com.google.gson.TypeAdapter -import com.google.gson.stream.JsonReader -import com.google.gson.stream.JsonWriter -import ru.dbotthepony.kstarbound.util.PathStack -import ru.dbotthepony.kstarbound.util.SBPattern - -/** - * Хранит данные (пару) вида "/example/image.png:sprite.name" - */ -data class SpriteReference( - val image: String, - val atlas: AtlasConfiguration, - val sprite: SBPattern -) { - val stateless by lazy { resolve { null } } - - fun resolve(with: (String) -> String?): AtlasConfiguration.Sprite { - val resolved = sprite.resolve(with) ?: return atlas.any() - return atlas[resolved] ?: atlas.any() - } - - class Adapter(private val remapper: PathStack, private val atlasRegistry: () -> AtlasConfiguration.Registry) : TypeAdapter() { - override fun write(out: JsonWriter, value: SpriteReference?) { - if (value == null) - out.nullValue() - else - out.value(value.image + ":" + value.sprite.raw) - } - - override fun read(`in`: JsonReader): SpriteReference? { - val value = `in`.nextString() ?: return null - - // TODO: если нам надо принудительно удалить спрайт с вышестоящей ступени, как это сделать? - // а ещё, зачем вы это сделали. - if (value == "") - return null - - return parse(remapper.remap(value)) - } - - fun parse(input: String): SpriteReference { - val grid = atlasRegistry.invoke().get(input.substringBefore(':')) - - return when (input.count { it == ':' }) { - 0 -> SpriteReference(input, grid, SBPattern.raw(grid.any().name)) - 1 -> SpriteReference(input.substringBefore(':'), grid, SBPattern.of(input.substringAfter(':'))) - else -> throw IllegalArgumentException("Invalid sprite reference: $input") - } - } - } -} diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/item/IInventoryIcon.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/item/IInventoryIcon.kt index 678a88dd..8e8d716b 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/item/IInventoryIcon.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/item/IInventoryIcon.kt @@ -1,9 +1,9 @@ package ru.dbotthepony.kstarbound.defs.item -import ru.dbotthepony.kstarbound.defs.image.SpriteReference +import ru.dbotthepony.kstarbound.defs.image.ImageReference import ru.dbotthepony.kstarbound.io.json.builder.JsonImplementation @JsonImplementation(InventoryIcon::class) interface IInventoryIcon { - val image: SpriteReference + val image: ImageReference } diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/item/InventoryIcon.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/item/InventoryIcon.kt index 558809c7..f9f2cd1a 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/item/InventoryIcon.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/item/InventoryIcon.kt @@ -1,24 +1,27 @@ package ru.dbotthepony.kstarbound.defs.item import com.google.gson.Gson +import com.google.gson.JsonPrimitive import com.google.gson.TypeAdapter import com.google.gson.TypeAdapterFactory +import com.google.gson.internal.bind.JsonTreeReader import com.google.gson.reflect.TypeToken import com.google.gson.stream.JsonReader import com.google.gson.stream.JsonToken import com.google.gson.stream.JsonWriter -import ru.dbotthepony.kstarbound.defs.image.SpriteReference +import ru.dbotthepony.kstarbound.defs.image.ImageReference import ru.dbotthepony.kstarbound.io.json.builder.FactoryAdapter import ru.dbotthepony.kstarbound.util.PathStack data class InventoryIcon( - override val image: SpriteReference + override val image: ImageReference ) : IInventoryIcon { - class Factory(val remapper: PathStack, val spriteRegistry: SpriteReference.Adapter) : TypeAdapterFactory { + class Factory(val remapper: PathStack) : TypeAdapterFactory { override fun create(gson: Gson, type: TypeToken): TypeAdapter? { if (type.rawType == InventoryIcon::class.java) { return object : TypeAdapter() { private val adapter = FactoryAdapter.Builder(InventoryIcon::class, InventoryIcon::image).build(gson) + private val images = gson.getAdapter(ImageReference::class.java) override fun write(out: JsonWriter, value: InventoryIcon?) { if (value == null) @@ -32,7 +35,7 @@ data class InventoryIcon( return null if (`in`.peek() == JsonToken.STRING) { - return InventoryIcon(spriteRegistry.parse(remapper.remap(`in`.nextString()))) + return InventoryIcon(images.read(JsonTreeReader(JsonPrimitive(remapper.remap(`in`.nextString()))))) } return adapter.read(`in`) diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/particle/ParticleConfig.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/particle/ParticleConfig.kt index 25167621..2d83dd46 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/particle/ParticleConfig.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/particle/ParticleConfig.kt @@ -3,7 +3,7 @@ package ru.dbotthepony.kstarbound.defs.particle import ru.dbotthepony.kstarbound.defs.AssetReference import ru.dbotthepony.kstarbound.defs.animation.AnimationDefinition import ru.dbotthepony.kstarbound.defs.animation.DestructionAction -import ru.dbotthepony.kstarbound.defs.image.SpriteReference +import ru.dbotthepony.kstarbound.defs.image.ImageReference import ru.dbotthepony.kstarbound.io.json.builder.JsonFactory import ru.dbotthepony.kstarbound.util.SBPattern import ru.dbotthepony.kvector.vector.Color @@ -14,7 +14,7 @@ import ru.dbotthepony.kvector.vector.ndouble.Vector4d data class ParticleConfig( val type: ParticleType, val animation: AssetReference? = null, - val image: SpriteReference? = null, + val image: ImageReference? = null, override val offset: Vector2d? = null, override val position: Vector2d? = null, override val offsetRegion: Vector4d? = null, diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/tile/RenderParameters.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/tile/RenderParameters.kt index 454f7f7f..c3b4d5dc 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/tile/RenderParameters.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/tile/RenderParameters.kt @@ -16,4 +16,8 @@ data class RenderParameters( val occludesBelow: Boolean = false, val lightTransparent: Boolean = false, val zLevel: Int, -) +) { + init { + checkNotNull(texture.imagePath.value) { "Tile render parameters are stateless, but provided image is a pattern: ${texture.raw}" } + } +} diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/util/PathStack.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/util/PathStack.kt index 7ab13fc2..f8bfd429 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/util/PathStack.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/util/PathStack.kt @@ -3,7 +3,7 @@ package ru.dbotthepony.kstarbound.util import com.google.common.collect.Interner import kotlin.concurrent.getOrSet -class PathStack(private val interner: Interner? = null) { +class PathStack(private val interner: Interner = Interner { it }) { private val _stack = ThreadLocal>() private val stack: ArrayDeque get() = _stack.getOrSet { ArrayDeque() } @@ -35,7 +35,7 @@ class PathStack(private val interner: Interner? = null) { if (b[0] == '/') return b - return interner?.intern("$a/$b") ?: "$a/$b" + return interner.intern("$a/$b") } fun remap(path: String): String { diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/util/SBPattern.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/util/SBPattern.kt index 013598a2..1bee7005 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/util/SBPattern.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/util/SBPattern.kt @@ -26,6 +26,7 @@ class SBPattern private constructor( val pieces: ImmutableList, val names: ImmutableSet ) { + val isPlainString get() = names.isEmpty() val value by lazy { resolve { null } } override fun toString(): String { @@ -72,15 +73,28 @@ class SBPattern private constructor( } fun with(params: Map): SBPattern { + return with(params::get) + } + + fun with(params: (String) -> String?): SBPattern { if (names.isEmpty()) return this val map = Object2ObjectArrayMap() map.putAll(this.params) + var any = false - for ((key, value) in params.entries) - if (names.contains(key)) - map[key] = value + for (name in names) { + val get = params.invoke(name) + + if (get != null && get != map[name]) { + map[name] = get + any = true + } + } + + if (!any) + return this return SBPattern(raw, ImmutableMap.copyOf(map), pieces, names) } @@ -97,10 +111,14 @@ class SBPattern private constructor( } companion object : TypeAdapterFactory { + private val interner = Interners.newWeakInterner() + + @JvmField + val EMPTY = raw("") + override fun create(gson: Gson, type: TypeToken): TypeAdapter? { if (type.rawType == SBPattern::class.java) { return object : TypeAdapter() { - private val interner = Interners.newWeakInterner() private val strings = gson.getAdapter(String::class.java) override fun write(out: JsonWriter, value: SBPattern?) { @@ -118,6 +136,9 @@ class SBPattern private constructor( @JvmStatic fun of(raw: String): SBPattern { + if (raw == "") + return EMPTY + val pieces = ImmutableList.Builder() var i = 0 @@ -144,10 +165,15 @@ class SBPattern private constructor( } val built = pieces.build() - return SBPattern(raw, pieces = built, params = ImmutableMap.of(), names = built.stream().map { it.name }.filter { it != null }.collect(ImmutableSet.toImmutableSet())) + return interner.intern(SBPattern(raw, pieces = built, params = ImmutableMap.of(), names = built.stream().map { it.name }.filter { it != null }.collect(ImmutableSet.toImmutableSet()))) } @JvmStatic - fun raw(raw: String): SBPattern = SBPattern(raw, ImmutableMap.of(), ImmutableList.of(), ImmutableSet.of()) + fun raw(raw: String): SBPattern { + if (raw == "") + return EMPTY + + return SBPattern(raw, ImmutableMap.of(), ImmutableList.of(), ImmutableSet.of()) + } } }