Немного не туда пошёл, но сохранится надо
This commit is contained in:
parent
2d8e3a7ff5
commit
be87ca7cc1
@ -41,6 +41,7 @@ import ru.dbotthepony.kstarbound.defs.world.dungeon.DungeonWorldDef
|
||||
import ru.dbotthepony.kstarbound.io.*
|
||||
import ru.dbotthepony.kstarbound.io.json.AABBTypeAdapter
|
||||
import ru.dbotthepony.kstarbound.io.json.AABBiTypeAdapter
|
||||
import ru.dbotthepony.kstarbound.io.json.MappedTypeFactories
|
||||
import ru.dbotthepony.kstarbound.io.json.builder.EnumAdapter
|
||||
import ru.dbotthepony.kstarbound.io.json.Vector2dTypeAdapter
|
||||
import ru.dbotthepony.kstarbound.io.json.Vector2fTypeAdapter
|
||||
@ -175,15 +176,15 @@ object Starbound {
|
||||
.also(SpriteReference::registerGson)
|
||||
.also(AtlasConfiguration::registerGson)
|
||||
|
||||
.registerTypeAdapter(LeveledStatusEffect.ADAPTER)
|
||||
.registerTypeAdapterFactory(LeveledStatusEffect.ADAPTER)
|
||||
.registerTypeAdapter(MaterialReference.Companion)
|
||||
.registerTypeAdapter(ThingDescription.Companion)
|
||||
|
||||
.registerTypeAdapter(ItemPrototype.ADAPTER)
|
||||
.registerTypeAdapter(CurrencyItemPrototype.ADAPTER)
|
||||
.registerTypeAdapter(ArmorItemPrototype.ADAPTER)
|
||||
.registerTypeAdapter(MaterialItemPrototype.ADAPTER)
|
||||
.registerTypeAdapter(LiquidItemPrototype.ADAPTER)
|
||||
.registerTypeAdapterFactory(ItemPrototype.ADAPTER)
|
||||
.registerTypeAdapterFactory(CurrencyItemPrototype.ADAPTER)
|
||||
.registerTypeAdapterFactory(ArmorItemPrototype.ADAPTER)
|
||||
.registerTypeAdapterFactory(MaterialItemPrototype.ADAPTER)
|
||||
.registerTypeAdapterFactory(LiquidItemPrototype.ADAPTER)
|
||||
|
||||
.registerTypeAdapter(IItemDefinition.InventoryIcon.ADAPTER)
|
||||
.registerTypeAdapter(IFossilItemDefinition.FossilSetDescription.ADAPTER)
|
||||
|
@ -64,9 +64,8 @@ class ArmorItemPrototype : ItemPrototype(), IArmorItemDefinition {
|
||||
.auto(ArmorItemPrototype::maleFrames)
|
||||
.auto(ArmorItemPrototype::femaleFrames)
|
||||
.auto(ArmorItemPrototype::level)
|
||||
.autoList(ArmorItemPrototype::leveledStatusEffects)
|
||||
.autoList(ArmorItemPrototype::scripts)
|
||||
.auto(ArmorItemPrototype::leveledStatusEffects)
|
||||
.auto(ArmorItemPrototype::scripts)
|
||||
.auto(ArmorItemPrototype::scriptDelta)
|
||||
.build()
|
||||
}
|
||||
}
|
||||
|
@ -51,13 +51,12 @@ class CurrencyItemPrototype : ItemPrototype(), ICurrencyItemDefinition {
|
||||
companion object {
|
||||
val ADAPTER = BuilderAdapter.Builder(::CurrencyItemPrototype)
|
||||
.also { addFields(it as BuilderAdapter.Builder<ItemPrototype>) } // безопасность: свойства родительского класса объявлены как final
|
||||
.autoList(CurrencyItemPrototype::pickupSoundsSmall)
|
||||
.autoList(CurrencyItemPrototype::pickupSoundsMedium)
|
||||
.autoList(CurrencyItemPrototype::pickupSoundsLarge)
|
||||
.auto(CurrencyItemPrototype::pickupSoundsSmall)
|
||||
.auto(CurrencyItemPrototype::pickupSoundsMedium)
|
||||
.auto(CurrencyItemPrototype::pickupSoundsLarge)
|
||||
.auto(CurrencyItemPrototype::smallStackLimit)
|
||||
.auto(CurrencyItemPrototype::mediumStackLimit)
|
||||
.auto(CurrencyItemPrototype::currency)
|
||||
.auto(CurrencyItemPrototype::value)
|
||||
.build()
|
||||
}
|
||||
}
|
||||
|
@ -20,7 +20,6 @@ data class LeveledStatusEffect(
|
||||
LeveledStatusEffect::levelFunction,
|
||||
LeveledStatusEffect::stat,
|
||||
LeveledStatusEffect::baseMultiplier,
|
||||
LeveledStatusEffect::amount,
|
||||
).build()
|
||||
LeveledStatusEffect::amount)
|
||||
}
|
||||
}
|
||||
|
@ -1,5 +1,6 @@
|
||||
package ru.dbotthepony.kstarbound.defs.item
|
||||
|
||||
import com.google.common.collect.ImmutableList
|
||||
import ru.dbotthepony.kstarbound.defs.ThingDescription
|
||||
import ru.dbotthepony.kstarbound.defs.util.enrollMap
|
||||
import ru.dbotthepony.kstarbound.io.json.builder.BuilderAdapter
|
||||
@ -13,16 +14,16 @@ open class ItemPrototype : IItemDefinition, INativeJsonHolder {
|
||||
final override var price: Long = 0L
|
||||
final override var rarity: ItemRarity = ItemRarity.COMMON
|
||||
final override var category: String? = null
|
||||
final override var inventoryIcon: List<IItemDefinition.InventoryIcon>? = null
|
||||
final override var itemTags: List<String> = listOf()
|
||||
final override var learnBlueprintsOnPickup: List<String> = listOf()
|
||||
final override var inventoryIcon: ArrayList<IItemDefinition.InventoryIcon>? = null
|
||||
final override var itemTags: ArrayList<String> = ArrayList()
|
||||
final override var learnBlueprintsOnPickup: ArrayList<String> = ArrayList()
|
||||
final override var maxStack: Long = 9999L
|
||||
final override var eventCategory: String? = null
|
||||
final override var consumeOnPickup: Boolean = false
|
||||
final override var pickupQuestTemplates: List<String> = listOf()
|
||||
final override var pickupQuestTemplates: ArrayList<String> = ArrayList()
|
||||
final override var tooltipKind: ItemTooltipKind = ItemTooltipKind.NORMAL
|
||||
final override var twoHanded: Boolean = false
|
||||
final override var radioMessagesOnPickup: List<String> = listOf()
|
||||
final override var radioMessagesOnPickup: ArrayList<String> = ArrayList()
|
||||
final override var fuelAmount: Long? = null
|
||||
|
||||
var descriptionData: ThingDescription by NotNullVar()
|
||||
@ -59,7 +60,6 @@ open class ItemPrototype : IItemDefinition, INativeJsonHolder {
|
||||
companion object {
|
||||
val ADAPTER = BuilderAdapter.Builder(::ItemPrototype)
|
||||
.also(::addFields)
|
||||
.build()
|
||||
|
||||
fun addFields(builder: BuilderAdapter.Builder<ItemPrototype>) {
|
||||
with(builder) {
|
||||
@ -69,18 +69,18 @@ open class ItemPrototype : IItemDefinition, INativeJsonHolder {
|
||||
auto(ItemPrototype::price)
|
||||
auto(ItemPrototype::rarity)
|
||||
auto(ItemPrototype::category)
|
||||
autoNullableList(ItemPrototype::inventoryIcon)
|
||||
autoList(ItemPrototype::itemTags)
|
||||
autoList(ItemPrototype::learnBlueprintsOnPickup)
|
||||
auto(ItemPrototype::inventoryIcon)
|
||||
auto(ItemPrototype::itemTags)
|
||||
auto(ItemPrototype::learnBlueprintsOnPickup)
|
||||
auto(ItemPrototype::maxStack)
|
||||
auto(ItemPrototype::eventCategory)
|
||||
auto(ItemPrototype::consumeOnPickup)
|
||||
autoList(ItemPrototype::pickupQuestTemplates)
|
||||
auto(ItemPrototype::pickupQuestTemplates)
|
||||
auto(ItemPrototype::tooltipKind)
|
||||
auto(ItemPrototype::twoHanded)
|
||||
autoList(ItemPrototype::radioMessagesOnPickup)
|
||||
auto(ItemPrototype::radioMessagesOnPickup)
|
||||
auto(ItemPrototype::fuelAmount)
|
||||
flat(ItemPrototype::descriptionData)
|
||||
auto(ItemPrototype::descriptionData, isFlat = true)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -49,6 +49,5 @@ class LiquidItemPrototype : ItemPrototype() {
|
||||
.auto(LiquidItemPrototype::liquid)
|
||||
.auto(LiquidItemPrototype::liquidId)
|
||||
.auto(LiquidItemPrototype::liquidName)
|
||||
.build()
|
||||
}
|
||||
}
|
||||
|
@ -49,6 +49,5 @@ class MaterialItemPrototype : ItemPrototype() {
|
||||
.auto(MaterialItemPrototype::material)
|
||||
.auto(MaterialItemPrototype::materialId)
|
||||
.auto(MaterialItemPrototype::materialName)
|
||||
.build()
|
||||
}
|
||||
}
|
||||
|
@ -1,5 +1,6 @@
|
||||
package ru.dbotthepony.kstarbound.defs.liquid
|
||||
|
||||
import com.google.common.collect.ImmutableList
|
||||
import com.google.gson.GsonBuilder
|
||||
import ru.dbotthepony.kstarbound.io.json.builder.FactoryAdapter
|
||||
import ru.dbotthepony.kstarbound.registerTypeAdapter
|
||||
@ -12,8 +13,8 @@ data class LiquidDefinition(
|
||||
val tickDelta: Int = 1,
|
||||
val color: Color,
|
||||
val itemDrop: String? = null,
|
||||
val statusEffects: List<String> = listOf(),
|
||||
val interactions: List<Interaction> = listOf(),
|
||||
val statusEffects: ImmutableList<String> = ImmutableList.of(),
|
||||
val interactions: ImmutableList<Interaction> = ImmutableList.of(),
|
||||
val texture: String,
|
||||
val bottomLightMix: Color,
|
||||
val textureMovementFactor: Double,
|
||||
@ -32,22 +33,20 @@ data class LiquidDefinition(
|
||||
.auto(LiquidDefinition::tickDelta)
|
||||
.auto(LiquidDefinition::color)
|
||||
.auto(LiquidDefinition::itemDrop)
|
||||
.autoList(LiquidDefinition::statusEffects)
|
||||
.autoList(LiquidDefinition::interactions)
|
||||
.auto(LiquidDefinition::statusEffects)
|
||||
.auto(LiquidDefinition::interactions)
|
||||
.auto(LiquidDefinition::texture)
|
||||
.auto(LiquidDefinition::bottomLightMix)
|
||||
.auto(LiquidDefinition::textureMovementFactor)
|
||||
.build()
|
||||
|
||||
val INTERACTION_ADAPTER = FactoryAdapter.Builder(Interaction::class)
|
||||
.auto(Interaction::liquid)
|
||||
.auto(Interaction::liquidResult)
|
||||
.auto(Interaction::materialResult)
|
||||
.build()
|
||||
|
||||
fun registerGson(gsonBuilder: GsonBuilder) {
|
||||
gsonBuilder.registerTypeAdapter(ADAPTER)
|
||||
gsonBuilder.registerTypeAdapter(INTERACTION_ADAPTER)
|
||||
gsonBuilder.registerTypeAdapterFactory(ADAPTER)
|
||||
gsonBuilder.registerTypeAdapterFactory(INTERACTION_ADAPTER)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -37,12 +37,12 @@ class ConfigurableProjectile : RawPrototype<ConfigurableProjectile, ConfiguredPr
|
||||
var bounces: Int = -1
|
||||
var frameNumber: Int = 1
|
||||
|
||||
var scripts: Array<String> = Array(0) { throw IllegalStateException() }
|
||||
var scripts: ArrayList<String> = ArrayList()
|
||||
|
||||
var hydrophobic: Boolean = false
|
||||
|
||||
// we can't have concrete type here, since final class is commanded by `action` property of each entry
|
||||
var actionOnReap: Array<JsonObject>? = null
|
||||
var actionOnReap: ArrayList<JsonObject>? = null
|
||||
|
||||
var piercing = false
|
||||
|
||||
@ -85,7 +85,7 @@ class ConfigurableProjectile : RawPrototype<ConfigurableProjectile, ConfiguredPr
|
||||
animationCycle = animationCycle,
|
||||
bounces = bounces,
|
||||
frameNumber = frameNumber,
|
||||
scripts = scripts,
|
||||
scripts = scripts.toTypedArray(),
|
||||
actionOnReap = ImmutableList.copyOf(actions),
|
||||
animationLoops = animationLoops,
|
||||
hydrophobic = hydrophobic,
|
||||
@ -119,10 +119,10 @@ class ConfigurableProjectile : RawPrototype<ConfigurableProjectile, ConfiguredPr
|
||||
ConfigurableProjectile::piercing,
|
||||
ConfigurableProjectile::speed,
|
||||
ConfigurableProjectile::power,
|
||||
).build()
|
||||
)
|
||||
|
||||
fun registerGson(gson: GsonBuilder) {
|
||||
gson.registerTypeAdapter(ADAPTER)
|
||||
gson.registerTypeAdapterFactory(ADAPTER)
|
||||
gson.registerTypeAdapter(EnumAdapter(ProjectilePhysics::class).neverNull())
|
||||
gson.registerTypeAdapter(ActionConfig.ADAPTER)
|
||||
gson.registerTypeAdapter(ActionProjectile.ADAPTER)
|
||||
|
@ -1,5 +1,6 @@
|
||||
package ru.dbotthepony.kstarbound.defs.tile
|
||||
|
||||
import com.google.common.collect.ImmutableList
|
||||
import com.google.gson.GsonBuilder
|
||||
import ru.dbotthepony.kstarbound.io.json.builder.FactoryAdapter
|
||||
|
||||
@ -12,7 +13,7 @@ data class MaterialModifier(
|
||||
val harvestLevel: Int = 0,
|
||||
val breaksWithTile: Boolean = true,
|
||||
val grass: Boolean = false,
|
||||
val miningSounds: List<String> = listOf(),
|
||||
val miningSounds: ImmutableList<String> = ImmutableList.of(),
|
||||
val miningParticle: String? = null,
|
||||
override val renderTemplate: RenderTemplate,
|
||||
override val renderParameters: RenderParameters
|
||||
@ -31,14 +32,13 @@ data class MaterialModifier(
|
||||
.auto(MaterialModifier::harvestLevel)
|
||||
.auto(MaterialModifier::breaksWithTile)
|
||||
.auto(MaterialModifier::grass)
|
||||
.autoList(MaterialModifier::miningSounds)
|
||||
.auto(MaterialModifier::miningSounds)
|
||||
.auto(MaterialModifier::miningParticle)
|
||||
.add(MaterialModifier::renderTemplate, RenderTemplate.CACHE)
|
||||
.auto(MaterialModifier::renderTemplate)
|
||||
.auto(MaterialModifier::renderParameters)
|
||||
.build()
|
||||
|
||||
fun registerGson(gsonBuilder: GsonBuilder) {
|
||||
gsonBuilder.registerTypeAdapter(MaterialModifier::class.java, ADAPTER)
|
||||
gsonBuilder.registerTypeAdapterFactory(ADAPTER)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -28,18 +28,15 @@ data class RenderParameters(
|
||||
|
||||
companion object {
|
||||
val ADAPTER = FactoryAdapter.Builder(RenderParameters::class)
|
||||
.auto(
|
||||
RenderParameters::texture,
|
||||
RenderParameters::variants,
|
||||
RenderParameters::multiColored,
|
||||
RenderParameters::occludesBelow,
|
||||
RenderParameters::lightTransparent,
|
||||
RenderParameters::zLevel,
|
||||
)
|
||||
.build()
|
||||
.auto(RenderParameters::texture)
|
||||
.auto(RenderParameters::variants)
|
||||
.auto(RenderParameters::multiColored)
|
||||
.auto(RenderParameters::occludesBelow)
|
||||
.auto(RenderParameters::lightTransparent)
|
||||
.auto(RenderParameters::zLevel)
|
||||
|
||||
fun registerGson(gsonBuilder: GsonBuilder) {
|
||||
gsonBuilder.registerTypeAdapter(RenderParameters::class.java, ADAPTER)
|
||||
gsonBuilder.registerTypeAdapterFactory(ADAPTER)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -1,5 +1,7 @@
|
||||
package ru.dbotthepony.kstarbound.defs.tile
|
||||
|
||||
import com.google.common.collect.ImmutableList
|
||||
import com.google.common.collect.ImmutableMap
|
||||
import com.google.gson.GsonBuilder
|
||||
import com.google.gson.JsonSyntaxException
|
||||
import com.google.gson.TypeAdapter
|
||||
@ -28,14 +30,11 @@ data class RenderPiece(
|
||||
) {
|
||||
companion object {
|
||||
val ADAPTER = FactoryAdapter.Builder(RenderPiece::class)
|
||||
.auto(
|
||||
RenderPiece::texture,
|
||||
RenderPiece::textureSize,
|
||||
RenderPiece::texturePosition,
|
||||
RenderPiece::colorStride,
|
||||
RenderPiece::variantStride,
|
||||
)
|
||||
.build()
|
||||
.auto(RenderPiece::texture)
|
||||
.auto(RenderPiece::textureSize)
|
||||
.auto(RenderPiece::texturePosition)
|
||||
.auto(RenderPiece::colorStride)
|
||||
.auto(RenderPiece::variantStride)
|
||||
}
|
||||
}
|
||||
|
||||
@ -44,7 +43,7 @@ fun interface EqualityRuleTester {
|
||||
}
|
||||
|
||||
data class RenderRuleList(
|
||||
val entries: List<Entry>,
|
||||
val entries: ImmutableList<Entry>,
|
||||
val join: Combination = Combination.ALL
|
||||
) {
|
||||
enum class Combination {
|
||||
@ -84,11 +83,9 @@ data class RenderRuleList(
|
||||
private val LOGGED = ObjectArraySet<String>()
|
||||
|
||||
val ADAPTER = FactoryAdapter.Builder(Entry::class)
|
||||
.auto(
|
||||
Entry::type,
|
||||
Entry::matchHue,
|
||||
Entry::inverse,
|
||||
)
|
||||
.auto(Entry::type)
|
||||
.auto(Entry::matchHue)
|
||||
.auto(Entry::inverse)
|
||||
.build()
|
||||
}
|
||||
}
|
||||
@ -119,17 +116,17 @@ data class RenderRuleList(
|
||||
|
||||
companion object {
|
||||
val ADAPTER = FactoryAdapter.Builder(RenderRuleList::class)
|
||||
.autoList(RenderRuleList::entries)
|
||||
.auto(RenderRuleList::entries)
|
||||
.auto(RenderRuleList::join)
|
||||
.build()
|
||||
}
|
||||
}
|
||||
|
||||
data class RenderMatch(
|
||||
val pieces: List<Piece> = listOf(),
|
||||
val matchAllPoints: List<Matcher> = listOf(),
|
||||
val matchAnyPoints: List<Matcher> = listOf(),
|
||||
val subMatches: List<RenderMatch> = listOf(),
|
||||
val pieces: ImmutableList<Piece> = ImmutableList.of(),
|
||||
val matchAllPoints: ImmutableList<Matcher> = ImmutableList.of(),
|
||||
val matchAnyPoints: ImmutableList<Matcher> = ImmutableList.of(),
|
||||
val subMatches: ImmutableList<RenderMatch> = ImmutableList.of(),
|
||||
val haltOnMatch: Boolean = false,
|
||||
val haltOnSubMatch: Boolean = false,
|
||||
) {
|
||||
@ -208,31 +205,28 @@ data class RenderMatch(
|
||||
|
||||
companion object {
|
||||
val ADAPTER = FactoryAdapter.Builder(RenderMatch::class)
|
||||
.autoList(RenderMatch::pieces)
|
||||
.autoList(RenderMatch::matchAllPoints)
|
||||
.autoList(RenderMatch::matchAnyPoints)
|
||||
.autoList(RenderMatch::subMatches)
|
||||
.auto(RenderMatch::pieces)
|
||||
.auto(RenderMatch::matchAllPoints)
|
||||
.auto(RenderMatch::matchAnyPoints)
|
||||
.auto(RenderMatch::subMatches)
|
||||
.auto(RenderMatch::haltOnMatch)
|
||||
.auto(RenderMatch::haltOnSubMatch)
|
||||
.build()
|
||||
|
||||
val PIECE_ADAPTER = FactoryAdapter.Builder(Piece::class)
|
||||
.auto(Piece::name)
|
||||
.auto(Piece::offset)
|
||||
.inputAsList()
|
||||
.build()
|
||||
|
||||
val MATCHER_ADAPTER = FactoryAdapter.Builder(Matcher::class)
|
||||
.auto(Matcher::offset)
|
||||
.auto(Matcher::ruleName)
|
||||
.inputAsList()
|
||||
.build()
|
||||
}
|
||||
}
|
||||
|
||||
data class RenderMatchList(
|
||||
val name: String,
|
||||
val list: List<RenderMatch>
|
||||
val list: ImmutableList<RenderMatch>
|
||||
) {
|
||||
fun resolve(template: RenderTemplate) {
|
||||
for (value in list) {
|
||||
@ -243,17 +237,16 @@ data class RenderMatchList(
|
||||
companion object {
|
||||
val ADAPTER = FactoryAdapter.Builder(RenderMatchList::class)
|
||||
.auto(RenderMatchList::name)
|
||||
.autoList(RenderMatchList::list)
|
||||
.auto(RenderMatchList::list)
|
||||
.inputAsList()
|
||||
.build()
|
||||
}
|
||||
}
|
||||
|
||||
data class RenderTemplate(
|
||||
val pieces: Map<String, RenderPiece>,
|
||||
val pieces: ImmutableMap<String, RenderPiece>,
|
||||
val representativePiece: String,
|
||||
val matches: List<RenderMatchList>,
|
||||
val rules: Map<String, RenderRuleList>,
|
||||
val matches: ImmutableList<RenderMatchList>,
|
||||
val rules: ImmutableMap<String, RenderRuleList>,
|
||||
) {
|
||||
init {
|
||||
for (value in matches) {
|
||||
@ -263,25 +256,23 @@ data class RenderTemplate(
|
||||
|
||||
companion object {
|
||||
val ADAPTER = FactoryAdapter.Builder(RenderTemplate::class)
|
||||
.mapAsObject(RenderTemplate::pieces)
|
||||
.auto(RenderTemplate::pieces)
|
||||
.auto(RenderTemplate::representativePiece)
|
||||
.autoList(RenderTemplate::matches)
|
||||
.mapAsObject(RenderTemplate::rules)
|
||||
.build()
|
||||
|
||||
val CACHE = ADAPTER.asReference()
|
||||
.auto(RenderTemplate::matches)
|
||||
.auto(RenderTemplate::rules)
|
||||
.asReference()
|
||||
|
||||
fun registerGson(gsonBuilder: GsonBuilder) {
|
||||
gsonBuilder.registerTypeAdapter(RenderPiece.ADAPTER)
|
||||
gsonBuilder.registerTypeAdapterFactory(RenderPiece.ADAPTER)
|
||||
gsonBuilder.registerTypeAdapter(RenderRuleList.ADAPTER)
|
||||
gsonBuilder.registerTypeAdapter(RenderMatch.ADAPTER)
|
||||
gsonBuilder.registerTypeAdapter(RenderMatch.PIECE_ADAPTER)
|
||||
gsonBuilder.registerTypeAdapter(RenderMatch.MATCHER_ADAPTER)
|
||||
gsonBuilder.registerTypeAdapter(RenderMatchList.ADAPTER)
|
||||
gsonBuilder.registerTypeAdapterFactory(RenderMatch.ADAPTER)
|
||||
gsonBuilder.registerTypeAdapterFactory(RenderMatch.PIECE_ADAPTER)
|
||||
gsonBuilder.registerTypeAdapterFactory(RenderMatch.MATCHER_ADAPTER)
|
||||
gsonBuilder.registerTypeAdapterFactory(RenderMatchList.ADAPTER)
|
||||
|
||||
gsonBuilder.registerTypeAdapter(RenderRuleList.Entry.ADAPTER)
|
||||
|
||||
gsonBuilder.registerTypeAdapter(ADAPTER)
|
||||
gsonBuilder.registerTypeAdapterFactory(ADAPTER)
|
||||
|
||||
gsonBuilder.registerTypeAdapter(EnumAdapter(RenderRuleList.Combination::class.java))
|
||||
}
|
||||
|
@ -25,27 +25,22 @@ data class TileDefinition(
|
||||
) : IRenderableTile {
|
||||
companion object {
|
||||
val ADAPTER = FactoryAdapter.Builder(TileDefinition::class)
|
||||
.auto(
|
||||
TileDefinition::materialId,
|
||||
TileDefinition::materialName,
|
||||
TileDefinition::particleColor,
|
||||
TileDefinition::itemDrop,
|
||||
TileDefinition::description,
|
||||
TileDefinition::shortdescription,
|
||||
TileDefinition::footstepSound,
|
||||
TileDefinition::blocksLiquidFlow,
|
||||
TileDefinition::soil,
|
||||
TileDefinition::health,
|
||||
TileDefinition::category
|
||||
)
|
||||
.add(TileDefinition::renderTemplate, RenderTemplate.CACHE)
|
||||
.auto(
|
||||
TileDefinition::renderParameters,
|
||||
)
|
||||
.build()
|
||||
.auto(TileDefinition::materialId)
|
||||
.auto(TileDefinition::materialName)
|
||||
.auto(TileDefinition::particleColor)
|
||||
.auto(TileDefinition::itemDrop)
|
||||
.auto(TileDefinition::description)
|
||||
.auto(TileDefinition::shortdescription)
|
||||
.auto(TileDefinition::footstepSound)
|
||||
.auto(TileDefinition::blocksLiquidFlow)
|
||||
.auto(TileDefinition::soil)
|
||||
.auto(TileDefinition::health)
|
||||
.auto(TileDefinition::category)
|
||||
.auto(TileDefinition::renderTemplate)
|
||||
.auto(TileDefinition::renderParameters)
|
||||
|
||||
fun registerGson(gsonBuilder: GsonBuilder) {
|
||||
gsonBuilder.registerTypeAdapter(ADAPTER)
|
||||
gsonBuilder.registerTypeAdapterFactory(ADAPTER)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -0,0 +1,26 @@
|
||||
package ru.dbotthepony.kstarbound.io.json
|
||||
|
||||
import com.google.gson.Gson
|
||||
import com.google.gson.TypeAdapter
|
||||
import com.google.gson.TypeAdapterFactory
|
||||
import com.google.gson.reflect.TypeToken
|
||||
import it.unimi.dsi.fastutil.objects.Reference2ObjectOpenHashMap
|
||||
|
||||
class MappedTypeFactories : TypeAdapterFactory {
|
||||
private val adapters = Reference2ObjectOpenHashMap<Class<*>, TypeAdapterFactory>()
|
||||
|
||||
fun <T> put(clazz: Class<T>, adapter: TypeAdapterFactory): MappedTypeFactories {
|
||||
check(adapters.put(clazz, adapter) == null) { "Already had type adapter for $clazz" }
|
||||
return this
|
||||
}
|
||||
|
||||
override fun <T : Any?> create(gson: Gson, type: TypeToken<T>): TypeAdapter<T>? {
|
||||
val factory = adapters[type.rawType]
|
||||
|
||||
if (factory != null) {
|
||||
return checkNotNull(factory.create(gson, type)) { "${type.rawType} should have type adapter (factory: $factory)" }
|
||||
}
|
||||
|
||||
return null
|
||||
}
|
||||
}
|
@ -25,6 +25,7 @@ import kotlin.properties.Delegates
|
||||
import kotlin.reflect.KClass
|
||||
import kotlin.reflect.KMutableProperty1
|
||||
import kotlin.reflect.full.isSubclassOf
|
||||
import kotlin.reflect.javaType
|
||||
|
||||
/**
|
||||
* Данный интерфейс имеет один единственный метод: [acceptJson]
|
||||
@ -72,12 +73,7 @@ class BuilderAdapter<T : Any> private constructor(
|
||||
/**
|
||||
* Свойства объекта [T], которые можно выставлять
|
||||
*/
|
||||
val properties: ImmutableMap<String, WrappedProperty<T, *>>,
|
||||
|
||||
/**
|
||||
* Свойства объекта [T], которые можно выставлять
|
||||
*/
|
||||
val flatProperties: ImmutableMap<String, WrappedProperty<T, *>>,
|
||||
val properties: ImmutableMap<String, IResolvedMutableProperty<T, *>>,
|
||||
|
||||
/**
|
||||
* Ключи, которые необходимо игнорировать при чтении JSON
|
||||
@ -104,13 +100,13 @@ class BuilderAdapter<T : Any> private constructor(
|
||||
@Suppress("name_shadowing")
|
||||
var reader = reader
|
||||
|
||||
val missing = ObjectOpenHashSet<WrappedProperty<T, *>>()
|
||||
val missing = ObjectOpenHashSet<IResolvedMutableProperty<T, *>>()
|
||||
missing.addAll(properties.values)
|
||||
|
||||
val instance = factory.invoke()
|
||||
var json: JsonObject by Delegates.notNull()
|
||||
|
||||
if (instance is IJsonHolder || flatProperties.isNotEmpty()) {
|
||||
if (instance is IJsonHolder || properties.values.any { it.isFlat }) {
|
||||
json = TypeAdapters.JSON_ELEMENT.read(reader) as JsonObject
|
||||
reader = JsonTreeReader(json)
|
||||
|
||||
@ -137,7 +133,7 @@ class BuilderAdapter<T : Any> private constructor(
|
||||
try {
|
||||
val peek = reader.peek()
|
||||
|
||||
if (!property.returnType.isMarkedNullable && peek == JsonToken.NULL) {
|
||||
if (!property.type.isMarkedNullable && peek == JsonToken.NULL) {
|
||||
throw NullPointerException("Property ${property.name} of ${instance::class.qualifiedName} does not accept nulls (JSON contains null)")
|
||||
} else if (peek == JsonToken.NULL) {
|
||||
property.set(instance, null)
|
||||
@ -145,7 +141,7 @@ class BuilderAdapter<T : Any> private constructor(
|
||||
} else {
|
||||
val readValue = property.adapter.read(reader)
|
||||
|
||||
if (!property.returnType.isMarkedNullable && readValue == null)
|
||||
if (!property.type.isMarkedNullable && readValue == null)
|
||||
throw JsonSyntaxException("Property ${property.name} of ${instance::class.qualifiedName} does not accept nulls (Type provider returned null)")
|
||||
|
||||
property.set(instance, readValue)
|
||||
@ -169,11 +165,14 @@ class BuilderAdapter<T : Any> private constructor(
|
||||
|
||||
reader.endObject()
|
||||
|
||||
for (property in flatProperties.values) {
|
||||
for (property in properties.values) {
|
||||
if (!property.isFlat || !missing.remove(property))
|
||||
continue
|
||||
|
||||
try {
|
||||
val read = property.adapter.read(JsonTreeReader(json))
|
||||
|
||||
if (!property.returnType.isMarkedNullable && read == null) {
|
||||
if (!property.type.isMarkedNullable && read == null) {
|
||||
throw NullPointerException("Property ${property.name} of ${instance::class.qualifiedName} does not accept nulls (flat property adapter returned NULL)")
|
||||
} else if (read == null) {
|
||||
property.set(instance, null)
|
||||
@ -189,7 +188,7 @@ class BuilderAdapter<T : Any> private constructor(
|
||||
if (property.mustBePresent == true) {
|
||||
throw JsonSyntaxException("${instance::class.qualifiedName} demands for ${property.name} to be present, however, it is missing from JSON structure")
|
||||
} else if (property.mustBePresent == null) {
|
||||
if (property.returnType.isMarkedNullable) {
|
||||
if (property.type.isMarkedNullable) {
|
||||
continue
|
||||
}
|
||||
|
||||
@ -210,51 +209,53 @@ class BuilderAdapter<T : Any> private constructor(
|
||||
return instance
|
||||
}
|
||||
|
||||
data class WrappedProperty<T, V : Any?>(
|
||||
val property: KMutableProperty1<T, V>,
|
||||
val adapter: TypeAdapter<V>,
|
||||
/**
|
||||
* @see PropertyConfigurator.mustBePresent
|
||||
*/
|
||||
val mustBePresent: Boolean?,
|
||||
) {
|
||||
inline val name get() = property.name
|
||||
// кеш
|
||||
val returnType = property.returnType
|
||||
|
||||
// так как дженерики тут немного слабенькие
|
||||
// Так что вот так...
|
||||
@Suppress("unchecked_cast")
|
||||
fun set(receiver: T, value: Any?) {
|
||||
property.set(receiver, value as V)
|
||||
}
|
||||
}
|
||||
|
||||
class PropertyConfigurator<T, V : Any?>(
|
||||
val property: KMutableProperty1<T, V>,
|
||||
val adapter: TypeAdapter<V>,
|
||||
) {
|
||||
/**
|
||||
* Обязана ли присутствовать эта переменная внутри JSON структуры.
|
||||
*
|
||||
* * `true` - всегда кидать исключения
|
||||
* * `false` - никогда не кидать исключения
|
||||
* * `null` - кидать исключения на усмотрение реализации (по умолчанию)
|
||||
*/
|
||||
var mustBePresent: Boolean? = null
|
||||
}
|
||||
|
||||
class Builder<T : Any>(val factory: () -> T, vararg fields: KMutableProperty1<T, *>) : TypeAdapterFactory {
|
||||
private val properties = ArrayList<WrappedProperty<T, *>>()
|
||||
private val flatProperties = ArrayList<WrappedProperty<T, *>>()
|
||||
private val properties = ArrayList<IResolvableMutableProperty<T, *>>()
|
||||
private val ignoreKeys = ObjectArraySet<String>()
|
||||
var extraPropertiesAreFatal = false
|
||||
var logMisses: Boolean? = null
|
||||
private val factoryReturnType by lazy { factory::invoke.returnType }
|
||||
|
||||
@OptIn(ExperimentalStdlibApi::class)
|
||||
override fun <T : Any?> create(gson: Gson, type: TypeToken<T>): TypeAdapter<T>? {
|
||||
if (type.isAssignableFrom(factoryReturnType.javaType)) {
|
||||
return build(gson) as TypeAdapter<T>
|
||||
}
|
||||
|
||||
return null
|
||||
}
|
||||
|
||||
fun build(gson: Gson): BuilderAdapter<T> {
|
||||
val map = ImmutableMap.Builder<String, IResolvedMutableProperty<T, *>>()
|
||||
|
||||
for (property in properties)
|
||||
map.put(property.property.name, property.resolve(gson))
|
||||
|
||||
return BuilderAdapter(
|
||||
factory = factory,
|
||||
properties = map.build(),
|
||||
ignoreKeys = ImmutableSet.copyOf(ignoreKeys),
|
||||
extraPropertiesAreFatal = extraPropertiesAreFatal,
|
||||
logMisses = logMisses ?: properties.none { it.isFlat },
|
||||
)
|
||||
}
|
||||
|
||||
@Deprecated("Используйте как TypeAdapterFactory")
|
||||
fun build(): BuilderAdapter<T> {
|
||||
val map = ImmutableMap.Builder<String, IResolvedMutableProperty<T, *>>()
|
||||
|
||||
for (property in properties)
|
||||
map.put(property.property.name, property.resolve(null))
|
||||
|
||||
return BuilderAdapter(
|
||||
factory = factory,
|
||||
properties = map.build(),
|
||||
ignoreKeys = ImmutableSet.copyOf(ignoreKeys),
|
||||
extraPropertiesAreFatal = extraPropertiesAreFatal,
|
||||
logMisses = logMisses ?: properties.none { it.isFlat },
|
||||
)
|
||||
}
|
||||
|
||||
/**
|
||||
* Являются ли "лишние" ключи в JSON структуре ошибкой.
|
||||
*
|
||||
@ -262,7 +263,7 @@ class BuilderAdapter<T : Any> private constructor(
|
||||
* то [extraPropertiesAreFatal] можно скомбинировать с [ignoreKey].
|
||||
*/
|
||||
fun extraPropertiesAreFatal(flag: Boolean = true): Builder<T> {
|
||||
check(flatProperties.isEmpty() || !flag) { "Can't have both flattened properties and extraPropertiesAreFatal" }
|
||||
check(properties.none { it.isFlat } || !flag) { "Can't have both flattened properties and extraPropertiesAreFatal" }
|
||||
extraPropertiesAreFatal = flag
|
||||
return this
|
||||
}
|
||||
@ -280,42 +281,10 @@ class BuilderAdapter<T : Any> private constructor(
|
||||
auto(field)
|
||||
}
|
||||
|
||||
private fun <V> _add(property: KMutableProperty1<T, V>, adapter: TypeAdapter<V>, configurator: PropertyConfigurator<T, V>.() -> Unit): PropertyConfigurator<T, *> {
|
||||
if (properties.any { it.property == property }) {
|
||||
throw IllegalArgumentException("Property $property is defined twice")
|
||||
}
|
||||
|
||||
if (flatProperties.any { it.property == property }) {
|
||||
throw IllegalArgumentException("Property $property is defined twice")
|
||||
}
|
||||
|
||||
ignoreKeys.remove(property.name)
|
||||
|
||||
val config = PropertyConfigurator(property, adapter)
|
||||
configurator.invoke(config)
|
||||
return config
|
||||
}
|
||||
|
||||
/**
|
||||
* Добавляет указанное свойство в будущий адаптер с указанным [adapter]
|
||||
*/
|
||||
fun <V> add(property: KMutableProperty1<T, V>, adapter: TypeAdapter<V>, configurator: PropertyConfigurator<T, V>.() -> Unit = {}): Builder<T> {
|
||||
val config = _add(property, adapter, configurator)
|
||||
|
||||
properties.add(
|
||||
WrappedProperty(
|
||||
property,
|
||||
adapter,
|
||||
mustBePresent = config.mustBePresent
|
||||
)
|
||||
)
|
||||
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
* Добавляет указанное свойство в будущий адаптер, как плоский объект внутри данного на том же уровне, что и сам объект, используя адаптер [adapter]
|
||||
*
|
||||
* Если указан [isFlat] как true, то данное свойство будет обработано как на одном уровне с данным объектом.
|
||||
* Пример:
|
||||
* ```json
|
||||
* {
|
||||
@ -328,77 +297,45 @@ class BuilderAdapter<T : Any> private constructor(
|
||||
* В данном случае, можно указать `b` как плоский класс внутри `a`.
|
||||
*
|
||||
* Данный подход позволяет избавиться от постоянного наследования и реализации одного и того же интерфейса во множестве других классов.
|
||||
*
|
||||
* Флаг [extraPropertiesAreFatal] не поддерживается с данными свойствами
|
||||
*
|
||||
* Если [logMisses] не указан явно, то он будет выставлен на false
|
||||
* Флаг [extraPropertiesAreFatal] не поддерживается с данными свойствами.
|
||||
* Если [logMisses] не указан явно, то он будет выставлен на false.
|
||||
*/
|
||||
fun <V> flat(property: KMutableProperty1<T, V>, adapter: TypeAdapter<V>, configurator: PropertyConfigurator<T, V>.() -> Unit = {}): Builder<T> {
|
||||
val config = _add(property, adapter, configurator)
|
||||
fun <V> add(property: KMutableProperty1<T, V>, adapter: TypeAdapter<V>, isFlat: Boolean = false, mustBePresent: Boolean? = null): Builder<T> {
|
||||
if (properties.any { it.property == property }) {
|
||||
throw IllegalArgumentException("Property $property is defined twice")
|
||||
}
|
||||
|
||||
check(!extraPropertiesAreFatal) { "Can't have both flattened properties and extraPropertiesAreFatal" }
|
||||
|
||||
flatProperties.add(
|
||||
WrappedProperty(
|
||||
property,
|
||||
adapter,
|
||||
mustBePresent = config.mustBePresent
|
||||
)
|
||||
)
|
||||
ignoreKeys.remove(property.name)
|
||||
properties.add(ResolvedMutableProperty(
|
||||
property = property,
|
||||
adapter = adapter,
|
||||
mustBePresent = mustBePresent,
|
||||
isFlat = isFlat,
|
||||
))
|
||||
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
* Автоматически определяет тип свойства и необходимый [TypeAdapter]
|
||||
*
|
||||
* Для флагов смотрите [add]
|
||||
*
|
||||
* @see add
|
||||
*/
|
||||
fun <V> auto(property: KMutableProperty1<T, V>, configurator: PropertyConfigurator<T, V>.() -> Unit = {}): Builder<T> {
|
||||
val returnType = property.returnType
|
||||
val classifier = returnType.classifier as? KClass<*> ?: throw ClassCastException("Unable to cast ${returnType.classifier} to KClass of property ${property.name}!")
|
||||
|
||||
if (classifier.isSubclassOf(List::class)) {
|
||||
throw IllegalArgumentException("${property.name} is a List, please use autoList() or directly specify type adapter method instead")
|
||||
fun <V> auto(property: KMutableProperty1<T, V>, isFlat: Boolean = false, mustBePresent: Boolean? = null): Builder<T> {
|
||||
if (properties.any { it.property == property }) {
|
||||
throw IllegalArgumentException("Property $property is defined twice")
|
||||
}
|
||||
|
||||
if (classifier.isSubclassOf(Map::class)) {
|
||||
throw IllegalArgumentException("${property.name} is a Map, please use autoMap() or directly specify type adapter method instead")
|
||||
}
|
||||
ignoreKeys.remove(property.name)
|
||||
properties.add(ResolvableMutableProperty(
|
||||
property = property,
|
||||
mustBePresent = mustBePresent,
|
||||
isFlat = isFlat,
|
||||
))
|
||||
|
||||
@Suppress("unchecked_cast") // classifier.java не имеет обозначенного типа
|
||||
return add(property, LazyTypeProvider(classifier.java) as TypeAdapter<V>, configurator)
|
||||
}
|
||||
|
||||
/**
|
||||
* Автоматически определяет тип свойства и необходимый [TypeAdapter], инкапсулированный в данный типе на одном уровне
|
||||
*/
|
||||
fun <V> flat(property: KMutableProperty1<T, V>, configurator: PropertyConfigurator<T, V>.() -> Unit = {}): Builder<T> {
|
||||
val returnType = property.returnType
|
||||
val classifier = returnType.classifier as? KClass<*> ?: throw ClassCastException("Unable to cast ${returnType.classifier} to KClass of property ${property.name}!")
|
||||
|
||||
if (classifier.isSubclassOf(List::class)) {
|
||||
throw IllegalArgumentException("${property.name} is a List, please use autoList() or directly specify type adapter method instead")
|
||||
}
|
||||
|
||||
if (classifier.isSubclassOf(Map::class)) {
|
||||
throw IllegalArgumentException("${property.name} is a Map, please use autoMap() or directly specify type adapter method instead")
|
||||
}
|
||||
|
||||
@Suppress("unchecked_cast") // classifier.java не имеет обозначенного типа
|
||||
return flat(property, LazyTypeProvider(classifier.java) as TypeAdapter<V>, configurator)
|
||||
}
|
||||
|
||||
/**
|
||||
* Автоматически создаёт [ListAdapter] для заданного свойства
|
||||
*/
|
||||
inline fun <reified V : Any> autoList(property: KMutableProperty1<T, List<V>>, noinline configurator: PropertyConfigurator<T, List<V>>.() -> Unit = {}): Builder<T> {
|
||||
return add(property, ListAdapter(V::class.java), configurator)
|
||||
}
|
||||
|
||||
/**
|
||||
* Автоматически создаёт [ListAdapter] для заданного свойства, но в данном случае само свойство может принимать значение null
|
||||
*/
|
||||
inline fun <reified V : Any> autoNullableList(property: KMutableProperty1<T, List<V>?>, noinline configurator: PropertyConfigurator<T, List<V>?>.() -> Unit = {}): Builder<T> {
|
||||
return add(property, ListAdapter(V::class.java).nullSafe(), configurator)
|
||||
return this
|
||||
}
|
||||
|
||||
fun ignoreKey(name: String): Builder<T> {
|
||||
@ -409,27 +346,6 @@ class BuilderAdapter<T : Any> private constructor(
|
||||
ignoreKeys.add(name)
|
||||
return this
|
||||
}
|
||||
|
||||
fun build(): BuilderAdapter<T> {
|
||||
val map = ImmutableMap.Builder<String, WrappedProperty<T, *>>()
|
||||
|
||||
for (property in properties)
|
||||
map.put(property.property.name, property)
|
||||
|
||||
val map2 = ImmutableMap.Builder<String, WrappedProperty<T, *>>()
|
||||
|
||||
for (property in flatProperties)
|
||||
map2.put(property.property.name, property)
|
||||
|
||||
return BuilderAdapter(
|
||||
factory = factory,
|
||||
properties = map.build(),
|
||||
flatProperties = map2.build(),
|
||||
ignoreKeys = ImmutableSet.copyOf(ignoreKeys),
|
||||
extraPropertiesAreFatal = extraPropertiesAreFatal,
|
||||
logMisses = logMisses ?: flatProperties.isEmpty(),
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
companion object {
|
||||
|
@ -40,20 +40,11 @@ import kotlin.reflect.full.isSupertypeOf
|
||||
*/
|
||||
class FactoryAdapter<T : Any> private constructor(
|
||||
val bound: KClass<T>,
|
||||
private val types: ImmutableList<PackedProperty<T, *>>,
|
||||
val types: ImmutableList<IResolvedProperty<T, *>>,
|
||||
val asJsonArray: Boolean,
|
||||
val storesJson: Boolean,
|
||||
val logMisses: Boolean,
|
||||
) : TypeAdapter<T>() {
|
||||
private data class PackedProperty<Clazz : Any, T>(
|
||||
val property: KProperty1<Clazz, T>,
|
||||
val adapter: TypeAdapter<T>,
|
||||
val transformer: (T) -> T = { it },
|
||||
val isFlat: Boolean
|
||||
) {
|
||||
val returnType = property.returnType
|
||||
}
|
||||
|
||||
private val name2index = Object2IntArrayMap<String>()
|
||||
private val loggedMisses = ObjectArraySet<String>()
|
||||
|
||||
@ -83,7 +74,7 @@ class FactoryAdapter<T : Any> private constructor(
|
||||
val nextParam = iterator.next()
|
||||
|
||||
val a = param.type
|
||||
val b = nextParam.returnType
|
||||
val b = nextParam.type
|
||||
|
||||
if (!a.isSupertypeOf(b) || a.isMarkedNullable != b.isMarkedNullable) {
|
||||
return@first false
|
||||
@ -114,7 +105,7 @@ class FactoryAdapter<T : Any> private constructor(
|
||||
* Синтетический конструктор класса, который создаётся Kotlin'ном, для создания классов со значениями по умолчанию
|
||||
*/
|
||||
private val syntheticFactory: Constructor<T>? = try {
|
||||
val typelist = types.map { (it.returnType.classifier as KClass<*>).java }.toMutableList()
|
||||
val typelist = types.map { (it.type.classifier as KClass<*>).java }.toMutableList()
|
||||
|
||||
if (storesJson)
|
||||
if (asJsonArray)
|
||||
@ -223,7 +214,7 @@ class FactoryAdapter<T : Any> private constructor(
|
||||
val (field, adapter) = tuple
|
||||
|
||||
try {
|
||||
readValues[fieldId] = (tuple.transformer as (Any?) -> Any?)(adapter.read(reader))
|
||||
readValues[fieldId] = adapter.read(reader)
|
||||
presentValues[fieldId] = true
|
||||
} catch(err: Throwable) {
|
||||
throw JsonSyntaxException("Reading field ${field.name} for ${bound.qualifiedName}", err)
|
||||
@ -275,7 +266,7 @@ class FactoryAdapter<T : Any> private constructor(
|
||||
val (field, adapter) = tuple
|
||||
|
||||
try {
|
||||
readValues[fieldId] = (tuple.transformer as (Any?) -> Any?)(adapter.read(reader))
|
||||
readValues[fieldId] = adapter.read(reader)
|
||||
presentValues[fieldId] = true
|
||||
} catch(err: Throwable) {
|
||||
throw JsonSyntaxException("Reading field ${field.name} for ${bound.qualifiedName}", err)
|
||||
@ -312,7 +303,7 @@ class FactoryAdapter<T : Any> private constructor(
|
||||
val (field) = tuple
|
||||
|
||||
if (readValues[i] == null) {
|
||||
if (!tuple.returnType.isMarkedNullable) {
|
||||
if (!tuple.type.isMarkedNullable) {
|
||||
throw JsonSyntaxException("Field ${field.name} of ${bound.qualifiedName} does not accept nulls")
|
||||
}
|
||||
|
||||
@ -366,7 +357,7 @@ class FactoryAdapter<T : Any> private constructor(
|
||||
throw JsonSyntaxException("Field ${field.name} of ${bound.qualifiedName} is missing")
|
||||
}
|
||||
|
||||
if (tuple.returnType.isMarkedNullable) {
|
||||
if (tuple.type.isMarkedNullable) {
|
||||
continue
|
||||
}
|
||||
|
||||
@ -392,16 +383,51 @@ class FactoryAdapter<T : Any> private constructor(
|
||||
}
|
||||
}
|
||||
|
||||
private val types = ArrayList<PackedProperty<T, *>>()
|
||||
private val types = ArrayList<IResolvableProperty<T, *>>()
|
||||
|
||||
override fun <T : Any?> create(gson: Gson, type: TypeToken<T>): TypeAdapter<T>? {
|
||||
if (type.rawType == clazz.java) {
|
||||
return build() as TypeAdapter<T>
|
||||
return build(gson) as TypeAdapter<T>
|
||||
}
|
||||
|
||||
return null
|
||||
}
|
||||
|
||||
/**
|
||||
* Собирает этот [FactoryAdapter] с указанным GSON объектом
|
||||
*/
|
||||
fun build(gson: Gson): FactoryAdapter<T> {
|
||||
check(!asList || types.none { it.isFlat }) { "Can't have both flat properties and json data array layout" }
|
||||
|
||||
return FactoryAdapter(
|
||||
bound = clazz,
|
||||
types = types.stream().map { it.resolve(gson) }.collect(ImmutableList.toImmutableList()),
|
||||
asJsonArray = asList,
|
||||
storesJson = storesJson,
|
||||
logMisses = logMisses,
|
||||
)
|
||||
}
|
||||
|
||||
/**
|
||||
* Собирает этот [FactoryAdapter] без GSON объекта
|
||||
*
|
||||
* Не рекомендуется использовать, лучше всего использовать как [TypeAdapterFactory]
|
||||
*
|
||||
* Несмотря на @Deprecated, данный вариант метода удалён не будет
|
||||
*/
|
||||
@Deprecated("Используйте как TypeAdapterFactory")
|
||||
fun build(): FactoryAdapter<T> {
|
||||
check(!asList || types.none { it.isFlat }) { "Can't have both flat properties and json data array layout" }
|
||||
|
||||
return FactoryAdapter(
|
||||
bound = clazz,
|
||||
types = types.stream().map { it.resolve(null) }.collect(ImmutableList.toImmutableList()),
|
||||
asJsonArray = asList,
|
||||
storesJson = storesJson,
|
||||
logMisses = logMisses,
|
||||
)
|
||||
}
|
||||
|
||||
/**
|
||||
* Принимает ли класс *последним* аргументом JSON структуру
|
||||
*
|
||||
@ -433,160 +459,19 @@ class FactoryAdapter<T : Any> private constructor(
|
||||
/**
|
||||
* Добавляет свойство с определённым [adapter]
|
||||
*/
|
||||
fun <V> add(field: KProperty1<T, V>, adapter: TypeAdapter<V>): Builder<T> {
|
||||
types.add(PackedProperty(field, adapter, isFlat = false))
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
* Добавляет свойство с определённым [adapter], которое находится на том же уровне что и данный объект внутри JSON структуры
|
||||
*/
|
||||
fun <V> flat(field: KProperty1<T, V>, adapter: TypeAdapter<V>): Builder<T> {
|
||||
types.add(PackedProperty(field, adapter, isFlat = true))
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
* Добавляет поля без generic типов и без преобразователей
|
||||
*/
|
||||
@Suppress("unchecked_cast")
|
||||
fun auto(vararg fields: KProperty1<T, *>): Builder<T> {
|
||||
for (field in fields)
|
||||
auto(field)
|
||||
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
* Добавляет поля без generic типов и без преобразователей
|
||||
*/
|
||||
@Suppress("unchecked_cast")
|
||||
fun autoFlat(vararg fields: KProperty1<T, *>): Builder<T> {
|
||||
for (field in fields)
|
||||
autoFlat(field)
|
||||
|
||||
fun <V> add(field: KProperty1<T, V>, adapter: TypeAdapter<V>, isFlat: Boolean = false): Builder<T> {
|
||||
types.add(ResolvedProperty(field, adapter, isFlat = isFlat))
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
* Автоматически определяет тип поля и необходимый адаптор типа к нему
|
||||
*
|
||||
* Можно указать [transform] для изменения определённого адаптера
|
||||
*/
|
||||
@Suppress("unchecked_cast")
|
||||
fun <In> auto(field: KProperty1<T, In>, transformer: (In) -> In = { it }): Builder<T> {
|
||||
val returnType = field.returnType
|
||||
val classifier = returnType.classifier as? KClass<*> ?: throw ClassCastException("Unable to cast ${returnType.classifier} to KClass of property ${field.name}!")
|
||||
|
||||
if (classifier.isSubclassOf(List::class)) {
|
||||
throw IllegalArgumentException("${field.name} is a List, please use autoList() method instead")
|
||||
}
|
||||
|
||||
if (classifier.isSubclassOf(Map::class)) {
|
||||
throw IllegalArgumentException("${field.name} is a Map, please use autoMap() method instead")
|
||||
}
|
||||
|
||||
types.add(PackedProperty(field, LazyTypeProvider(classifier.java) as TypeAdapter<Any?>, transformer = transformer as (Any?) -> Any?, isFlat = false))
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
* Автоматически определяет тип поля и необходимый адаптор типа к нему
|
||||
*/
|
||||
@Suppress("unchecked_cast")
|
||||
fun <In> autoFlat(field: KProperty1<T, In>, transformer: (In) -> In = { it }): Builder<T> {
|
||||
val returnType = field.returnType
|
||||
val classifier = returnType.classifier as? KClass<*> ?: throw ClassCastException("Unable to cast ${returnType.classifier} to KClass of property ${field.name}!")
|
||||
|
||||
if (classifier.isSubclassOf(List::class)) {
|
||||
throw IllegalArgumentException("${field.name} is a List")
|
||||
}
|
||||
|
||||
if (classifier.isSubclassOf(Map::class)) {
|
||||
throw IllegalArgumentException("${field.name} is a Map")
|
||||
}
|
||||
|
||||
types.add(PackedProperty(field, LazyTypeProvider(classifier.java) as TypeAdapter<Any?>, transformer = transformer as (Any?) -> Any?, isFlat = true))
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
* Добавляет поле, которое содержит список значений V (без null).
|
||||
*
|
||||
* Список неизменяем (создаётся объект [ImmutableList])
|
||||
*/
|
||||
fun <V : Any> list(field: KProperty1<T, List<V>?>, type: Class<V>, transformer: (V) -> V = { it }): Builder<T> {
|
||||
types.add(PackedProperty(field, ListAdapter(type, transformer).nullSafe(), isFlat = false))
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
* Добавляет поле, которое содержит список значений V (без null).
|
||||
*
|
||||
* Список неизменяем (создаётся объект [ImmutableList])
|
||||
*/
|
||||
inline fun <reified V : Any> autoList(field: KProperty1<T, List<V>?>, noinline transformer: (V) -> V = { it }): Builder<T> {
|
||||
return add(field, ListAdapter(V::class.java, transformer).nullSafe())
|
||||
}
|
||||
|
||||
/**
|
||||
* Добавляет поле-таблицу, которое кодируется как [[key, value], [key, value], ...]
|
||||
*
|
||||
* Таблица неизменяема (создаётся объект [ImmutableMap])
|
||||
*/
|
||||
fun <K, V> mapAsArray(field: KProperty1<T, Map<K, V>>, keyType: Class<K>, valueType: Class<V>): Builder<T> {
|
||||
types.add(PackedProperty(field, MapAdapter(keyType, valueType), isFlat = false))
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
* Добавляет поле-таблицу, кодирование зависит от контекста:
|
||||
* * Если [K] является [String], кодирование происходит как Json Object
|
||||
* * Иначе кодирование происходит как Json Array
|
||||
*
|
||||
* Таблица неизменяема (создаётся объект [ImmutableMap])
|
||||
*/
|
||||
inline fun <reified K, reified V> autoMap(field: KProperty1<T, Map<K, V>>): Builder<T> {
|
||||
if (K::class == String::class)
|
||||
return this.mapAsObject(field as KProperty1<T, Map<String, V>?>, V::class.java)
|
||||
|
||||
return this.mapAsArray(field, K::class.java, V::class.java)
|
||||
}
|
||||
|
||||
/**
|
||||
* Добавляет поле-таблицу, которое кодируется как [[key, value], [key, value], ...]
|
||||
*
|
||||
* Таблица неизменяема (создаётся объект [ImmutableMap])
|
||||
*/
|
||||
fun <K : Any, V : Any> mapAsArray(field: KProperty1<T, Map<K, V>?>, keyType: KClass<K>, valueType: KClass<V>): Builder<T> {
|
||||
types.add(PackedProperty(field, MapAdapter(keyType.java, valueType.java).nullSafe(), isFlat = false))
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
* Добавляет поле-таблицу, которое кодируется как {"a": value, "b": value, ...}
|
||||
*
|
||||
* Таблица неизменяема (создаётся объект [ImmutableMap])
|
||||
*/
|
||||
fun <V> mapAsObject(field: KProperty1<T, Map<String, V>?>, valueType: Class<V>): Builder<T> {
|
||||
types.add(PackedProperty(field, String2ObjectAdapter(valueType).nullSafe(), isFlat = false))
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
* Добавляет поле-таблицу, которое кодируется как {"a": value, "b": value, ...}
|
||||
*
|
||||
* Таблица неизменяема (создаётся объект [ImmutableMap])
|
||||
*/
|
||||
inline fun <reified V> mapAsObject(field: KProperty1<T, Map<String, V>?>): Builder<T> {
|
||||
return mapAsObject(field, V::class.java)
|
||||
}
|
||||
|
||||
/**
|
||||
* Добавляет поле-таблицу, которое кодируется как {"a": value, "b": value, ...}
|
||||
*
|
||||
* Таблица неизменяема (создаётся объект [ImmutableMap])
|
||||
*/
|
||||
fun <V : Any> mapAsObject(field: KProperty1<T, Map<String, V>?>, valueType: KClass<V>): Builder<T> {
|
||||
types.add(PackedProperty(field, String2ObjectAdapter(valueType.java).nullSafe(), isFlat = false))
|
||||
fun <V> auto(field: KProperty1<T, V>, isFlat: Boolean = false, transform: (TypeAdapter<V>) -> TypeAdapter<V> = { it }): Builder<T> {
|
||||
types.add(ResolvableProperty(field, isFlat = isFlat, transform = transform))
|
||||
return this
|
||||
}
|
||||
|
||||
@ -601,18 +486,6 @@ class FactoryAdapter<T : Any> private constructor(
|
||||
asList = true
|
||||
return this
|
||||
}
|
||||
|
||||
fun build(): FactoryAdapter<T> {
|
||||
check(!asList || types.none { it.isFlat }) { "Can't have both flat properties and json data array layout" }
|
||||
|
||||
return FactoryAdapter(
|
||||
bound = clazz,
|
||||
types = ImmutableList.copyOf(types),
|
||||
asJsonArray = asList,
|
||||
storesJson = storesJson,
|
||||
logMisses = logMisses,
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
companion object {
|
||||
|
@ -0,0 +1,117 @@
|
||||
package ru.dbotthepony.kstarbound.io.json.builder
|
||||
|
||||
import com.google.gson.Gson
|
||||
import com.google.gson.TypeAdapter
|
||||
import com.google.gson.reflect.TypeToken
|
||||
import kotlin.reflect.KMutableProperty1
|
||||
import kotlin.reflect.KProperty1
|
||||
import kotlin.reflect.KType
|
||||
import kotlin.reflect.javaType
|
||||
|
||||
interface IProperty {
|
||||
val isFlat: Boolean
|
||||
val name: String
|
||||
val mustBePresent: Boolean?
|
||||
}
|
||||
|
||||
interface IReferencedProperty<T : Any, V> : IProperty {
|
||||
val property: KProperty1<T, V>
|
||||
override val name: String get() = property.name
|
||||
override val mustBePresent: Boolean?
|
||||
get() = null
|
||||
}
|
||||
|
||||
interface IResolvableProperty<T : Any, V> : IReferencedProperty<T, V> {
|
||||
fun resolve(gson: Gson?): IResolvedProperty<T, V>
|
||||
}
|
||||
|
||||
interface IResolvedProperty<T : Any, V> : IReferencedProperty<T, V> {
|
||||
val type: KType
|
||||
val adapter: TypeAdapter<V>
|
||||
|
||||
operator fun component1() = property
|
||||
operator fun component2() = adapter
|
||||
}
|
||||
|
||||
class ResolvedProperty<T : Any, V>(
|
||||
override val property: KProperty1<T, V>,
|
||||
override val adapter: TypeAdapter<V>,
|
||||
override val isFlat: Boolean
|
||||
) : IResolvableProperty<T, V>, IResolvedProperty<T, V> {
|
||||
override val type: KType = property.returnType
|
||||
|
||||
override fun resolve(gson: Gson?): IResolvedProperty<T, V> {
|
||||
return this
|
||||
}
|
||||
}
|
||||
|
||||
class ResolvableProperty<T : Any, V>(
|
||||
override val property: KProperty1<T, V>,
|
||||
override val isFlat: Boolean,
|
||||
val transform: (TypeAdapter<V>) -> TypeAdapter<V> = { it }
|
||||
) : IResolvableProperty<T, V> {
|
||||
@OptIn(ExperimentalStdlibApi::class)
|
||||
override fun resolve(gson: Gson?): IResolvedProperty<T, V> {
|
||||
gson ?: throw NullPointerException("Can not resolve without Gson present")
|
||||
|
||||
return ResolvedProperty(
|
||||
property = property,
|
||||
adapter = transform(gson.getAdapter(TypeToken.get(property.returnType.javaType)) as TypeAdapter<V>),
|
||||
isFlat = isFlat
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
interface IReferencedMutableProperty<T : Any, V> : IProperty {
|
||||
val property: KMutableProperty1<T, V>
|
||||
override val name: String get() = property.name
|
||||
}
|
||||
|
||||
interface IResolvableMutableProperty<T : Any, V> : IReferencedMutableProperty<T, V> {
|
||||
fun resolve(gson: Gson?): IResolvedMutableProperty<T, V>
|
||||
}
|
||||
|
||||
interface IResolvedMutableProperty<T : Any, V> : IResolvableMutableProperty<T, V> {
|
||||
val type: KType
|
||||
val adapter: TypeAdapter<V>
|
||||
|
||||
operator fun component1() = property
|
||||
operator fun component2() = adapter
|
||||
|
||||
@Suppress("unchecked_cast")
|
||||
fun set(receiver: T, value: Any?) {
|
||||
property.set(receiver, value as V)
|
||||
}
|
||||
}
|
||||
|
||||
class ResolvedMutableProperty<T : Any, V>(
|
||||
override val property: KMutableProperty1<T, V>,
|
||||
override val adapter: TypeAdapter<V>,
|
||||
override val isFlat: Boolean,
|
||||
override val mustBePresent: Boolean?
|
||||
) : IResolvableMutableProperty<T, V>, IResolvedMutableProperty<T, V> {
|
||||
override val type: KType = property.returnType
|
||||
|
||||
override fun resolve(gson: Gson?): IResolvedMutableProperty<T, V> {
|
||||
return this
|
||||
}
|
||||
}
|
||||
|
||||
class ResolvableMutableProperty<T : Any, V>(
|
||||
override val property: KMutableProperty1<T, V>,
|
||||
override val isFlat: Boolean,
|
||||
val transform: (TypeAdapter<V>) -> TypeAdapter<V> = { it },
|
||||
override val mustBePresent: Boolean?
|
||||
) : IResolvableMutableProperty<T, V> {
|
||||
@OptIn(ExperimentalStdlibApi::class)
|
||||
override fun resolve(gson: Gson?): IResolvedMutableProperty<T, V> {
|
||||
gson ?: throw NullPointerException("Can not resolve without Gson present")
|
||||
|
||||
return ResolvedMutableProperty(
|
||||
property = property,
|
||||
adapter = transform(gson.getAdapter(TypeToken.get(property.returnType.javaType)) as TypeAdapter<V>),
|
||||
isFlat = isFlat,
|
||||
mustBePresent = mustBePresent,
|
||||
)
|
||||
}
|
||||
}
|
@ -15,25 +15,25 @@ object ImmutableCollectionAdapterFactory : TypeAdapterFactory {
|
||||
when (type.rawType) {
|
||||
ImmutableList::class.java -> {
|
||||
val elementType = `$Gson$Types`.getCollectionElementType(type.type, type.rawType)
|
||||
return ImmutableListTypeAdapter(gson.getAdapter(TypeToken.get(elementType))) as TypeAdapter<T>
|
||||
return ImmutableListTypeAdapter(gson.getAdapter(TypeToken.get(elementType))).nullSafe() as TypeAdapter<T>
|
||||
}
|
||||
|
||||
ImmutableSet::class.java -> {
|
||||
val elementType = `$Gson$Types`.getCollectionElementType(type.type, type.rawType)
|
||||
return ImmutableSetTypeAdapter(gson.getAdapter(TypeToken.get(elementType))) as TypeAdapter<T>
|
||||
return ImmutableSetTypeAdapter(gson.getAdapter(TypeToken.get(elementType))).nullSafe() as TypeAdapter<T>
|
||||
}
|
||||
|
||||
ImmutableMap::class.java -> {
|
||||
val (elementType0, elementType1) = `$Gson$Types`.getMapKeyAndValueTypes(type.type, type.rawType)
|
||||
|
||||
if (`$Gson$Types`.getRawType(elementType0) == String::class.java) {
|
||||
return ImmutableMapTypeAdapter(gson.getAdapter(TypeToken.get(elementType1))) as TypeAdapter<T>
|
||||
return ImmutableMapTypeAdapter(gson.getAdapter(TypeToken.get(elementType1))).nullSafe() as TypeAdapter<T>
|
||||
}
|
||||
|
||||
return ImmutableArrayMapTypeAdapter(
|
||||
gson.getAdapter(TypeToken.get(elementType0)),
|
||||
gson.getAdapter(TypeToken.get(elementType1))
|
||||
) as TypeAdapter<T>
|
||||
).nullSafe() as TypeAdapter<T>
|
||||
}
|
||||
|
||||
else -> return null
|
||||
|
Loading…
Reference in New Issue
Block a user