Image Reference как и для изображений, так и для спрайтов

This commit is contained in:
DBotThePony 2023-02-13 21:50:19 +07:00
parent 3ef02ad450
commit 791e57cb0f
Signed by: DBot
GPG Key ID: DCC23B5715498507
15 changed files with 149 additions and 107 deletions

View File

@ -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))

View File

@ -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!!))
}
/**

View File

@ -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
}

View File

@ -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<String>,
val shirt: ImmutableSet<RegistryReference<IItemDefinition>>,

View File

@ -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<DirectAssetReference> = ImmutableList.of(),
override val scriptDelta: Int = 1,
val animationConfig: AssetReference<AnimationDefinition>? = null,

View File

@ -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<String>? = null,
val offset: Vector2d? = null,
val image: SpriteReference? = null
val image: ImageReference? = null
) {
companion object {
val EMPTY = Properties()

View File

@ -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(

View File

@ -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<String, String>): 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 <T : Any?> create(gson: Gson, type: TypeToken<T>): TypeAdapter<T>? {
if (type.rawType == ImageReference::class.java) {
return object : TypeAdapter<ImageReference>() {
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<T>
}
@ -47,4 +113,8 @@ data class ImageReference(
return null
}
}
companion object {
val NEVER = ImageReference(DirectAssetReference("", ""), SBPattern.EMPTY, null, null) { null }
}
}

View File

@ -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<SpriteReference>() {
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")
}
}
}
}

View File

@ -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
}

View File

@ -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 <T : Any?> create(gson: Gson, type: TypeToken<T>): TypeAdapter<T>? {
if (type.rawType == InventoryIcon::class.java) {
return object : TypeAdapter<InventoryIcon>() {
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`)

View File

@ -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<AnimationDefinition>? = null,
val image: SpriteReference? = null,
val image: ImageReference? = null,
override val offset: Vector2d? = null,
override val position: Vector2d? = null,
override val offsetRegion: Vector4d? = null,

View File

@ -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}" }
}
}

View File

@ -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<String>? = null) {
class PathStack(private val interner: Interner<String> = Interner { it }) {
private val _stack = ThreadLocal<ArrayDeque<String>>()
private val stack: ArrayDeque<String> get() = _stack.getOrSet { ArrayDeque() }
@ -35,7 +35,7 @@ class PathStack(private val interner: Interner<String>? = null) {
if (b[0] == '/')
return b
return interner?.intern("$a/$b") ?: "$a/$b"
return interner.intern("$a/$b")
}
fun remap(path: String): String {

View File

@ -26,6 +26,7 @@ class SBPattern private constructor(
val pieces: ImmutableList<Piece>,
val names: ImmutableSet<String>
) {
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<String, String>): SBPattern {
return with(params::get)
}
fun with(params: (String) -> String?): SBPattern {
if (names.isEmpty())
return this
val map = Object2ObjectArrayMap<String, String>()
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<SBPattern>()
@JvmField
val EMPTY = raw("")
override fun <T : Any?> create(gson: Gson, type: TypeToken<T>): TypeAdapter<T>? {
if (type.rawType == SBPattern::class.java) {
return object : TypeAdapter<SBPattern>() {
private val interner = Interners.newWeakInterner<SBPattern>()
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<Piece>()
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())
}
}
}