diff --git a/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/ninja/Ninja.kt b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/ninja/Ninja.kt index c5bf7027..bc82c22c 100644 --- a/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/ninja/Ninja.kt +++ b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/ninja/Ninja.kt @@ -10,46 +10,26 @@ import world.phantasmal.lib.fileFormats.vec3F32 private const val NJCM: Int = 0x4D434A4E -class NjObject( - val evaluationFlags: NjEvaluationFlags, - val model: Model?, - val position: Vec3, - /** - * Euler angles in radians. - */ - val rotation: Vec3, - val scale: Vec3, - val children: List>, -) - -class NjEvaluationFlags( - val noTranslate: Boolean, - val noRotate: Boolean, - val noScale: Boolean, - val hidden: Boolean, - val breakChildTrace: Boolean, - val zxyRotationOrder: Boolean, - val skip: Boolean, - val shapeSkip: Boolean, -) - -fun parseNj(cursor: Cursor): PwResult>> = +fun parseNj(cursor: Cursor): PwResult>> = parseNinja(cursor, ::parseNjcmModel, mutableMapOf()) -private fun parseNinja( +fun parseXj(cursor: Cursor): PwResult>> = + parseNinja(cursor, { _, _ -> XjModel() }, Unit) + +private fun parseNinja( cursor: Cursor, - parse_model: (cursor: Cursor, context: Context) -> Model, + parseModel: (cursor: Cursor, context: Context) -> Model, context: Context, -): PwResult>> = +): PwResult>> = when (val parseIffResult = parseIff(cursor)) { is Failure -> parseIffResult is Success -> { // POF0 and other chunks types are ignored. val njcmChunks = parseIffResult.value.filter { chunk -> chunk.type == NJCM } - val objects: MutableList> = mutableListOf() + val objects: MutableList> = mutableListOf() for (chunk in njcmChunks) { - objects.addAll(parseSiblingObjects(chunk.data, parse_model, context)) + objects.addAll(parseSiblingObjects(chunk.data, parseModel, context)) } Success(objects, parseIffResult.problems) @@ -57,11 +37,11 @@ private fun parseNinja( } // TODO: cache model and object offsets so we don't reparse the same data. -private fun parseSiblingObjects( +private fun parseSiblingObjects( cursor: Cursor, - parse_model: (cursor: Cursor, context: Context) -> Model, + parseModel: (cursor: Cursor, context: Context) -> Model, context: Context, -): List> { +): MutableList> { val evalFlags = cursor.uInt() val noTranslate = (evalFlags and 0b1u) != 0u val noRotate = (evalFlags and 0b10u) != 0u @@ -87,25 +67,25 @@ private fun parseSiblingObjects( null } else { cursor.seekStart(modelOffset) - parse_model(cursor, context) + parseModel(cursor, context) } val children = if (childOffset == 0) { - emptyList() + mutableListOf() } else { cursor.seekStart(childOffset) - parseSiblingObjects(cursor, parse_model, context) + parseSiblingObjects(cursor, parseModel, context) } val siblings = if (siblingOffset == 0) { - emptyList() + mutableListOf() } else { cursor.seekStart(siblingOffset) - parseSiblingObjects(cursor, parse_model, context) + parseSiblingObjects(cursor, parseModel, context) } - val obj = NjObject( - NjEvaluationFlags( + val obj = NinjaObject( + NinjaEvaluationFlags( noTranslate, noRotate, noScale, @@ -122,5 +102,6 @@ private fun parseSiblingObjects( children, ) - return listOf(obj) + siblings + siblings.add(0, obj) + return siblings } diff --git a/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/ninja/NinjaObject.kt b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/ninja/NinjaObject.kt new file mode 100644 index 00000000..1524530f --- /dev/null +++ b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/ninja/NinjaObject.kt @@ -0,0 +1,152 @@ +package world.phantasmal.lib.fileFormats.ninja + +import world.phantasmal.lib.fileFormats.Vec2 +import world.phantasmal.lib.fileFormats.Vec3 + +class NinjaObject( + val evaluationFlags: NinjaEvaluationFlags, + val model: Model?, + val position: Vec3, + /** + * Euler angles in radians. + */ + val rotation: Vec3, + val scale: Vec3, + children: MutableList>, +) { + private val _children = children + val children: List> = _children + + fun addChild(child: NinjaObject) { + _children.add(child) + } +} + +class NinjaEvaluationFlags( + var noTranslate: Boolean, + var noRotate: Boolean, + var noScale: Boolean, + var hidden: Boolean, + var breakChildTrace: Boolean, + var zxyRotationOrder: Boolean, + var skip: Boolean, + var shapeSkip: Boolean, +) + +sealed class NinjaModel + +/** + * The model type used in .nj files. + */ +class NjcmModel( + /** + * Sparse list of vertices. + */ + val vertices: List, + val meshes: List, + val collisionSphereCenter: Vec3, + val collisionSphereRadius: Float, +) : NinjaModel() + +class NjcmVertex( + val position: Vec3, + val normal: Vec3?, + val boneWeight: Float, + val boneWeightStatus: Int, + val calcContinue: Boolean, +) + +class NjcmTriangleStrip( + val ignoreLight: Boolean, + val ignoreSpecular: Boolean, + val ignoreAmbient: Boolean, + val useAlpha: Boolean, + val doubleSide: Boolean, + val flatShading: Boolean, + val environmentMapping: Boolean, + val clockwiseWinding: Boolean, + val hasTexCoords: Boolean, + val hasNormal: Boolean, + var textureId: UInt?, + var srcAlpha: UByte?, + var dstAlpha: UByte?, + val vertices: List, +) + +class NjcmMeshVertex( + val index: UShort, + val normal: Vec3?, + val texCoords: Vec2?, +) + +sealed class NjcmChunk(val typeId: UByte) { + class Unknown(typeId: UByte) : NjcmChunk(typeId) + + object Null : NjcmChunk(0u) + + class Bits(typeId: UByte, val srcAlpha: UByte, val dstAlpha: UByte) : NjcmChunk(typeId) + + class CachePolygonList(val cacheIndex: UByte, val offset: Int) : NjcmChunk(4u) + + class DrawPolygonList(val cacheIndex: UByte) : NjcmChunk(5u) + + class Tiny( + typeId: UByte, + val flipU: Boolean, + val flipV: Boolean, + val clampU: Boolean, + val clampV: Boolean, + val mipmapDAdjust: UInt, + val filterMode: UInt, + val superSample: Boolean, + val textureId: UInt, + ) : NjcmChunk(typeId) + + class Material( + typeId: UByte, + val srcAlpha: UByte, + val dstAlpha: UByte, + val diffuse: NjcmArgb?, + val ambient: NjcmArgb?, + val specular: NjcmErgb?, + ) : NjcmChunk(typeId) + + class Vertex(typeId: UByte, val vertices: List) : NjcmChunk(typeId) + + class Volume(typeId: UByte) : NjcmChunk(typeId) + + class Strip(typeId: UByte, val triangleStrips: List) : NjcmChunk(typeId) + + object End : NjcmChunk(255u) +} + +class NjcmChunkVertex( + val index: Int, + val position: Vec3, + val normal: Vec3?, + val boneWeight: Float, + val boneWeightStatus: Int, + val calcContinue: Boolean, +) + +/** + * Channels are in range [0, 1]. + */ +class NjcmArgb( + val a: Float, + val r: Float, + val g: Float, + val b: Float, +) + +class NjcmErgb( + val e: UByte, + val r: UByte, + val g: UByte, + val b: UByte, +) + +/** + * The model type used in .xj files. + */ +class XjModel : NinjaModel() diff --git a/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/ninja/Nj.kt b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/ninja/Njcm.kt similarity index 81% rename from lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/ninja/Nj.kt rename to lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/ninja/Njcm.kt index 106bc98f..dd5e9ebd 100644 --- a/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/ninja/Nj.kt +++ b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/ninja/Njcm.kt @@ -14,120 +14,12 @@ import kotlin.math.abs private val logger = KotlinLogging.logger {} -class NjcmModel( - /** - * Sparse list of vertices. - */ - val vertices: List, - val meshes: List, - val collisionSphereCenter: Vec3, - val collisionSphereRadius: Float, -) - -class NjcmVertex( - val position: Vec3, - val normal: Vec3?, - val boneWeight: Float, - val boneWeightStatus: UByte, - val calcContinue: Boolean, -) - -class NjcmTriangleStrip( - val ignoreLight: Boolean, - val ignoreSpecular: Boolean, - val ignoreAmbient: Boolean, - val useAlpha: Boolean, - val doubleSide: Boolean, - val flatShading: Boolean, - val environmentMapping: Boolean, - val clockwiseWinding: Boolean, - val hasTexCoords: Boolean, - val hasNormal: Boolean, - var textureId: UInt?, - var srcAlpha: UByte?, - var dstAlpha: UByte?, - val vertices: List, -) - -class NjcmMeshVertex( - val index: UShort, - val normal: Vec3?, - val texCoords: Vec2?, -) - -sealed class NjcmChunk(val typeId: UByte) { - class Unknown(typeId: UByte) : NjcmChunk(typeId) - - object Null : NjcmChunk(0u) - - class Bits(typeId: UByte, val srcAlpha: UByte, val dstAlpha: UByte) : NjcmChunk(typeId) - - class CachePolygonList(val cacheIndex: UByte, val offset: Int) : NjcmChunk(4u) - - class DrawPolygonList(val cacheIndex: UByte) : NjcmChunk(5u) - - class Tiny( - typeId: UByte, - val flipU: Boolean, - val flipV: Boolean, - val clampU: Boolean, - val clampV: Boolean, - val mipmapDAdjust: UInt, - val filterMode: UInt, - val superSample: Boolean, - val textureId: UInt, - ) : NjcmChunk(typeId) - - class Material( - typeId: UByte, - val srcAlpha: UByte, - val dstAlpha: UByte, - val diffuse: NjcmArgb?, - val ambient: NjcmArgb?, - val specular: NjcmErgb?, - ) : NjcmChunk(typeId) - - class Vertex(typeId: UByte, val vertices: List) : NjcmChunk(typeId) - - class Volume(typeId: UByte) : NjcmChunk(typeId) - - class Strip(typeId: UByte, val triangleStrips: List) : NjcmChunk(typeId) - - object End : NjcmChunk(255u) -} - -class NjcmChunkVertex( - val index: Int, - val position: Vec3, - val normal: Vec3?, - val boneWeight: Float, - val boneWeightStatus: UByte, - val calcContinue: Boolean, -) - -/** - * Channels are in range [0, 1]. - */ -class NjcmArgb( - val a: Float, - val r: Float, - val g: Float, - val b: Float, -) - -class NjcmErgb( - val e: UByte, - val r: UByte, - val g: UByte, - val b: UByte, -) - fun parseNjcmModel(cursor: Cursor, cachedChunkOffsets: MutableMap): NjcmModel { val vlistOffset = cursor.int() // Vertex list val plistOffset = cursor.int() // Triangle strip index list val boundingSphereCenter = cursor.vec3F32() val boundingSphereRadius = cursor.float() - val vertices: MutableList = mutableListOf() + val vertices: MutableList = mutableListOf() val meshes: MutableList = mutableListOf() if (vlistOffset != 0) { @@ -136,6 +28,10 @@ fun parseNjcmModel(cursor: Cursor, cachedChunkOffsets: MutableMap): for (chunk in parseChunks(cursor, cachedChunkOffsets, true)) { if (chunk is NjcmChunk.Vertex) { for (vertex in chunk.vertices) { + while (vertices.size <= vertex.index) { + vertices.add(null) + } + vertices[vertex.index] = NjcmVertex( vertex.position, vertex.normal, @@ -156,23 +52,19 @@ fun parseNjcmModel(cursor: Cursor, cachedChunkOffsets: MutableMap): var dstAlpha: UByte? = null for (chunk in parseChunks(cursor, cachedChunkOffsets, false)) { - @Suppress("UNUSED_VALUE") // Ignore useless warning due to compiler bug. when (chunk) { is NjcmChunk.Bits -> { srcAlpha = chunk.srcAlpha dstAlpha = chunk.dstAlpha - break } is NjcmChunk.Tiny -> { textureId = chunk.textureId - break } is NjcmChunk.Material -> { srcAlpha = chunk.srcAlpha dstAlpha = chunk.dstAlpha - break } is NjcmChunk.Strip -> { @@ -183,7 +75,6 @@ fun parseNjcmModel(cursor: Cursor, cachedChunkOffsets: MutableMap): } meshes.addAll(chunk.triangleStrips) - break } else -> { @@ -357,7 +248,7 @@ private fun parseVertexChunk( chunkTypeId: UByte, flags: UByte, ): List { - val boneWeightStatus = flags and 0b11u + val boneWeightStatus = (flags and 0b11u).toInt() val calcContinue = (flags and 0x80u) != ZERO_U8 val index = cursor.uShort() @@ -451,9 +342,9 @@ private fun parseTriangleStripChunk( val flatShading = (flags and 0b100000u) != ZERO_U8 val environmentMapping = (flags and 0b1000000u) != ZERO_U8 - val userOffsetAndStripCount = cursor.uShort() - val userFlagsSize = (userOffsetAndStripCount.toUInt() shr 14).toInt() - val stripCount = userOffsetAndStripCount and 0x3fffu + val userOffsetAndStripCount = cursor.short().toInt() + val userFlagsSize = (userOffsetAndStripCount ushr 14) + val stripCount = userOffsetAndStripCount and 0x3FFF var hasTexCoords = false var hasColor = false @@ -490,14 +381,14 @@ private fun parseTriangleStripChunk( val strips: MutableList = mutableListOf() - repeat(stripCount.toInt()) { + repeat(stripCount) { val windingFlagAndIndexCount = cursor.short() val clockwiseWinding = windingFlagAndIndexCount < 1 val indexCount = abs(windingFlagAndIndexCount.toInt()) val vertices: MutableList = mutableListOf() - for (j in 0..indexCount) { + for (j in 0 until indexCount) { val index = cursor.uShort() val texCoords = if (hasTexCoords) { diff --git a/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/NpcTypeFromData.kt b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/NpcTypeFromData.kt new file mode 100644 index 00000000..8616d814 --- /dev/null +++ b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/NpcTypeFromData.kt @@ -0,0 +1,163 @@ +package world.phantasmal.lib.fileFormats.quest + +// TODO: detect Mothmant, St. Rappy, Hallo Rappy, Egg Rappy, Death Gunner, Bulk and Recon. +fun npcTypeFromQuestNpc(npc: QuestNpc): NpcType { + val episode = npc.episode + val special = npc.special + val skin = npc.skin + val areaId = npc.areaId + + return when (npc.typeId.toInt()) { + 0x004 -> NpcType.FemaleFat + 0x005 -> NpcType.FemaleMacho + 0x007 -> NpcType.FemaleTall + 0x00A -> NpcType.MaleDwarf + 0x00B -> NpcType.MaleFat + 0x00C -> NpcType.MaleMacho + 0x00D -> NpcType.MaleOld + 0x019 -> NpcType.BlueSoldier + 0x01A -> NpcType.RedSoldier + 0x01B -> NpcType.Principal + 0x01C -> NpcType.Tekker + 0x01D -> NpcType.GuildLady + 0x01E -> NpcType.Scientist + 0x01F -> NpcType.Nurse + 0x020 -> NpcType.Irene + 0x040 -> when (skin % 2) { + 0 -> if (episode == Episode.II) NpcType.Hildebear2 else NpcType.Hildebear + else -> if (episode == Episode.II) NpcType.Hildeblue2 else NpcType.Hildeblue + } + 0x041 -> when (skin % 2) { + 0 -> when (episode) { + Episode.I -> NpcType.RagRappy + Episode.II -> NpcType.RagRappy2 + Episode.IV -> NpcType.SandRappy + } + else -> when (episode) { + Episode.I -> NpcType.AlRappy + Episode.II -> NpcType.LoveRappy + Episode.IV -> NpcType.DelRappy + } + } + 0x042 -> if (episode == Episode.II) NpcType.Monest2 else NpcType.Monest + 0x043 -> when (special) { + true -> if (episode == Episode.II) NpcType.BarbarousWolf2 else NpcType.BarbarousWolf + false -> if (episode == Episode.II) NpcType.SavageWolf2 else NpcType.SavageWolf + } + 0x044 -> when (skin % 3) { + 0 -> NpcType.Booma + 1 -> NpcType.Gobooma + else -> NpcType.Gigobooma + } + 0x060 -> if (episode == Episode.II) NpcType.GrassAssassin2 else NpcType.GrassAssassin + 0x061 -> when { + areaId > 15 -> NpcType.DelLily + special -> if (episode == Episode.II) NpcType.NarLily2 else NpcType.NarLily + else -> if (episode == Episode.II) NpcType.PoisonLily2 else NpcType.PoisonLily + } + 0x062 -> NpcType.NanoDragon + 0x063 -> when (skin % 3) { + 0 -> NpcType.EvilShark + 1 -> NpcType.PalShark + else -> NpcType.GuilShark + } + 0x064 -> if (special) NpcType.PouillySlime else NpcType.PofuillySlime + 0x065 -> if (episode == Episode.II) NpcType.PanArms2 else NpcType.PanArms + 0x080 -> when (skin % 2) { + 0 -> if (episode == Episode.II) NpcType.Dubchic2 else NpcType.Dubchic + else -> if (episode == Episode.II) NpcType.Gilchic2 else NpcType.Gilchic + } + 0x081 -> if (episode == Episode.II) NpcType.Garanz2 else NpcType.Garanz + 0x082 -> if (special) NpcType.SinowGold else NpcType.SinowBeat + 0x083 -> NpcType.Canadine + 0x084 -> NpcType.Canane + 0x085 -> if (episode == Episode.II) NpcType.Dubswitch2 else NpcType.Dubswitch + 0x0A0 -> if (episode == Episode.II) NpcType.Delsaber2 else NpcType.Delsaber + 0x0A1 -> if (episode == Episode.II) NpcType.ChaosSorcerer2 else NpcType.ChaosSorcerer + 0x0A2 -> NpcType.DarkGunner + 0x0A4 -> NpcType.ChaosBringer + 0x0A5 -> if (episode == Episode.II) NpcType.DarkBelra2 else NpcType.DarkBelra + 0x0A6 -> when (skin % 3) { + 0 -> if (episode == Episode.II) NpcType.Dimenian2 else NpcType.Dimenian + 1 -> if (episode == Episode.II) NpcType.LaDimenian2 else NpcType.LaDimenian + else -> if (episode == Episode.II) NpcType.SoDimenian2 else NpcType.SoDimenian + } + 0x0A7 -> NpcType.Bulclaw + 0x0A8 -> NpcType.Claw + 0x0C0 -> if (episode == Episode.II) NpcType.GalGryphon else NpcType.Dragon + 0x0C1 -> NpcType.DeRolLe + 0x0C2 -> NpcType.VolOptPart1 + 0x0C5 -> NpcType.VolOptPart2 + 0x0C8 -> NpcType.DarkFalz + 0x0CA -> NpcType.OlgaFlow + 0x0CB -> NpcType.BarbaRay + 0x0CC -> NpcType.GolDragon + 0x0D4 -> when (skin % 2) { + 0 -> NpcType.SinowBerill + else -> NpcType.SinowSpigell + } + 0x0D5 -> when (skin % 2) { + 0 -> NpcType.Merillia + else -> NpcType.Meriltas + } + 0x0D6 -> when (skin % 3) { + 0 -> NpcType.Mericarol + 1 -> NpcType.Mericus + else -> NpcType.Merikle + } + 0x0D7 -> when (skin % 2) { + 0 -> NpcType.UlGibbon + else -> NpcType.ZolGibbon + } + 0x0D8 -> NpcType.Gibbles + 0x0D9 -> NpcType.Gee + 0x0DA -> NpcType.GiGue + 0x0DB -> NpcType.Deldepth + 0x0DC -> NpcType.Delbiter + 0x0DD -> when (skin % 2) { + 0 -> NpcType.Dolmolm + else -> NpcType.Dolmdarl + } + 0x0DE -> NpcType.Morfos + 0x0DF -> NpcType.Recobox + 0x0E0 -> when { + areaId > 15 -> NpcType.Epsilon + skin % 2 == 0 -> NpcType.SinowZoa + else -> NpcType.SinowZele + } + 0x0E1 -> NpcType.IllGill + 0x0F1 -> NpcType.ItemShop + 0x0FE -> NpcType.Nurse2 + 0x110 -> NpcType.Astark + 0x111 -> if (special) NpcType.Yowie else NpcType.SatelliteLizard + 0x112 -> when (skin % 2) { + 0 -> NpcType.MerissaA + else -> NpcType.MerissaAA + } + 0x113 -> NpcType.Girtablulu + 0x114 -> when (skin % 2) { + 0 -> NpcType.Zu + else -> NpcType.Pazuzu + } + 0x115 -> when (skin % 3) { + 0 -> NpcType.Boota + 1 -> NpcType.ZeBoota + else -> NpcType.BaBoota + } + 0x116 -> when (skin % 2) { + 0 -> NpcType.Dorphon + else -> NpcType.DorphonEclair + } + 0x117 -> when (skin % 3) { + 0 -> NpcType.Goran + 1 -> NpcType.PyroGoran + else -> NpcType.GoranDetonator + } + 0x119 -> when { + special -> NpcType.Kondrieu + skin % 2 == 0 -> NpcType.SaintMilion + else -> NpcType.Shambertin + } + else -> NpcType.Unknown + } +} diff --git a/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/ObjectType.kt b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/ObjectType.kt index 2770ae57..2df31d7a 100644 --- a/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/ObjectType.kt +++ b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/ObjectType.kt @@ -1,3 +1,2562 @@ package world.phantasmal.lib.fileFormats.quest -enum class ObjectType : EntityType +enum class ObjectType( + override val uniqueName: String, + /** + * The valid area IDs per episode in which this object can appear. + */ + val areaIds: Map>, + val typeId: Int?, + /** + * Default object-specific properties. + */ + val properties: List = emptyList(), +) : EntityType { + Unknown( + uniqueName = "Unknown", + areaIds = mapOf(), + typeId = null, + ), + + PlayerSet( + uniqueName = "Player Set", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 0), + ), + typeId = 0, + properties = listOf( + EntityProp(name = "Slot ID", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Return flag", offset = 52, type = EntityPropType.I32), + ), + ), + Particle( + uniqueName = "Particle", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 0), + ), + typeId = 1, + ), + Teleporter( + uniqueName = "Teleporter", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 6, 7, 11, 12, 13, 14), + Episode.II to listOf(0, 1, 2, 3, 4, 12, 13, 14, 15), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 0), + ), + typeId = 2, + properties = listOf( + EntityProp(name = "Area ID", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Color blue", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Color red", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Floor ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Display no.", offset = 56, type = EntityPropType.I32), + EntityProp(name = "No display no.", offset = 60, type = EntityPropType.I32), + ), + ), + Warp( + uniqueName = "Warp", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 6, 7, 11, 12, 13, 14, 16, 17), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 0), + ), + typeId = 3, + properties = listOf( + EntityProp(name = "Destination x", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Destination y", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Destination z", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Dst. rotation y", offset = 52, type = EntityPropType.Angle), + ), + ), + LightCollision( + uniqueName = "Light Collision", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 15, 16, 17), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 0), + ), + typeId = 4, + ), + Item( + uniqueName = "Item", + areaIds = mapOf(), + typeId = 5, + ), + EnvSound( + uniqueName = "Env Sound", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 16, 17), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 13), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 0), + ), + typeId = 6, + properties = listOf( + EntityProp(name = "Radius", offset = 48, type = EntityPropType.F32), + EntityProp(name = "SE", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Volume", offset = 56, type = EntityPropType.I32), + ), + ), + FogCollision( + uniqueName = "Fog Collision", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 16, 17), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 0), + ), + typeId = 7, + properties = listOf( + EntityProp(name = "Radius", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Fog index no.", offset = 52, type = EntityPropType.I32), + ), + ), + EventCollision( + uniqueName = "Event Collision", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 16, 17), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 0), + ), + typeId = 8, + properties = listOf( + EntityProp(name = "Radius", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Event ID", offset = 52, type = EntityPropType.U32), + ), + ), + CharaCollision( + uniqueName = "Chara Collision", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 8, 9, 10), + Episode.II to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 9, + ), + ElementalTrap( + uniqueName = "Elemental Trap", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 16, 17), + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9), + ), + typeId = 10, + properties = listOf( + EntityProp(name = "Radius", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Trap link", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Damage", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Subtype", offset = 56, type = EntityPropType.I32), + EntityProp(name = "Delay", offset = 60, type = EntityPropType.I32), + ), + ), + StatusTrap( + uniqueName = "Status Trap", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 16, 17), + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9), + ), + typeId = 11, + properties = listOf( + EntityProp(name = "Radius", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Trap link", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Subtype", offset = 56, type = EntityPropType.I32), + EntityProp(name = "Delay", offset = 60, type = EntityPropType.I32), + ), + ), + HealTrap( + uniqueName = "Heal Trap", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 16, 17), + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9), + ), + typeId = 12, + properties = listOf( + EntityProp(name = "Radius", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Trap link", offset = 48, type = EntityPropType.F32), + EntityProp(name = "HP", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Subtype", offset = 56, type = EntityPropType.I32), + EntityProp(name = "Delay", offset = 60, type = EntityPropType.I32), + ), + ), + LargeElementalTrap( + uniqueName = "Large Elemental Trap", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 16, 17), + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9), + ), + typeId = 13, + properties = listOf( + EntityProp(name = "Radus", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Trap link", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Damage", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Subtype", offset = 56, type = EntityPropType.I32), + EntityProp(name = "Delay", offset = 60, type = EntityPropType.I32), + ), + ), + ObjRoomID( + uniqueName = "Obj Room ID", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13), + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9), + ), + typeId = 14, + properties = listOf( + EntityProp(name = "SCL_TAMA", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Next section", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Previous section ", offset = 48, type = EntityPropType.F32), + ), + ), + Sensor( + uniqueName = "Sensor", + areaIds = mapOf( + Episode.I to listOf(1, 2, 4, 5, 6, 7), + ), + typeId = 15, + ), + UnknownItem16( + uniqueName = "Unknown Item (16)", + areaIds = mapOf(), + typeId = 16, + ), + LensFlare( + uniqueName = "Lens Flare", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 8, 14), + ), + typeId = 17, + ), + ScriptCollision( + uniqueName = "Script Collision", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 0), + ), + typeId = 18, + properties = listOf( + EntityProp(name = "Radius", offset = 40, type = EntityPropType.F32), + ), + ), + HealRing( + uniqueName = "Heal Ring", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10), + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 19, + ), + MapCollision( + uniqueName = "Map Collision", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 8, 9, 10, 16, 17), + Episode.II to listOf(0, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(0), + ), + typeId = 20, + ), + ScriptCollisionA( + uniqueName = "Script Collision A", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 0), + ), + typeId = 21, + ), + ItemLight( + uniqueName = "Item Light", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 0), + ), + typeId = 22, + properties = listOf( + EntityProp(name = "Subtype", offset = 40, type = EntityPropType.F32), + ), + ), + RadarCollision( + uniqueName = "Radar Collision", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15), + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 23, + properties = listOf( + EntityProp(name = "Radius", offset = 40, type = EntityPropType.F32), + ), + ), + FogCollisionSW( + uniqueName = "Fog Collision SW", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14), + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 24, + properties = listOf( + EntityProp(name = "Radius", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Status", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Fog index no.", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Switch ID", offset = 60, type = EntityPropType.I32), + ), + ), + BossTeleporter( + uniqueName = "Boss Teleporter", + areaIds = mapOf( + Episode.I to listOf(0, 2, 5, 7, 10), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(5, 6, 7, 8, 0), + ), + typeId = 25, + ), + ImageBoard( + uniqueName = "Image Board", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.II to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 26, + properties = listOf( + EntityProp(name = "Scale x", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Scale y", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Scale z", offset = 48, type = EntityPropType.F32), + ), + ), + QuestWarp( + uniqueName = "Quest Warp", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 11, 12, 13, 14), + Episode.IV to listOf(9), + ), + typeId = 27, + ), + Epilogue( + uniqueName = "Epilogue", + areaIds = mapOf( + Episode.I to listOf(14), + Episode.II to listOf(13), + Episode.IV to listOf(9), + ), + typeId = 28, + ), + UnknownItem29( + uniqueName = "Unknown Item (29)", + areaIds = mapOf( + Episode.I to listOf(1), + ), + typeId = 29, + ), + UnknownItem30( + uniqueName = "Unknown Item (30)", + areaIds = mapOf( + Episode.I to listOf(1, 2, 17), + Episode.II to listOf(1, 2, 14), + Episode.IV to listOf(1, 2, 3, 4, 5), + ), + typeId = 30, + ), + UnknownItem31( + uniqueName = "Unknown Item (31)", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 16, 17), + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 31, + ), + BoxDetectObject( + uniqueName = "Box Detect Object", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 16, 17), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 0), + ), + typeId = 32, + properties = listOf( + EntityProp(name = "Radius", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Plate ID", offset = 52, type = EntityPropType.I32), + ), + ), + SymbolChatObject( + uniqueName = "Symbol Chat Object", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 16, 17), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 0), + ), + typeId = 33, + properties = listOf( + EntityProp(name = "Radius", offset = 40, type = EntityPropType.F32), + ), + ), + TouchPlateObject( + uniqueName = "Touch plate Object", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 16, 17), + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 34, + properties = listOf( + EntityProp(name = "Radius", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + ), + ), + TargetableObject( + uniqueName = "Targetable Object", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 16, 17), + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 35, + properties = listOf( + EntityProp(name = "Switch ID", offset = 48, type = EntityPropType.F32), + EntityProp(name = "HP", offset = 52, type = EntityPropType.I32), + ), + ), + EffectObject( + uniqueName = "Effect object", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 16, 17), + Episode.II to listOf(0, 1, 2, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17), + Episode.IV to listOf(0), + ), + typeId = 36, + ), + CountDownObject( + uniqueName = "Count Down Object", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 16, 17), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 0), + ), + typeId = 37, + ), + UnknownItem38( + uniqueName = "Unknown Item (38)", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 16, 17), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 0), + ), + typeId = 38, + ), + UnknownItem39( + uniqueName = "Unknown Item (39)", + areaIds = mapOf( + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 39, + ), + UnknownItem40( + uniqueName = "Unknown Item (40)", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 4, 5, 6, 7, 8, 9, 10, 13, 16, 17), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 13, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 0), + ), + typeId = 40, + ), + UnknownItem41( + uniqueName = "Unknown Item (41)", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 4, 5, 6, 7, 8, 9, 10, 13, 16, 17), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 13, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 0), + ), + typeId = 41, + ), + MenuActivation( + uniqueName = "Menu activation", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.II to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 64, + properties = listOf( + EntityProp(name = "Menu ID", offset = 52, type = EntityPropType.I32), + ), + ), + TelepipeLocation( + uniqueName = "Telepipe Location", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.II to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 65, + properties = listOf( + EntityProp(name = "Slot ID", offset = 52, type = EntityPropType.I32), + ), + ), + BGMCollision( + uniqueName = "BGM Collision", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.II to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 66, + ), + MainRagolTeleporter( + uniqueName = "Main Ragol Teleporter", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.II to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 67, + ), + LobbyTeleporter( + uniqueName = "Lobby Teleporter", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.II to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 68, + ), + PrincipalWarp( + uniqueName = "Principal warp", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.II to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 69, + properties = listOf( + EntityProp(name = "Destination x", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Destination y", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Destination z", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Dst. rotation y", offset = 52, type = EntityPropType.Angle), + EntityProp(name = "Model", offset = 60, type = EntityPropType.U32), + ), + ), + ShopDoor( + uniqueName = "Shop Door", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 70, + ), + HuntersGuildDoor( + uniqueName = "Hunter's Guild Door", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 71, + ), + TeleporterDoor( + uniqueName = "Teleporter Door", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 72, + ), + MedicalCenterDoor( + uniqueName = "Medical Center Door", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 73, + ), + Elevator( + uniqueName = "Elevator", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 74, + ), + EasterEgg( + uniqueName = "Easter Egg", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.II to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 75, + ), + ValentinesHeart( + uniqueName = "Valentines Heart", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.II to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 76, + ), + ChristmasTree( + uniqueName = "Christmas Tree", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.II to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 77, + ), + ChristmasWreath( + uniqueName = "Christmas Wreath", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.II to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 78, + ), + HalloweenPumpkin( + uniqueName = "Halloween Pumpkin", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.II to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 79, + ), + TwentyFirstCentury( + uniqueName = "21st Century", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.II to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 80, + ), + Sonic( + uniqueName = "Sonic", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.II to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 81, + properties = listOf( + EntityProp(name = "Model", offset = 52, type = EntityPropType.U32), + ), + ), + WelcomeBoard( + uniqueName = "Welcome Board", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.II to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 82, + ), + Firework( + uniqueName = "Firework", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.II to listOf(0, 16), + Episode.IV to listOf(0), + ), + typeId = 83, + properties = listOf( + EntityProp(name = "Mdl IDX", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Area width", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Rise height", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Area depth", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Freq", offset = 56, type = EntityPropType.I32), + ), + ), + LobbyScreenDoor( + uniqueName = "Lobby Screen Door", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 84, + ), + MainRagolTeleporterBattleInNextArea( + uniqueName = "Main Ragol Teleporter (Battle in next area?)", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.II to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 85, + ), + LabTeleporterDoor( + uniqueName = "Lab Teleporter Door", + areaIds = mapOf( + Episode.II to listOf(0), + ), + typeId = 86, + ), + Pioneer2InvisibleTouchplate( + uniqueName = "Pioneer 2 Invisible Touchplate", + areaIds = mapOf( + Episode.I to listOf(0), + Episode.II to listOf(0), + Episode.IV to listOf(0), + ), + typeId = 87, + properties = listOf( + EntityProp(name = "Radius", offset = 40, type = EntityPropType.F32), + ), + ), + ForestDoor( + uniqueName = "Forest Door", + areaIds = mapOf( + Episode.I to listOf(1, 2), + ), + typeId = 128, + properties = listOf( + EntityProp(name = "Door ID", offset = 52, type = EntityPropType.I32), + ), + ), + ForestSwitch( + uniqueName = "Forest Switch", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5), + Episode.II to listOf(1, 2, 3, 4), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 129, + properties = listOf( + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Color", offset = 60, type = EntityPropType.I32), + ), + ), + LaserFence( + uniqueName = "Laser Fence", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 130, + properties = listOf( + EntityProp(name = "Color", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Model", offset = 60, type = EntityPropType.U32), + ), + ), + LaserSquareFence( + uniqueName = "Laser Square Fence", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 16, 17), + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 131, + properties = listOf( + EntityProp(name = "Color", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Model", offset = 60, type = EntityPropType.U32), + ), + ), + ForestLaserFenceSwitch( + uniqueName = "Forest Laser Fence Switch", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 16, 17), + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 132, + properties = listOf( + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Color", offset = 60, type = EntityPropType.I32), + ), + ), + LightRays( + uniqueName = "Light rays", + areaIds = mapOf( + Episode.I to listOf(1, 2), + Episode.II to listOf(5, 6, 7, 8, 9), + Episode.IV to listOf(6, 7, 8), + ), + typeId = 133, + properties = listOf( + EntityProp(name = "Scale x", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Scale y", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Scale z", offset = 48, type = EntityPropType.F32), + ), + ), + BlueButterfly( + uniqueName = "Blue Butterfly", + areaIds = mapOf( + Episode.I to listOf(1, 2), + Episode.IV to listOf(6, 7, 8), + ), + typeId = 134, + ), + Probe( + uniqueName = "Probe", + areaIds = mapOf( + Episode.I to listOf(1, 2), + ), + typeId = 135, + properties = listOf( + EntityProp(name = "Model", offset = 40, type = EntityPropType.F32), + ), + ), + RandomTypeBox1( + uniqueName = "Random Type Box 1", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7), + Episode.II to listOf(10, 11, 13), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 136, + ), + ForestWeatherStation( + uniqueName = "Forest Weather Station", + areaIds = mapOf( + Episode.I to listOf(1, 2), + ), + typeId = 137, + ), + Battery( + uniqueName = "Battery", + areaIds = mapOf(), + typeId = 138, + ), + ForestConsole( + uniqueName = "Forest Console", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 16, 17), + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 139, + properties = listOf( + EntityProp(name = "Script label", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Model", offset = 56, type = EntityPropType.U32), + ), + ), + BlackSlidingDoor( + uniqueName = "Black Sliding Door", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3), + ), + typeId = 140, + properties = listOf( + EntityProp(name = "Distance", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Speed", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Switch ID", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Switch no.", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Disable effect", offset = 56, type = EntityPropType.I32), + EntityProp(name = "Enable effect", offset = 60, type = EntityPropType.I32), + ), + ), + RicoMessagePod( + uniqueName = "Rico Message Pod", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 13), + ), + typeId = 141, + ), + EnergyBarrier( + uniqueName = "Energy Barrier", + areaIds = mapOf( + Episode.I to listOf(1, 2, 4, 5, 6, 7), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 142, + properties = listOf( + EntityProp(name = "Door ID", offset = 52, type = EntityPropType.I32), + ), + ), + ForestRisingBridge( + uniqueName = "Forest Rising Bridge", + areaIds = mapOf( + Episode.I to listOf(1, 2), + ), + typeId = 143, + properties = listOf( + EntityProp(name = "Door ID", offset = 52, type = EntityPropType.I32), + ), + ), + SwitchNoneDoor( + uniqueName = "Switch (none door)", + areaIds = mapOf( + Episode.I to listOf(1, 2, 6, 7, 16, 17), + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 144, + properties = listOf( + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + ), + ), + EnemyBoxGrey( + uniqueName = "Enemy Box (Grey)", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7), + Episode.II to listOf(10, 11), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 145, + properties = listOf( + EntityProp(name = "Event ID", offset = 40, type = EntityPropType.F32), + ), + ), + FixedTypeBox( + uniqueName = "Fixed Type Box", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 11, 12, 13, 14), + Episode.II to listOf(10, 11, 13), + Episode.IV to listOf(1, 2, 3, 4, 6, 7, 8, 9), + ), + typeId = 146, + properties = listOf( + EntityProp(name = "Full random", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Random item", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Fixed item", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Item parameter", offset = 52, type = EntityPropType.I32), + ), + ), + EnemyBoxBrown( + uniqueName = "Enemy Box (Brown)", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7), + Episode.II to listOf(10, 11), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 147, + properties = listOf( + EntityProp(name = "Event ID", offset = 40, type = EntityPropType.F32), + ), + ), + EmptyTypeBox( + uniqueName = "Empty Type Box", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 149, + properties = listOf( + EntityProp(name = "Event ID", offset = 40, type = EntityPropType.F32), + ), + ), + LaserFenceEx( + uniqueName = "Laser Fence Ex", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 16, 17), + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 150, + properties = listOf( + EntityProp(name = "Color", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Collision width", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Collision depth", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Model", offset = 60, type = EntityPropType.U32), + ), + ), + LaserSquareFenceEx( + uniqueName = "Laser Square Fence Ex", + areaIds = mapOf(), + typeId = 151, + properties = listOf( + EntityProp(name = "Color", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Collision width", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Collision depth", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Model", offset = 60, type = EntityPropType.U32), + ), + ), + FloorPanel1( + uniqueName = "Floor Panel 1", + areaIds = mapOf( + Episode.I to listOf(3, 4, 5, 16, 17), + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 192, + properties = listOf( + EntityProp(name = "Scale x", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Scale y", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Scale z", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Stay active", offset = 56, type = EntityPropType.I32), + ), + ), + Caves4ButtonDoor( + uniqueName = "Caves 4 Button door", + areaIds = mapOf( + Episode.I to listOf(3, 4, 5), + ), + typeId = 193, + properties = listOf( + EntityProp(name = "Door ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Switch total", offset = 56, type = EntityPropType.I32), + EntityProp(name = "Stay active", offset = 60, type = EntityPropType.I32), + ), + ), + CavesNormalDoor( + uniqueName = "Caves Normal door", + areaIds = mapOf( + Episode.I to listOf(3, 4, 5), + ), + typeId = 194, + properties = listOf( + EntityProp(name = "Door ID", offset = 52, type = EntityPropType.I32), + ), + ), + CavesSmashingPillar( + uniqueName = "Caves Smashing Pillar", + areaIds = mapOf( + Episode.I to listOf(3, 4, 5), + Episode.II to listOf(1, 2, 3, 4, 17), + ), + typeId = 195, + ), + CavesSign1( + uniqueName = "Caves Sign 1", + areaIds = mapOf( + Episode.I to listOf(4, 5), + ), + typeId = 196, + ), + CavesSign2( + uniqueName = "Caves Sign 2", + areaIds = mapOf( + Episode.I to listOf(4, 5), + ), + typeId = 197, + ), + CavesSign3( + uniqueName = "Caves Sign 3", + areaIds = mapOf( + Episode.I to listOf(4, 5), + ), + typeId = 198, + ), + HexagonalTank( + uniqueName = "Hexagonal Tank", + areaIds = mapOf( + Episode.I to listOf(4, 5), + ), + typeId = 199, + ), + BrownPlatform( + uniqueName = "Brown Platform", + areaIds = mapOf( + Episode.I to listOf(4, 5), + ), + typeId = 200, + ), + WarningLightObject( + uniqueName = "Warning Light Object", + areaIds = mapOf( + Episode.I to listOf(4, 5), + Episode.IV to listOf(5), + ), + typeId = 201, + ), + Rainbow( + uniqueName = "Rainbow", + areaIds = mapOf( + Episode.I to listOf(4), + ), + typeId = 203, + properties = listOf( + EntityProp(name = "Scale x", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Scale y", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Scale z", offset = 48, type = EntityPropType.F32), + ), + ), + FloatingJellyfish( + uniqueName = "Floating Jellyfish", + areaIds = mapOf( + Episode.I to listOf(4), + Episode.II to listOf(10, 11), + ), + typeId = 204, + ), + FloatingDragonfly( + uniqueName = "Floating Dragonfly", + areaIds = mapOf( + Episode.I to listOf(4, 16), + Episode.II to listOf(3, 4), + Episode.IV to listOf(6, 7, 8), + ), + typeId = 205, + ), + CavesSwitchDoor( + uniqueName = "Caves Switch Door", + areaIds = mapOf( + Episode.I to listOf(3, 4, 5), + ), + typeId = 206, + properties = listOf( + EntityProp(name = "Door ID", offset = 52, type = EntityPropType.I32), + ), + ), + RobotRechargeStation( + uniqueName = "Robot Recharge Station", + areaIds = mapOf( + Episode.I to listOf(3, 4, 5, 6, 7), + Episode.II to listOf(17), + ), + typeId = 207, + ), + CavesCakeShop( + uniqueName = "Caves Cake Shop", + areaIds = mapOf( + Episode.I to listOf(5), + ), + typeId = 208, + ), + Caves1SmallRedRock( + uniqueName = "Caves 1 Small Red Rock", + areaIds = mapOf( + Episode.I to listOf(3), + ), + typeId = 209, + ), + Caves1MediumRedRock( + uniqueName = "Caves 1 Medium Red Rock", + areaIds = mapOf( + Episode.I to listOf(3), + ), + typeId = 210, + ), + Caves1LargeRedRock( + uniqueName = "Caves 1 Large Red Rock", + areaIds = mapOf( + Episode.I to listOf(3), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 211, + ), + Caves2SmallRock1( + uniqueName = "Caves 2 Small Rock 1", + areaIds = mapOf( + Episode.I to listOf(4), + ), + typeId = 212, + ), + Caves2MediumRock1( + uniqueName = "Caves 2 Medium Rock 1", + areaIds = mapOf( + Episode.I to listOf(4), + ), + typeId = 213, + ), + Caves2LargeRock1( + uniqueName = "Caves 2 Large Rock 1", + areaIds = mapOf( + Episode.I to listOf(4), + ), + typeId = 214, + ), + Caves2SmallRock2( + uniqueName = "Caves 2 Small Rock 2", + areaIds = mapOf( + Episode.I to listOf(4), + ), + typeId = 215, + ), + Caves2MediumRock2( + uniqueName = "Caves 2 Medium Rock 2", + areaIds = mapOf( + Episode.I to listOf(4), + ), + typeId = 216, + ), + Caves2LargeRock2( + uniqueName = "Caves 2 Large Rock 2", + areaIds = mapOf( + Episode.I to listOf(4), + ), + typeId = 217, + ), + Caves3SmallRock( + uniqueName = "Caves 3 Small Rock", + areaIds = mapOf( + Episode.I to listOf(5), + ), + typeId = 218, + ), + Caves3MediumRock( + uniqueName = "Caves 3 Medium Rock", + areaIds = mapOf( + Episode.I to listOf(5), + ), + typeId = 219, + ), + Caves3LargeRock( + uniqueName = "Caves 3 Large Rock", + areaIds = mapOf( + Episode.I to listOf(5), + ), + typeId = 220, + ), + FloorPanel2( + uniqueName = "Floor Panel 2", + areaIds = mapOf( + Episode.I to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 16, 17), + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 222, + properties = listOf( + EntityProp(name = "Scale x", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Scale y", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Scale z", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Stay active", offset = 56, type = EntityPropType.I32), + ), + ), + DestructableRockCaves1( + uniqueName = "Destructable Rock (Caves 1)", + areaIds = mapOf( + Episode.I to listOf(3), + ), + typeId = 223, + ), + DestructableRockCaves2( + uniqueName = "Destructable Rock (Caves 2)", + areaIds = mapOf( + Episode.I to listOf(4), + ), + typeId = 224, + ), + DestructableRockCaves3( + uniqueName = "Destructable Rock (Caves 3)", + areaIds = mapOf( + Episode.I to listOf(5), + ), + typeId = 225, + ), + MinesDoor( + uniqueName = "Mines Door", + areaIds = mapOf( + Episode.I to listOf(6, 7), + ), + typeId = 256, + properties = listOf( + EntityProp(name = "Door ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Switch total", offset = 56, type = EntityPropType.I32), + EntityProp(name = "Stay active", offset = 60, type = EntityPropType.I32), + ), + ), + FloorPanel3( + uniqueName = "Floor Panel 3", + areaIds = mapOf( + Episode.I to listOf(1, 2, 6, 7, 16, 17), + Episode.II to listOf(1, 2, 3, 4), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 257, + properties = listOf( + EntityProp(name = "Scale x", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Scale y", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Scale z", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Stay active", offset = 56, type = EntityPropType.I32), + ), + ), + MinesSwitchDoor( + uniqueName = "Mines Switch Door", + areaIds = mapOf( + Episode.I to listOf(6, 7), + Episode.IV to listOf(6, 7, 8), + ), + typeId = 258, + properties = listOf( + EntityProp(name = "Door ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Switch total", offset = 56, type = EntityPropType.I32), + EntityProp(name = "Stay active", offset = 60, type = EntityPropType.I32), + ), + ), + LargeCryoTube( + uniqueName = "Large Cryo-Tube", + areaIds = mapOf( + Episode.I to listOf(6, 7), + Episode.II to listOf(17), + ), + typeId = 259, + ), + ComputerLikeCalus( + uniqueName = "Computer (like calus)", + areaIds = mapOf( + Episode.I to listOf(6, 7), + Episode.II to listOf(17), + ), + typeId = 260, + ), + GreenScreenOpeningAndClosing( + uniqueName = "Green Screen opening and closing", + areaIds = mapOf( + Episode.I to listOf(6, 7), + Episode.II to listOf(17), + ), + typeId = 261, + ), + FloatingRobot( + uniqueName = "Floating Robot", + areaIds = mapOf( + Episode.I to listOf(6, 7), + ), + typeId = 262, + ), + FloatingBlueLight( + uniqueName = "Floating Blue Light", + areaIds = mapOf( + Episode.I to listOf(6, 7), + ), + typeId = 263, + ), + SelfDestructingObject1( + uniqueName = "Self Destructing Object 1", + areaIds = mapOf( + Episode.I to listOf(6, 7), + ), + typeId = 264, + ), + SelfDestructingObject2( + uniqueName = "Self Destructing Object 2", + areaIds = mapOf( + Episode.I to listOf(6, 7), + ), + typeId = 265, + ), + SelfDestructingObject3( + uniqueName = "Self Destructing Object 3", + areaIds = mapOf( + Episode.I to listOf(6, 7), + ), + typeId = 266, + ), + SparkMachine( + uniqueName = "Spark Machine", + areaIds = mapOf( + Episode.I to listOf(6, 7), + ), + typeId = 267, + properties = listOf( + EntityProp(name = "Scale x", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Scale y", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Scale z", offset = 48, type = EntityPropType.F32), + ), + ), + MinesLargeFlashingCrate( + uniqueName = "Mines Large Flashing Crate", + areaIds = mapOf( + Episode.I to listOf(6, 7), + ), + typeId = 268, + ), + RuinsSeal( + uniqueName = "Ruins Seal", + areaIds = mapOf( + Episode.I to listOf(13), + ), + typeId = 304, + ), + RuinsTeleporter( + uniqueName = "Ruins Teleporter", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 320, + properties = listOf( + EntityProp(name = "Area no.", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Color blue", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Color red", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Floor no.", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Display no.", offset = 56, type = EntityPropType.I32), + EntityProp(name = "No display no.", offset = 60, type = EntityPropType.I32), + ), + ), + RuinsWarpSiteToSite( + uniqueName = "Ruins Warp (Site to Site)", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 321, + properties = listOf( + EntityProp(name = "Destination x", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Destination y", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Destination z", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Dst. rotation y", offset = 52, type = EntityPropType.Angle), + ), + ), + RuinsSwitch( + uniqueName = "Ruins Switch", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 322, + properties = listOf( + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + ), + ), + FloorPanel4( + uniqueName = "Floor Panel 4", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 323, + properties = listOf( + EntityProp(name = "Scale x", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Scale y", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Scale z", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Plate ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Stay active", offset = 56, type = EntityPropType.I32), + ), + ), + Ruins1Door( + uniqueName = "Ruins 1 Door", + areaIds = mapOf( + Episode.I to listOf(8), + ), + typeId = 324, + properties = listOf( + EntityProp(name = "Door ID", offset = 52, type = EntityPropType.I32), + ), + ), + Ruins3Door( + uniqueName = "Ruins 3 Door", + areaIds = mapOf( + Episode.I to listOf(10), + ), + typeId = 325, + properties = listOf( + EntityProp(name = "Door ID", offset = 52, type = EntityPropType.I32), + ), + ), + Ruins2Door( + uniqueName = "Ruins 2 Door", + areaIds = mapOf( + Episode.I to listOf(9), + ), + typeId = 326, + properties = listOf( + EntityProp(name = "Door ID", offset = 52, type = EntityPropType.I32), + ), + ), + Ruins11ButtonDoor( + uniqueName = "Ruins 1-1 Button Door", + areaIds = mapOf( + Episode.I to listOf(8), + ), + typeId = 327, + properties = listOf( + EntityProp(name = "Door ID", offset = 52, type = EntityPropType.I32), + ), + ), + Ruins21ButtonDoor( + uniqueName = "Ruins 2-1 Button Door", + areaIds = mapOf( + Episode.I to listOf(9), + ), + typeId = 328, + properties = listOf( + EntityProp(name = "Door ID", offset = 52, type = EntityPropType.I32), + ), + ), + Ruins31ButtonDoor( + uniqueName = "Ruins 3-1 Button Door", + areaIds = mapOf( + Episode.I to listOf(10), + ), + typeId = 329, + properties = listOf( + EntityProp(name = "Door ID", offset = 52, type = EntityPropType.I32), + ), + ), + Ruins4ButtonDoor( + uniqueName = "Ruins 4-Button Door", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 330, + properties = listOf( + EntityProp(name = "Door ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Stay active", offset = 60, type = EntityPropType.I32), + ), + ), + Ruins2ButtonDoor( + uniqueName = "Ruins 2-Button Door", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 331, + properties = listOf( + EntityProp(name = "Door ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Stay active", offset = 60, type = EntityPropType.I32), + ), + ), + RuinsSensor( + uniqueName = "Ruins Sensor", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 332, + ), + RuinsFenceSwitch( + uniqueName = "Ruins Fence Switch", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 333, + properties = listOf( + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Color", offset = 56, type = EntityPropType.I32), + ), + ), + RuinsLaserFence4x2( + uniqueName = "Ruins Laser Fence 4x2", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 334, + properties = listOf( + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Color", offset = 56, type = EntityPropType.I32), + ), + ), + RuinsLaserFence6x2( + uniqueName = "Ruins Laser Fence 6x2", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 335, + properties = listOf( + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Color", offset = 56, type = EntityPropType.I32), + ), + ), + RuinsLaserFence4x4( + uniqueName = "Ruins Laser Fence 4x4", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 336, + properties = listOf( + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Color", offset = 56, type = EntityPropType.I32), + ), + ), + RuinsLaserFence6x4( + uniqueName = "Ruins Laser Fence 6x4", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 337, + properties = listOf( + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Color", offset = 56, type = EntityPropType.I32), + ), + ), + RuinsPoisonBlob( + uniqueName = "Ruins poison Blob", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + Episode.II to listOf(5, 6, 7, 8, 9), + Episode.IV to listOf(6, 7, 8), + ), + typeId = 338, + ), + RuinsPillarTrap( + uniqueName = "Ruins Pillar Trap", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + Episode.II to listOf(1, 2, 3, 4), + ), + typeId = 339, + ), + PopupTrapNoTech( + uniqueName = "Popup Trap (No Tech)", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 340, + properties = listOf( + EntityProp(name = "Radius", offset = 40, type = EntityPropType.F32), + ), + ), + RuinsCrystal( + uniqueName = "Ruins Crystal", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 341, + ), + Monument( + uniqueName = "Monument", + areaIds = mapOf( + Episode.I to listOf(2, 4, 7), + ), + typeId = 342, + ), + RuinsRock1( + uniqueName = "Ruins Rock 1", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 345, + ), + RuinsRock2( + uniqueName = "Ruins Rock 2", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 346, + ), + RuinsRock3( + uniqueName = "Ruins Rock 3", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 347, + ), + RuinsRock4( + uniqueName = "Ruins Rock 4", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 348, + ), + RuinsRock5( + uniqueName = "Ruins Rock 5", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 349, + ), + RuinsRock6( + uniqueName = "Ruins Rock 6", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 350, + ), + RuinsRock7( + uniqueName = "Ruins Rock 7", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 351, + ), + Poison( + uniqueName = "Poison", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10, 13), + Episode.II to listOf(3, 4, 10, 11), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 352, + properties = listOf( + EntityProp(name = "Radius", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Power", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Link", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Switch mode", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Fog index no.", offset = 56, type = EntityPropType.I32), + EntityProp(name = "Switch ID", offset = 60, type = EntityPropType.I32), + ), + ), + FixedBoxTypeRuins( + uniqueName = "Fixed Box Type (Ruins)", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10, 16, 17), + Episode.II to listOf(1, 2, 3, 4, 14, 15), + ), + typeId = 353, + properties = listOf( + EntityProp(name = "Full random", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Random item", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Fixed item", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Item parameter", offset = 52, type = EntityPropType.I32), + ), + ), + RandomBoxTypeRuins( + uniqueName = "Random Box Type (Ruins)", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10, 16, 17), + Episode.II to listOf(1, 2, 3, 4, 14, 15), + ), + typeId = 354, + ), + EnemyTypeBoxYellow( + uniqueName = "Enemy Type Box (Yellow)", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10, 16, 17), + Episode.II to listOf(1, 2, 3, 4), + ), + typeId = 355, + ), + EnemyTypeBoxBlue( + uniqueName = "Enemy Type Box (Blue)", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10, 16, 17), + Episode.II to listOf(1, 2, 3, 4), + ), + typeId = 356, + ), + EmptyTypeBoxBlue( + uniqueName = "Empty Type Box (Blue)", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10, 16, 17), + Episode.II to listOf(1, 2, 3, 4), + ), + typeId = 357, + ), + DestructableRock( + uniqueName = "Destructable Rock", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + ), + typeId = 358, + ), + PopupTrapsTechs( + uniqueName = "Popup Traps (techs)", + areaIds = mapOf( + Episode.I to listOf(6, 7, 8, 9, 10), + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + ), + typeId = 359, + properties = listOf( + EntityProp(name = "Radius", offset = 40, type = EntityPropType.F32), + EntityProp(name = "HP", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Action", offset = 56, type = EntityPropType.I32), + EntityProp(name = "Tech", offset = 60, type = EntityPropType.I32), + ), + ), + FlyingWhiteBird( + uniqueName = "Flying White Bird", + areaIds = mapOf( + Episode.I to listOf(14, 16), + Episode.II to listOf(3, 4), + ), + typeId = 368, + ), + Tower( + uniqueName = "Tower", + areaIds = mapOf( + Episode.I to listOf(14), + ), + typeId = 369, + ), + FloatingRocks( + uniqueName = "Floating Rocks", + areaIds = mapOf( + Episode.I to listOf(14), + ), + typeId = 370, + ), + FloatingSoul( + uniqueName = "Floating Soul", + areaIds = mapOf( + Episode.I to listOf(14), + ), + typeId = 371, + ), + Butterfly( + uniqueName = "Butterfly", + areaIds = mapOf( + Episode.I to listOf(14), + ), + typeId = 372, + ), + LobbyGameMenu( + uniqueName = "Lobby Game menu", + areaIds = mapOf( + Episode.I to listOf(15), + ), + typeId = 384, + ), + LobbyWarpObject( + uniqueName = "Lobby Warp Object", + areaIds = mapOf( + Episode.I to listOf(15), + ), + typeId = 385, + ), + Lobby1EventObjectDefaultTree( + uniqueName = "Lobby 1 Event Object (Default Tree)", + areaIds = mapOf( + Episode.I to listOf(15), + ), + typeId = 386, + ), + UnknownItem387( + uniqueName = "Unknown Item (387)", + areaIds = mapOf( + Episode.I to listOf(15), + ), + typeId = 387, + ), + UnknownItem388( + uniqueName = "Unknown Item (388)", + areaIds = mapOf( + Episode.I to listOf(15), + ), + typeId = 388, + ), + UnknownItem389( + uniqueName = "Unknown Item (389)", + areaIds = mapOf( + Episode.I to listOf(15), + ), + typeId = 389, + ), + LobbyEventObjectStaticPumpkin( + uniqueName = "Lobby Event Object (Static Pumpkin)", + areaIds = mapOf( + Episode.I to listOf(15), + ), + typeId = 390, + ), + LobbyEventObject3ChristmasWindows( + uniqueName = "Lobby Event Object (3 Christmas Windows)", + areaIds = mapOf( + Episode.I to listOf(15), + ), + typeId = 391, + ), + LobbyEventObjectRedAndWhiteCurtain( + uniqueName = "Lobby Event Object (Red and White Curtain)", + areaIds = mapOf( + Episode.I to listOf(15), + ), + typeId = 392, + ), + UnknownItem393( + uniqueName = "Unknown Item (393)", + areaIds = mapOf( + Episode.I to listOf(15), + ), + typeId = 393, + ), + UnknownItem394( + uniqueName = "Unknown Item (394)", + areaIds = mapOf( + Episode.I to listOf(15), + ), + typeId = 394, + ), + LobbyFishTank( + uniqueName = "Lobby Fish Tank", + areaIds = mapOf( + Episode.I to listOf(15), + ), + typeId = 395, + ), + LobbyEventObjectButterflies( + uniqueName = "Lobby Event Object (Butterflies)", + areaIds = mapOf( + Episode.I to listOf(15), + ), + typeId = 396, + ), + UnknownItem400( + uniqueName = "Unknown Item (400)", + areaIds = mapOf( + Episode.I to listOf(16), + Episode.II to listOf(3, 4), + ), + typeId = 400, + ), + GreyWallLow( + uniqueName = "grey wall low", + areaIds = mapOf( + Episode.I to listOf(16), + Episode.II to listOf(3, 4, 17), + ), + typeId = 401, + ), + SpaceshipDoor( + uniqueName = "Spaceship Door", + areaIds = mapOf( + Episode.I to listOf(16), + Episode.II to listOf(3, 4), + ), + typeId = 402, + properties = listOf( + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + ), + ), + GreyWallHigh( + uniqueName = "grey wall high", + areaIds = mapOf( + Episode.I to listOf(16), + Episode.II to listOf(3, 4, 17), + ), + typeId = 403, + ), + TempleNormalDoor( + uniqueName = "Temple Normal Door", + areaIds = mapOf( + Episode.I to listOf(17), + Episode.II to listOf(1, 2), + ), + typeId = 416, + properties = listOf( + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + ), + ), + BreakableWallWallButUnbreakable( + uniqueName = "\"breakable wall wall, but unbreakable\"", + areaIds = mapOf( + Episode.I to listOf(17), + Episode.II to listOf(1, 2), + ), + typeId = 417, + ), + BrokenCylinderAndRubble( + uniqueName = "Broken cylinder and rubble", + areaIds = mapOf( + Episode.I to listOf(17), + Episode.II to listOf(1, 2), + ), + typeId = 418, + ), + ThreeBrokenWallPiecesOnFloor( + uniqueName = "3 broken wall pieces on floor", + areaIds = mapOf( + Episode.I to listOf(17), + Episode.II to listOf(1, 2), + ), + typeId = 419, + ), + HighBrickCylinder( + uniqueName = "high brick cylinder", + areaIds = mapOf( + Episode.I to listOf(17), + Episode.II to listOf(1, 2), + ), + typeId = 420, + ), + LyingCylinder( + uniqueName = "lying cylinder", + areaIds = mapOf( + Episode.I to listOf(17), + Episode.II to listOf(1, 2), + ), + typeId = 421, + ), + BrickConeWithFlatTop( + uniqueName = "brick cone with flat top", + areaIds = mapOf( + Episode.I to listOf(17), + Episode.II to listOf(1, 2), + ), + typeId = 422, + ), + BreakableTempleWall( + uniqueName = "breakable temple wall", + areaIds = mapOf( + Episode.I to listOf(17), + Episode.II to listOf(1, 2), + ), + typeId = 423, + ), + TempleMapDetect( + uniqueName = "Temple Map Detect", + areaIds = mapOf( + Episode.I to listOf(17), + Episode.II to listOf(1, 2, 14), + Episode.IV to listOf(1, 2, 3, 4, 5), + ), + typeId = 424, + ), + SmallBrownBrickRisingBridge( + uniqueName = "small brown brick rising bridge", + areaIds = mapOf( + Episode.I to listOf(17), + Episode.II to listOf(1, 2), + ), + typeId = 425, + ), + LongRisingBridgeWithPinkHighEdges( + uniqueName = "long rising bridge (with pink high edges)", + areaIds = mapOf( + Episode.I to listOf(17), + Episode.II to listOf(1, 2), + ), + typeId = 426, + ), + FourSwitchTempleDoor( + uniqueName = "4 Switch Temple Door", + areaIds = mapOf( + Episode.II to listOf(1, 2), + ), + typeId = 427, + ), + FourButtonSpaceshipDoor( + uniqueName = "4 button Spaceship Door", + areaIds = mapOf( + Episode.II to listOf(3, 4), + ), + typeId = 448, + ), + ItemBoxCca( + uniqueName = "Item Box CCA", + areaIds = mapOf( + Episode.II to listOf(5, 6, 7, 8, 9, 12, 16, 17), + Episode.IV to listOf(5), + ), + typeId = 512, + ), + TeleporterEp2( + uniqueName = "Teleporter (Ep. II)", + areaIds = mapOf( + Episode.II to listOf(5, 6, 7, 8, 9, 10, 11, 12, 13, 16, 17), + ), + typeId = 513, + ), + CcaDoor( + uniqueName = "CCA Door", + areaIds = mapOf( + Episode.II to listOf(5, 6, 7, 8, 9, 16, 17), + ), + typeId = 514, + properties = listOf( + EntityProp(name = "Scale x", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Scale y", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Scale z", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Switch amount", offset = 56, type = EntityPropType.I32), + EntityProp(name = "Stay active", offset = 60, type = EntityPropType.I32), + ), + ), + SpecialBoxCca( + uniqueName = "Special Box CCA", + areaIds = mapOf( + Episode.II to listOf(5, 6, 7, 8, 9, 12, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5), + ), + typeId = 515, + ), + BigCcaDoor( + uniqueName = "Big CCA Door", + areaIds = mapOf( + Episode.II to listOf(5), + ), + typeId = 516, + ), + BigCcaDoorSwitch( + uniqueName = "Big CCA Door Switch", + areaIds = mapOf( + Episode.II to listOf(5, 6, 7, 8, 9, 16, 17), + ), + typeId = 517, + ), + LittleRock( + uniqueName = "Little Rock", + areaIds = mapOf( + Episode.II to listOf(5, 6, 7, 8, 9, 16), + ), + typeId = 518, + ), + Little3StoneWall( + uniqueName = "Little 3 Stone Wall", + areaIds = mapOf( + Episode.II to listOf(5, 6, 7, 8, 9, 16), + ), + typeId = 519, + properties = listOf( + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + ), + ), + Medium3StoneWall( + uniqueName = "Medium 3 Stone Wall", + areaIds = mapOf( + Episode.II to listOf(5, 6, 7, 8, 9, 16), + ), + typeId = 520, + ), + SpiderPlant( + uniqueName = "Spider Plant", + areaIds = mapOf( + Episode.II to listOf(5, 6, 7, 8, 9, 16), + ), + typeId = 521, + ), + CcaAreaTeleporter( + uniqueName = "CCA Area Teleporter", + areaIds = mapOf( + Episode.II to listOf(5, 6, 7, 8, 9, 16, 17), + ), + typeId = 522, + ), + UnknownItem523( + uniqueName = "Unknown Item (523)", + areaIds = mapOf( + Episode.II to listOf(5, 12), + ), + typeId = 523, + ), + WhiteBird( + uniqueName = "White Bird", + areaIds = mapOf( + Episode.II to listOf(6, 7, 9, 16, 17), + Episode.IV to listOf(6, 7, 8), + ), + typeId = 524, + ), + OrangeBird( + uniqueName = "Orange Bird", + areaIds = mapOf( + Episode.II to listOf(6, 7, 9, 17), + ), + typeId = 525, + ), + Saw( + uniqueName = "Saw", + areaIds = mapOf( + Episode.II to listOf(5, 6, 7, 8, 9, 10, 11, 16, 17), + ), + typeId = 527, + properties = listOf( + EntityProp(name = "Speed", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Model", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Arc", offset = 56, type = EntityPropType.I32), + EntityProp(name = "Switch flag", offset = 60, type = EntityPropType.I32), + ), + ), + LaserDetect( + uniqueName = "Laser Detect", + areaIds = mapOf( + Episode.II to listOf(5, 6, 7, 8, 9, 10, 11, 16, 17), + ), + typeId = 528, + properties = listOf( + EntityProp(name = "Model", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Arc", offset = 56, type = EntityPropType.I32), + ), + ), + UnknownItem529( + uniqueName = "Unknown Item (529)", + areaIds = mapOf( + Episode.II to listOf(5, 6, 7), + Episode.IV to listOf(6, 7, 8), + ), + typeId = 529, + ), + UnknownItem530( + uniqueName = "Unknown Item (530)", + areaIds = mapOf( + Episode.II to listOf(5, 6, 7, 8, 9, 17), + ), + typeId = 530, + ), + Seagull( + uniqueName = "Seagull", + areaIds = mapOf( + Episode.II to listOf(6, 7, 8, 9, 16), + Episode.IV to listOf(6, 7, 8), + ), + typeId = 531, + ), + Fish( + uniqueName = "Fish", + areaIds = mapOf( + Episode.I to listOf(15), + Episode.II to listOf(6, 9, 10, 11, 16), + ), + typeId = 544, + ), + SeabedDoorWithBlueEdges( + uniqueName = "Seabed Door (with Blue Edges)", + areaIds = mapOf( + Episode.II to listOf(10, 11), + ), + typeId = 545, + properties = listOf( + EntityProp(name = "Scale x", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Scale y", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Scale z", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Switch ID", offset = 52, type = EntityPropType.I32), + EntityProp(name = "Switch amount", offset = 56, type = EntityPropType.I32), + EntityProp(name = "Stay active", offset = 60, type = EntityPropType.I32), + ), + ), + SeabedDoorAlwaysOpenNonTriggerable( + uniqueName = "Seabed Door (Always Open, Non-Triggerable)", + areaIds = mapOf( + Episode.II to listOf(10, 11), + ), + typeId = 546, + ), + LittleCryotube( + uniqueName = "Little Cryotube", + areaIds = mapOf( + Episode.II to listOf(10, 11, 17), + ), + typeId = 547, + properties = listOf( + EntityProp(name = "Model", offset = 52, type = EntityPropType.U32), + ), + ), + WideGlassWallBreakable( + uniqueName = "Wide Glass Wall (Breakable)", + areaIds = mapOf( + Episode.II to listOf(10, 11), + ), + typeId = 548, + ), + BlueFloatingRobot( + uniqueName = "Blue Floating Robot", + areaIds = mapOf( + Episode.II to listOf(10, 11), + ), + typeId = 549, + ), + RedFloatingRobot( + uniqueName = "Red Floating Robot", + areaIds = mapOf( + Episode.II to listOf(10, 11), + ), + typeId = 550, + ), + Dolphin( + uniqueName = "Dolphin", + areaIds = mapOf( + Episode.II to listOf(10, 11), + ), + typeId = 551, + ), + CaptureTrap( + uniqueName = "Capture Trap", + areaIds = mapOf( + Episode.II to listOf(5, 6, 7, 8, 9, 10, 11, 16, 17), + ), + typeId = 552, + ), + VRLink( + uniqueName = "VR Link", + areaIds = mapOf( + Episode.II to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17), + ), + typeId = 553, + ), + UnknownItem576( + uniqueName = "Unknown Item (576)", + areaIds = mapOf( + Episode.II to listOf(12), + ), + typeId = 576, + ), + WarpInBarbaRayRoom( + uniqueName = "Warp in Barba Ray Room", + areaIds = mapOf( + Episode.II to listOf(14), + ), + typeId = 640, + ), + UnknownItem672( + uniqueName = "Unknown Item (672)", + areaIds = mapOf( + Episode.II to listOf(15), + ), + typeId = 672, + ), + GeeNest( + uniqueName = "Gee Nest", + areaIds = mapOf( + Episode.I to listOf(8, 9, 10), + Episode.II to listOf(5, 6, 7, 8, 9, 16, 17), + Episode.IV to listOf(6, 7, 8), + ), + typeId = 688, + ), + LabComputerConsole( + uniqueName = "Lab Computer Console", + areaIds = mapOf( + Episode.II to listOf(0), + ), + typeId = 689, + ), + LabComputerConsoleGreenScreen( + uniqueName = "Lab Computer Console (Green Screen)", + areaIds = mapOf( + Episode.II to listOf(0), + ), + typeId = 690, + ), + ChairYellowPillow( + uniqueName = "Chair, Yellow Pillow", + areaIds = mapOf( + Episode.II to listOf(0), + ), + typeId = 691, + ), + OrangeWallWithHoleInMiddle( + uniqueName = "Orange Wall with Hole in Middle", + areaIds = mapOf( + Episode.II to listOf(0), + ), + typeId = 692, + ), + GreyWallWithHoleInMiddle( + uniqueName = "Grey Wall with Hole in Middle", + areaIds = mapOf( + Episode.II to listOf(0), + ), + typeId = 693, + ), + LongTable( + uniqueName = "Long Table", + areaIds = mapOf( + Episode.II to listOf(0), + ), + typeId = 694, + ), + GBAStation( + uniqueName = "GBA Station", + areaIds = mapOf(), + typeId = 695, + ), + TalkLinkToSupport( + uniqueName = "Talk (Link to Support)", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 0), + ), + typeId = 696, + ), + InstaWarp( + uniqueName = "Insta-Warp", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 6, 7, 11, 12, 13, 14, 16, 17), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 0), + ), + typeId = 697, + ), + LabInvisibleObject( + uniqueName = "Lab Invisible Object", + areaIds = mapOf( + Episode.I to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14), + Episode.II to listOf(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17), + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 0), + ), + typeId = 698, + ), + LabGlassWindowDoor( + uniqueName = "Lab Glass Window Door", + areaIds = mapOf( + Episode.II to listOf(0), + ), + typeId = 699, + ), + UnknownItem700( + uniqueName = "Unknown Item (700)", + areaIds = mapOf( + Episode.II to listOf(13), + ), + typeId = 700, + ), + LabCeilingWarp( + uniqueName = "Lab Ceiling Warp", + areaIds = mapOf( + Episode.II to listOf(0), + ), + typeId = 701, + ), + Ep4LightSource( + uniqueName = "Ep. IV Light Source", + areaIds = mapOf( + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8, 9), + ), + typeId = 768, + ), + Cactus( + uniqueName = "Cactus", + areaIds = mapOf( + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 769, + properties = listOf( + EntityProp(name = "Scale x", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Scale y", offset = 44, type = EntityPropType.F32), + EntityProp(name = "Scale z", offset = 48, type = EntityPropType.F32), + EntityProp(name = "Model", offset = 52, type = EntityPropType.U32), + ), + ), + BigBrownRock( + uniqueName = "Big Brown Rock", + areaIds = mapOf( + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 770, + properties = listOf( + EntityProp(name = "Model", offset = 52, type = EntityPropType.U32), + ), + ), + BreakableBrownRock( + uniqueName = "Breakable Brown Rock", + areaIds = mapOf( + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 771, + ), + UnknownItem832( + uniqueName = "Unknown Item (832)", + areaIds = mapOf(), + typeId = 832, + ), + UnknownItem833( + uniqueName = "Unknown Item (833)", + areaIds = mapOf(), + typeId = 833, + ), + PoisonPlant( + uniqueName = "Poison Plant", + areaIds = mapOf( + Episode.IV to listOf(6, 7, 8), + ), + typeId = 896, + ), + UnknownItem897( + uniqueName = "Unknown Item (897)", + areaIds = mapOf( + Episode.IV to listOf(6, 7, 8), + ), + typeId = 897, + ), + UnknownItem898( + uniqueName = "Unknown Item (898)", + areaIds = mapOf( + Episode.IV to listOf(6, 7, 8), + ), + typeId = 898, + ), + OozingDesertPlant( + uniqueName = "Oozing Desert Plant", + areaIds = mapOf( + Episode.IV to listOf(6, 7, 8), + ), + typeId = 899, + ), + UnknownItem901( + uniqueName = "Unknown Item (901)", + areaIds = mapOf( + Episode.IV to listOf(6, 7, 8), + ), + typeId = 901, + ), + BigBlackRocks( + uniqueName = "Big Black Rocks", + areaIds = mapOf( + Episode.IV to listOf(1, 2, 3, 4, 5, 6, 7, 8), + ), + typeId = 902, + properties = listOf( + EntityProp(name = "Model", offset = 52, type = EntityPropType.U32), + ), + ), + UnknownItem903( + uniqueName = "Unknown Item (903)", + areaIds = mapOf( + Episode.IV to listOf(6, 7, 8), + ), + typeId = 903, + ), + UnknownItem904( + uniqueName = "Unknown Item (904)", + areaIds = mapOf( + Episode.IV to listOf(6, 7, 8), + ), + typeId = 904, + ), + UnknownItem905( + uniqueName = "Unknown Item (905)", + areaIds = mapOf(), + typeId = 905, + ), + UnknownItem906( + uniqueName = "Unknown Item (906)", + areaIds = mapOf(), + typeId = 906, + ), + FallingRock( + uniqueName = "Falling Rock", + areaIds = mapOf( + Episode.IV to listOf(6, 7, 8), + ), + typeId = 907, + ), + DesertPlantHasCollision( + uniqueName = "Desert Plant (Has Collision)", + areaIds = mapOf( + Episode.IV to listOf(6, 7, 8), + ), + typeId = 908, + ), + DesertFixedTypeBoxBreakableCrystals( + uniqueName = "Desert Fixed Type Box (Breakable Crystals)", + areaIds = mapOf( + Episode.IV to listOf(6, 7, 8), + ), + typeId = 909, + ), + UnknownItem910( + uniqueName = "Unknown Item (910)", + areaIds = mapOf(), + typeId = 910, + ), + BeeHive( + uniqueName = "Bee Hive", + areaIds = mapOf( + Episode.IV to listOf(6, 7, 8), + ), + typeId = 911, + properties = listOf( + EntityProp(name = "Model", offset = 52, type = EntityPropType.U32), + ), + ), + UnknownItem912( + uniqueName = "Unknown Item (912)", + areaIds = mapOf( + Episode.IV to listOf(6, 7, 8), + ), + typeId = 912, + ), + Heat( + uniqueName = "Heat", + areaIds = mapOf( + Episode.IV to listOf(6, 7, 8), + ), + typeId = 913, + properties = listOf( + EntityProp(name = "Radius", offset = 40, type = EntityPropType.F32), + EntityProp(name = "Fog index no.", offset = 52, type = EntityPropType.I32), + ), + ), + TopOfSaintMillionEgg( + uniqueName = "Top of Saint Million Egg", + areaIds = mapOf( + Episode.IV to listOf(9), + ), + typeId = 960, + ), + UnknownItem961( + uniqueName = "Unknown Item (961)", + areaIds = mapOf( + Episode.IV to listOf(9), + ), + typeId = 961, + ); + + override val simpleName = uniqueName +} diff --git a/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/ObjectTypeFromId.kt b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/ObjectTypeFromId.kt new file mode 100644 index 00000000..1cbd8cbe --- /dev/null +++ b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/ObjectTypeFromId.kt @@ -0,0 +1,285 @@ +package world.phantasmal.lib.fileFormats.quest + +fun objectTypeFromId(id: Int): ObjectType = + when (id) { + 0 -> ObjectType.PlayerSet + 1 -> ObjectType.Particle + 2 -> ObjectType.Teleporter + 3 -> ObjectType.Warp + 4 -> ObjectType.LightCollision + 5 -> ObjectType.Item + 6 -> ObjectType.EnvSound + 7 -> ObjectType.FogCollision + 8 -> ObjectType.EventCollision + 9 -> ObjectType.CharaCollision + 10 -> ObjectType.ElementalTrap + 11 -> ObjectType.StatusTrap + 12 -> ObjectType.HealTrap + 13 -> ObjectType.LargeElementalTrap + 14 -> ObjectType.ObjRoomID + 15 -> ObjectType.Sensor + 16 -> ObjectType.UnknownItem16 + 17 -> ObjectType.LensFlare + 18 -> ObjectType.ScriptCollision + 19 -> ObjectType.HealRing + 20 -> ObjectType.MapCollision + 21 -> ObjectType.ScriptCollisionA + 22 -> ObjectType.ItemLight + 23 -> ObjectType.RadarCollision + 24 -> ObjectType.FogCollisionSW + 25 -> ObjectType.BossTeleporter + 26 -> ObjectType.ImageBoard + 27 -> ObjectType.QuestWarp + 28 -> ObjectType.Epilogue + 29 -> ObjectType.UnknownItem29 + 30 -> ObjectType.UnknownItem30 + 31 -> ObjectType.UnknownItem31 + 32 -> ObjectType.BoxDetectObject + 33 -> ObjectType.SymbolChatObject + 34 -> ObjectType.TouchPlateObject + 35 -> ObjectType.TargetableObject + 36 -> ObjectType.EffectObject + 37 -> ObjectType.CountDownObject + 38 -> ObjectType.UnknownItem38 + 39 -> ObjectType.UnknownItem39 + 40 -> ObjectType.UnknownItem40 + 41 -> ObjectType.UnknownItem41 + 64 -> ObjectType.MenuActivation + 65 -> ObjectType.TelepipeLocation + 66 -> ObjectType.BGMCollision + 67 -> ObjectType.MainRagolTeleporter + 68 -> ObjectType.LobbyTeleporter + 69 -> ObjectType.PrincipalWarp + 70 -> ObjectType.ShopDoor + 71 -> ObjectType.HuntersGuildDoor + 72 -> ObjectType.TeleporterDoor + 73 -> ObjectType.MedicalCenterDoor + 74 -> ObjectType.Elevator + 75 -> ObjectType.EasterEgg + 76 -> ObjectType.ValentinesHeart + 77 -> ObjectType.ChristmasTree + 78 -> ObjectType.ChristmasWreath + 79 -> ObjectType.HalloweenPumpkin + 80 -> ObjectType.TwentyFirstCentury + 81 -> ObjectType.Sonic + 82 -> ObjectType.WelcomeBoard + 83 -> ObjectType.Firework + 84 -> ObjectType.LobbyScreenDoor + 85 -> ObjectType.MainRagolTeleporterBattleInNextArea + 86 -> ObjectType.LabTeleporterDoor + 87 -> ObjectType.Pioneer2InvisibleTouchplate + 128 -> ObjectType.ForestDoor + 129 -> ObjectType.ForestSwitch + 130 -> ObjectType.LaserFence + 131 -> ObjectType.LaserSquareFence + 132 -> ObjectType.ForestLaserFenceSwitch + 133 -> ObjectType.LightRays + 134 -> ObjectType.BlueButterfly + 135 -> ObjectType.Probe + 136 -> ObjectType.RandomTypeBox1 + 137 -> ObjectType.ForestWeatherStation + 138 -> ObjectType.Battery + 139 -> ObjectType.ForestConsole + 140 -> ObjectType.BlackSlidingDoor + 141 -> ObjectType.RicoMessagePod + 142 -> ObjectType.EnergyBarrier + 143 -> ObjectType.ForestRisingBridge + 144 -> ObjectType.SwitchNoneDoor + 145 -> ObjectType.EnemyBoxGrey + 146 -> ObjectType.FixedTypeBox + 147 -> ObjectType.EnemyBoxBrown + 149 -> ObjectType.EmptyTypeBox + 150 -> ObjectType.LaserFenceEx + 151 -> ObjectType.LaserSquareFenceEx + 192 -> ObjectType.FloorPanel1 + 193 -> ObjectType.Caves4ButtonDoor + 194 -> ObjectType.CavesNormalDoor + 195 -> ObjectType.CavesSmashingPillar + 196 -> ObjectType.CavesSign1 + 197 -> ObjectType.CavesSign2 + 198 -> ObjectType.CavesSign3 + 199 -> ObjectType.HexagonalTank + 200 -> ObjectType.BrownPlatform + 201 -> ObjectType.WarningLightObject + 203 -> ObjectType.Rainbow + 204 -> ObjectType.FloatingJellyfish + 205 -> ObjectType.FloatingDragonfly + 206 -> ObjectType.CavesSwitchDoor + 207 -> ObjectType.RobotRechargeStation + 208 -> ObjectType.CavesCakeShop + 209 -> ObjectType.Caves1SmallRedRock + 210 -> ObjectType.Caves1MediumRedRock + 211 -> ObjectType.Caves1LargeRedRock + 212 -> ObjectType.Caves2SmallRock1 + 213 -> ObjectType.Caves2MediumRock1 + 214 -> ObjectType.Caves2LargeRock1 + 215 -> ObjectType.Caves2SmallRock2 + 216 -> ObjectType.Caves2MediumRock2 + 217 -> ObjectType.Caves2LargeRock2 + 218 -> ObjectType.Caves3SmallRock + 219 -> ObjectType.Caves3MediumRock + 220 -> ObjectType.Caves3LargeRock + 222 -> ObjectType.FloorPanel2 + 223 -> ObjectType.DestructableRockCaves1 + 224 -> ObjectType.DestructableRockCaves2 + 225 -> ObjectType.DestructableRockCaves3 + 256 -> ObjectType.MinesDoor + 257 -> ObjectType.FloorPanel3 + 258 -> ObjectType.MinesSwitchDoor + 259 -> ObjectType.LargeCryoTube + 260 -> ObjectType.ComputerLikeCalus + 261 -> ObjectType.GreenScreenOpeningAndClosing + 262 -> ObjectType.FloatingRobot + 263 -> ObjectType.FloatingBlueLight + 264 -> ObjectType.SelfDestructingObject1 + 265 -> ObjectType.SelfDestructingObject2 + 266 -> ObjectType.SelfDestructingObject3 + 267 -> ObjectType.SparkMachine + 268 -> ObjectType.MinesLargeFlashingCrate + 304 -> ObjectType.RuinsSeal + 320 -> ObjectType.RuinsTeleporter + 321 -> ObjectType.RuinsWarpSiteToSite + 322 -> ObjectType.RuinsSwitch + 323 -> ObjectType.FloorPanel4 + 324 -> ObjectType.Ruins1Door + 325 -> ObjectType.Ruins3Door + 326 -> ObjectType.Ruins2Door + 327 -> ObjectType.Ruins11ButtonDoor + 328 -> ObjectType.Ruins21ButtonDoor + 329 -> ObjectType.Ruins31ButtonDoor + 330 -> ObjectType.Ruins4ButtonDoor + 331 -> ObjectType.Ruins2ButtonDoor + 332 -> ObjectType.RuinsSensor + 333 -> ObjectType.RuinsFenceSwitch + 334 -> ObjectType.RuinsLaserFence4x2 + 335 -> ObjectType.RuinsLaserFence6x2 + 336 -> ObjectType.RuinsLaserFence4x4 + 337 -> ObjectType.RuinsLaserFence6x4 + 338 -> ObjectType.RuinsPoisonBlob + 339 -> ObjectType.RuinsPillarTrap + 340 -> ObjectType.PopupTrapNoTech + 341 -> ObjectType.RuinsCrystal + 342 -> ObjectType.Monument + 345 -> ObjectType.RuinsRock1 + 346 -> ObjectType.RuinsRock2 + 347 -> ObjectType.RuinsRock3 + 348 -> ObjectType.RuinsRock4 + 349 -> ObjectType.RuinsRock5 + 350 -> ObjectType.RuinsRock6 + 351 -> ObjectType.RuinsRock7 + 352 -> ObjectType.Poison + 353 -> ObjectType.FixedBoxTypeRuins + 354 -> ObjectType.RandomBoxTypeRuins + 355 -> ObjectType.EnemyTypeBoxYellow + 356 -> ObjectType.EnemyTypeBoxBlue + 357 -> ObjectType.EmptyTypeBoxBlue + 358 -> ObjectType.DestructableRock + 359 -> ObjectType.PopupTrapsTechs + 368 -> ObjectType.FlyingWhiteBird + 369 -> ObjectType.Tower + 370 -> ObjectType.FloatingRocks + 371 -> ObjectType.FloatingSoul + 372 -> ObjectType.Butterfly + 384 -> ObjectType.LobbyGameMenu + 385 -> ObjectType.LobbyWarpObject + 386 -> ObjectType.Lobby1EventObjectDefaultTree + 387 -> ObjectType.UnknownItem387 + 388 -> ObjectType.UnknownItem388 + 389 -> ObjectType.UnknownItem389 + 390 -> ObjectType.LobbyEventObjectStaticPumpkin + 391 -> ObjectType.LobbyEventObject3ChristmasWindows + 392 -> ObjectType.LobbyEventObjectRedAndWhiteCurtain + 393 -> ObjectType.UnknownItem393 + 394 -> ObjectType.UnknownItem394 + 395 -> ObjectType.LobbyFishTank + 396 -> ObjectType.LobbyEventObjectButterflies + 400 -> ObjectType.UnknownItem400 + 401 -> ObjectType.GreyWallLow + 402 -> ObjectType.SpaceshipDoor + 403 -> ObjectType.GreyWallHigh + 416 -> ObjectType.TempleNormalDoor + 417 -> ObjectType.BreakableWallWallButUnbreakable + 418 -> ObjectType.BrokenCylinderAndRubble + 419 -> ObjectType.ThreeBrokenWallPiecesOnFloor + 420 -> ObjectType.HighBrickCylinder + 421 -> ObjectType.LyingCylinder + 422 -> ObjectType.BrickConeWithFlatTop + 423 -> ObjectType.BreakableTempleWall + 424 -> ObjectType.TempleMapDetect + 425 -> ObjectType.SmallBrownBrickRisingBridge + 426 -> ObjectType.LongRisingBridgeWithPinkHighEdges + 427 -> ObjectType.FourSwitchTempleDoor + 448 -> ObjectType.FourButtonSpaceshipDoor + 512 -> ObjectType.ItemBoxCca + 513 -> ObjectType.TeleporterEp2 + 514 -> ObjectType.CcaDoor + 515 -> ObjectType.SpecialBoxCca + 516 -> ObjectType.BigCcaDoor + 517 -> ObjectType.BigCcaDoorSwitch + 518 -> ObjectType.LittleRock + 519 -> ObjectType.Little3StoneWall + 520 -> ObjectType.Medium3StoneWall + 521 -> ObjectType.SpiderPlant + 522 -> ObjectType.CcaAreaTeleporter + 523 -> ObjectType.UnknownItem523 + 524 -> ObjectType.WhiteBird + 525 -> ObjectType.OrangeBird + 527 -> ObjectType.Saw + 528 -> ObjectType.LaserDetect + 529 -> ObjectType.UnknownItem529 + 530 -> ObjectType.UnknownItem530 + 531 -> ObjectType.Seagull + 544 -> ObjectType.Fish + 545 -> ObjectType.SeabedDoorWithBlueEdges + 546 -> ObjectType.SeabedDoorAlwaysOpenNonTriggerable + 547 -> ObjectType.LittleCryotube + 548 -> ObjectType.WideGlassWallBreakable + 549 -> ObjectType.BlueFloatingRobot + 550 -> ObjectType.RedFloatingRobot + 551 -> ObjectType.Dolphin + 552 -> ObjectType.CaptureTrap + 553 -> ObjectType.VRLink + 576 -> ObjectType.UnknownItem576 + 640 -> ObjectType.WarpInBarbaRayRoom + 672 -> ObjectType.UnknownItem672 + 688 -> ObjectType.GeeNest + 689 -> ObjectType.LabComputerConsole + 690 -> ObjectType.LabComputerConsoleGreenScreen + 691 -> ObjectType.ChairYellowPillow + 692 -> ObjectType.OrangeWallWithHoleInMiddle + 693 -> ObjectType.GreyWallWithHoleInMiddle + 694 -> ObjectType.LongTable + 695 -> ObjectType.GBAStation + 696 -> ObjectType.TalkLinkToSupport + 697 -> ObjectType.InstaWarp + 698 -> ObjectType.LabInvisibleObject + 699 -> ObjectType.LabGlassWindowDoor + 700 -> ObjectType.UnknownItem700 + 701 -> ObjectType.LabCeilingWarp + 768 -> ObjectType.Ep4LightSource + 769 -> ObjectType.Cactus + 770 -> ObjectType.BigBrownRock + 771 -> ObjectType.BreakableBrownRock + 832 -> ObjectType.UnknownItem832 + 833 -> ObjectType.UnknownItem833 + 896 -> ObjectType.PoisonPlant + 897 -> ObjectType.UnknownItem897 + 898 -> ObjectType.UnknownItem898 + 899 -> ObjectType.OozingDesertPlant + 901 -> ObjectType.UnknownItem901 + 902 -> ObjectType.BigBlackRocks + 903 -> ObjectType.UnknownItem903 + 904 -> ObjectType.UnknownItem904 + 905 -> ObjectType.UnknownItem905 + 906 -> ObjectType.UnknownItem906 + 907 -> ObjectType.FallingRock + 908 -> ObjectType.DesertPlantHasCollision + 909 -> ObjectType.DesertFixedTypeBoxBreakableCrystals + 910 -> ObjectType.UnknownItem910 + 911 -> ObjectType.BeeHive + 912 -> ObjectType.UnknownItem912 + 913 -> ObjectType.Heat + 960 -> ObjectType.TopOfSaintMillionEgg + 961 -> ObjectType.UnknownItem961 + else -> ObjectType.Unknown + } diff --git a/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/QuestEntity.kt b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/QuestEntity.kt new file mode 100644 index 00000000..6abb12b3 --- /dev/null +++ b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/QuestEntity.kt @@ -0,0 +1,14 @@ +package world.phantasmal.lib.fileFormats.quest + +import world.phantasmal.lib.fileFormats.Vec3 + +interface QuestEntity { + val type: Type + + /** + * Section-relative position. + */ + var position: Vec3 + + var rotation: Vec3 +} diff --git a/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/QuestNpc.kt b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/QuestNpc.kt index 56c9f62e..6a470e5e 100644 --- a/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/QuestNpc.kt +++ b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/QuestNpc.kt @@ -1,9 +1,76 @@ package world.phantasmal.lib.fileFormats.quest import world.phantasmal.lib.buffer.Buffer +import world.phantasmal.lib.fileFormats.Vec3 +import world.phantasmal.lib.fileFormats.ninja.angleToRad +import world.phantasmal.lib.fileFormats.ninja.radToAngle import kotlin.math.roundToInt -class QuestNpc(var episode: Episode, var areaId: Int, val data: Buffer) { +class QuestNpc(var episode: Episode, var areaId: Int, val data: Buffer) : QuestEntity { + var typeId: Short + get() = data.getShort(0) + set(value) { + data.setShort(0, value) + } + + override var type: NpcType + get() = npcTypeFromQuestNpc(this) + set(value) { + value.episode?.let { episode = it } + typeId = (value.typeId ?: 0).toShort() + + when (value) { + NpcType.SaintMilion, + NpcType.SavageWolf, + NpcType.BarbarousWolf, + NpcType.PoisonLily, + NpcType.NarLily, + NpcType.PofuillySlime, + NpcType.PouillySlime, + NpcType.PoisonLily2, + NpcType.NarLily2, + NpcType.SavageWolf2, + NpcType.BarbarousWolf2, + NpcType.Kondrieu, + NpcType.Shambertin, + NpcType.SinowBeat, + NpcType.SinowGold, + NpcType.SatelliteLizard, + NpcType.Yowie, + -> special = value.special ?: false + + else -> { + // Do nothing. + } + } + + skin = value.skin ?: 0 + + if (value.areaIds.isNotEmpty() && areaId !in value.areaIds) { + areaId = value.areaIds.first() + } + } + + override var position: Vec3 + get() = Vec3(data.getFloat(20), data.getFloat(24), data.getFloat(28)) + set(value) { + data.setFloat(20, value.x) + data.setFloat(24, value.y) + data.setFloat(28, value.z) + } + + override var rotation: Vec3 + get() = Vec3( + angleToRad(data.getInt(32)), + angleToRad(data.getInt(36)), + angleToRad(data.getInt(40)), + ) + set(value) { + data.setInt(32, radToAngle(value.x)) + data.setInt(36, radToAngle(value.y)) + data.setInt(40, radToAngle(value.z)) + } + /** * Only seems to be valid for non-enemies. */ @@ -19,6 +86,12 @@ class QuestNpc(var episode: Episode, var areaId: Int, val data: Buffer) { data.setInt(64, value) } + var special: Boolean + get() = data.getFloat(48).roundToInt() == 1 + set(value) { + data.setFloat(48, if (value) 1f else 0f) + } + init { require(data.size == NPC_BYTE_SIZE) { "Data size should be $NPC_BYTE_SIZE but was ${data.size}." diff --git a/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/QuestObject.kt b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/QuestObject.kt index 36932d14..812dbfae 100644 --- a/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/QuestObject.kt +++ b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/QuestObject.kt @@ -1,13 +1,89 @@ package world.phantasmal.lib.fileFormats.quest import world.phantasmal.lib.buffer.Buffer +import world.phantasmal.lib.fileFormats.Vec3 +import world.phantasmal.lib.fileFormats.ninja.angleToRad +import world.phantasmal.lib.fileFormats.ninja.radToAngle +import kotlin.math.roundToInt -class QuestObject(var areaId: Int, val data: Buffer) { - var type: ObjectType - get() = TODO() - set(_) = TODO() - val scriptLabel: Int? = null // TODO Implement scriptLabel. - val scriptLabel2: Int? = null // TODO Implement scriptLabel2. +class QuestObject(var areaId: Int, val data: Buffer) : QuestEntity { + var typeId: Int + get() = data.getInt(0) + set(value) { + data.setInt(0, value) + } + + override var type: ObjectType + get() = objectTypeFromId(typeId) + set(value) { + typeId = value.typeId ?: -1 + } + + override var position: Vec3 + get() = Vec3(data.getFloat(16), data.getFloat(20), data.getFloat(24)) + set(value) { + data.setFloat(16, value.x) + data.setFloat(20, value.y) + data.setFloat(24, value.z) + } + + override var rotation: Vec3 + get() = Vec3( + angleToRad(data.getInt(28)), + angleToRad(data.getInt(32)), + angleToRad(data.getInt(36)), + ) + set(value) { + data.setInt(28, radToAngle(value.x)) + data.setInt(32, radToAngle(value.y)) + data.setInt(36, radToAngle(value.z)) + } + + val scriptLabel: Int? + get() = when (type) { + ObjectType.ScriptCollision, + ObjectType.ForestConsole, + ObjectType.TalkLinkToSupport, + -> data.getInt(52) + + ObjectType.RicoMessagePod, + -> data.getInt(56) + + else -> null + } + + val scriptLabel2: Int? + get() = if (type == ObjectType.RicoMessagePod) data.getInt(60) else null + + val model: Int? + get() = when (type) { + ObjectType.Probe, + -> data.getFloat(40).roundToInt() + + ObjectType.Saw, + ObjectType.LaserDetect, + -> data.getFloat(48).roundToInt() + + ObjectType.Sonic, + ObjectType.LittleCryotube, + ObjectType.Cactus, + ObjectType.BigBrownRock, + ObjectType.BigBlackRocks, + ObjectType.BeeHive, + -> data.getInt(52) + + ObjectType.ForestConsole, + -> data.getInt(56) + + ObjectType.PrincipalWarp, + ObjectType.LaserFence, + ObjectType.LaserSquareFence, + ObjectType.LaserFenceEx, + ObjectType.LaserSquareFenceEx, + -> data.getInt(60) + + else -> null + } init { require(data.size == OBJECT_BYTE_SIZE) { diff --git a/lib/src/commonTest/kotlin/world/phantasmal/lib/fileFormats/ninja/NinjaTests.kt b/lib/src/commonTest/kotlin/world/phantasmal/lib/fileFormats/ninja/NinjaTests.kt new file mode 100644 index 00000000..b5bd160a --- /dev/null +++ b/lib/src/commonTest/kotlin/world/phantasmal/lib/fileFormats/ninja/NinjaTests.kt @@ -0,0 +1,18 @@ +package world.phantasmal.lib.fileFormats.ninja + +import world.phantasmal.core.Success +import world.phantasmal.lib.test.asyncTest +import world.phantasmal.lib.test.readFile +import kotlin.test.Test +import kotlin.test.assertEquals +import kotlin.test.assertTrue + +class NinjaTests { + @Test + fun can_parse_rag_rappy_model() = asyncTest { + val result = parseNj(readFile("/RagRappy.nj")) + + assertTrue(result is Success) + assertEquals(1, result.value.size) + } +} diff --git a/lib/src/commonTest/resources/RagRappy.nj b/lib/src/commonTest/resources/RagRappy.nj new file mode 100644 index 00000000..e8d4b587 Binary files /dev/null and b/lib/src/commonTest/resources/RagRappy.nj differ diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/ListVal.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/ListVal.kt index 3a1c00ec..fdee657b 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/ListVal.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/ListVal.kt @@ -6,7 +6,7 @@ import world.phantasmal.observable.value.Val interface ListVal : Val> { val sizeVal: Val - fun observeList(observer: ListValObserver): Disposable + fun observeList(callNow: Boolean = false, observer: ListValObserver): Disposable fun sumBy(selector: (E) -> Int): Val = fold(0) { acc, el -> acc + selector(el) } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/ListValCreation.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/ListValCreation.kt index 89e6342e..83e846ca 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/ListValCreation.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/ListValCreation.kt @@ -1,6 +1,8 @@ package world.phantasmal.observable.value.list +fun listVal(vararg elements: E): ListVal = StaticListVal(elements.toList()) + fun mutableListVal( elements: MutableList = mutableListOf(), - extractObservables: ObservablesExtractor? = null + extractObservables: ObservablesExtractor? = null, ): MutableListVal = SimpleListVal(elements, extractObservables) diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/SimpleListVal.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/SimpleListVal.kt index c46b5e19..f0d5294f 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/SimpleListVal.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/SimpleListVal.kt @@ -97,7 +97,7 @@ class SimpleListVal( observers.add(observer) if (callNow) { - observer(ValChangeEvent(value, value)) + observer(ValChangeEvent(elements, elements)) } return disposable { @@ -106,13 +106,17 @@ class SimpleListVal( } } - override fun observeList(observer: ListValObserver): Disposable { + override fun observeList(callNow: Boolean, observer: ListValObserver): Disposable { if (elementObservers.isEmpty() && extractObservables != null) { replaceElementObservers(0, elementObservers.size, elements) } listObservers.add(observer) + if (callNow) { + observer(ListValChangeEvent.Change(0, emptyList(), elements)) + } + return disposable { listObservers.remove(observer) disposeElementObserversIfNecessary() @@ -141,7 +145,7 @@ class SimpleListVal( observer(event) } - val regularEvent = ValChangeEvent(value, value) + val regularEvent = ValChangeEvent(elements, elements) observers.forEach { observer: ValObserver> -> observer(regularEvent) diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/StaticListVal.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/StaticListVal.kt new file mode 100644 index 00000000..d5b5cd7d --- /dev/null +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/StaticListVal.kt @@ -0,0 +1,33 @@ +package world.phantasmal.observable.value.list + +import world.phantasmal.core.disposable.Disposable +import world.phantasmal.core.disposable.stubDisposable +import world.phantasmal.observable.Observer +import world.phantasmal.observable.value.Val +import world.phantasmal.observable.value.ValChangeEvent +import world.phantasmal.observable.value.ValObserver +import world.phantasmal.observable.value.value + +class StaticListVal(elements: List) : ListVal { + override val sizeVal: Val = value(elements.size) + + override val value: List = elements + + override fun observe(callNow: Boolean, observer: ValObserver>): Disposable { + if (callNow) { + observer(ValChangeEvent(value, value)) + } + + return stubDisposable() + } + + override fun observe(observer: Observer>): Disposable = stubDisposable() + + override fun observeList(callNow: Boolean, observer: ListValObserver): Disposable { + if (callNow) { + observer(ListValChangeEvent.Change(0, emptyList(), value)) + } + + return stubDisposable() + } +} diff --git a/observable/src/commonTest/kotlin/world/phantasmal/observable/value/StaticValTests.kt b/observable/src/commonTest/kotlin/world/phantasmal/observable/value/StaticValTests.kt index 2a303ed9..1ee9c19a 100644 --- a/observable/src/commonTest/kotlin/world/phantasmal/observable/value/StaticValTests.kt +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/value/StaticValTests.kt @@ -2,6 +2,7 @@ package world.phantasmal.observable.value import world.phantasmal.testUtils.TestSuite import kotlin.test.Test +import kotlin.test.assertEquals class StaticValTests : TestSuite() { @Test @@ -12,4 +13,15 @@ class StaticValTests : TestSuite() { static.observe(callNow = false) {} static.observe(callNow = true) {} } + + @Test + fun observe_respects_callNow() = test { + val static = StaticVal("test value") + var calls = 0 + + static.observe(callNow = false) { calls++ } + static.observe(callNow = true) { calls++ } + + assertEquals(1, calls) + } } diff --git a/observable/src/commonTest/kotlin/world/phantasmal/observable/value/list/StaticListValTests.kt b/observable/src/commonTest/kotlin/world/phantasmal/observable/value/list/StaticListValTests.kt new file mode 100644 index 00000000..a7b95002 --- /dev/null +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/value/list/StaticListValTests.kt @@ -0,0 +1,40 @@ +package world.phantasmal.observable.value.list + +import world.phantasmal.testUtils.TestSuite +import kotlin.test.Test +import kotlin.test.assertEquals + +class StaticListValTests : TestSuite() { + @Test + fun observing_StaticListVal_should_never_create_leaks() = test { + val static = StaticListVal(listOf(1, 2, 3)) + + static.observe {} + static.observe(callNow = false) {} + static.observe(callNow = true) {} + static.observeList(callNow = false) {} + static.observeList(callNow = true) {} + } + + @Test + fun observe_respects_callNow() = test { + val static = StaticListVal(listOf(1, 2, 3)) + var calls = 0 + + static.observe(callNow = false) { calls++ } + static.observe(callNow = true) { calls++ } + + assertEquals(1, calls) + } + + @Test + fun observeList_respects_callNow() = test { + val static = StaticListVal(listOf(1, 2, 3)) + var calls = 0 + + static.observeList(callNow = false) { calls++ } + static.observeList(callNow = true) { calls++ } + + assertEquals(1, calls) + } +} diff --git a/web/src/main/kotlin/world/phantasmal/web/Main.kt b/web/src/main/kotlin/world/phantasmal/web/Main.kt index 087788fd..a9a62e7b 100644 --- a/web/src/main/kotlin/world/phantasmal/web/Main.kt +++ b/web/src/main/kotlin/world/phantasmal/web/Main.kt @@ -15,9 +15,9 @@ import world.phantasmal.core.disposable.TrackedDisposable import world.phantasmal.core.disposable.disposable import world.phantasmal.observable.value.mutableVal import world.phantasmal.web.application.Application -import world.phantasmal.web.core.HttpAssetLoader +import world.phantasmal.web.core.loading.AssetLoader import world.phantasmal.web.core.stores.ApplicationUrl -import world.phantasmal.web.externals.Engine +import world.phantasmal.web.externals.babylon.Engine import world.phantasmal.webui.dom.disposableListener import world.phantasmal.webui.dom.root @@ -44,8 +44,9 @@ private fun init(): Disposable { disposer.add(disposable { httpClient.cancel() }) val pathname = window.location.pathname - val basePath = window.location.origin + - (if (pathname.lastOrNull() == '/') pathname.dropLast(1) else pathname) + val assetBasePath = window.location.origin + + (if (pathname.lastOrNull() == '/') pathname.dropLast(1) else pathname) + + "/assets" val scope = CoroutineScope(SupervisorJob()) disposer.add(disposable { scope.cancel() }) @@ -54,7 +55,7 @@ private fun init(): Disposable { Application( scope, rootElement, - HttpAssetLoader(httpClient, basePath), + AssetLoader(assetBasePath, httpClient), disposer.add(HistoryApplicationUrl()), createEngine = { Engine(it) } ) diff --git a/web/src/main/kotlin/world/phantasmal/web/application/Application.kt b/web/src/main/kotlin/world/phantasmal/web/application/Application.kt index dce6bc8f..ca06fac5 100644 --- a/web/src/main/kotlin/world/phantasmal/web/application/Application.kt +++ b/web/src/main/kotlin/world/phantasmal/web/application/Application.kt @@ -12,11 +12,11 @@ import world.phantasmal.web.application.controllers.NavigationController import world.phantasmal.web.application.widgets.ApplicationWidget import world.phantasmal.web.application.widgets.MainContentWidget import world.phantasmal.web.application.widgets.NavigationWidget -import world.phantasmal.web.core.AssetLoader +import world.phantasmal.web.core.loading.AssetLoader import world.phantasmal.web.core.stores.ApplicationUrl import world.phantasmal.web.core.stores.PwTool import world.phantasmal.web.core.stores.UiStore -import world.phantasmal.web.externals.Engine +import world.phantasmal.web.externals.babylon.Engine import world.phantasmal.web.huntOptimizer.HuntOptimizer import world.phantasmal.web.questEditor.QuestEditor import world.phantasmal.webui.DisposableContainer @@ -56,11 +56,19 @@ class Application( scope, NavigationWidget(scope, navigationController), MainContentWidget(scope, mainContentController, mapOf( - PwTool.QuestEditor to { s -> - addDisposable(QuestEditor(s, uiStore, createEngine)).createWidget() + PwTool.QuestEditor to { widgetScope -> + addDisposable(QuestEditor( + widgetScope, + assetLoader, + createEngine + )).createWidget() }, - PwTool.HuntOptimizer to { s -> - addDisposable(HuntOptimizer(s, assetLoader, uiStore)).createWidget() + PwTool.HuntOptimizer to { widgetScope -> + addDisposable(HuntOptimizer( + widgetScope, + assetLoader, + uiStore + )).createWidget() }, )) ) diff --git a/web/src/main/kotlin/world/phantasmal/web/application/widgets/NavigationWidget.kt b/web/src/main/kotlin/world/phantasmal/web/application/widgets/NavigationWidget.kt index b99c5164..68b4d590 100644 --- a/web/src/main/kotlin/world/phantasmal/web/application/widgets/NavigationWidget.kt +++ b/web/src/main/kotlin/world/phantasmal/web/application/widgets/NavigationWidget.kt @@ -4,7 +4,10 @@ import kotlinx.coroutines.CoroutineScope import org.w3c.dom.Node import world.phantasmal.observable.value.trueVal import world.phantasmal.web.application.controllers.NavigationController +import world.phantasmal.web.core.dom.externalLink +import world.phantasmal.webui.dom.Icon import world.phantasmal.webui.dom.div +import world.phantasmal.webui.dom.icon import world.phantasmal.webui.widgets.Select import world.phantasmal.webui.widgets.Widget @@ -36,6 +39,13 @@ class NavigationWidget( ) addWidget(serverSelect.label!!) addChild(serverSelect) + + externalLink("https://github.com/DaanVandenBosch/phantasmal-world") { + className = "pw-application-navigation-github" + title = "Phantasmal World is open source, code available on GitHub" + + icon(Icon.GitHub) + } } } @@ -67,11 +77,7 @@ class NavigationWidget( } .pw-application-navigation-github { - display: flex; - flex-direction: row; - align-items: center; - justify-content: center; - width: 30px; + margin: 0 6px 0 4px; font-size: 16px; color: var(--pw-control-text-color); } diff --git a/web/src/main/kotlin/world/phantasmal/web/core/AssetLoader.kt b/web/src/main/kotlin/world/phantasmal/web/core/AssetLoader.kt deleted file mode 100644 index 413e68f4..00000000 --- a/web/src/main/kotlin/world/phantasmal/web/core/AssetLoader.kt +++ /dev/null @@ -1,18 +0,0 @@ -package world.phantasmal.web.core - -import io.ktor.client.* -import io.ktor.client.request.* -import world.phantasmal.web.core.dto.QuestDto -import world.phantasmal.web.core.models.Server - -interface AssetLoader { - suspend fun getQuests(server: Server): List -} - -class HttpAssetLoader( - private val httpClient: HttpClient, - private val basePath: String, -) : AssetLoader { - override suspend fun getQuests(server: Server): List = - httpClient.get("$basePath/assets/quests.${server.slug}.json") -} diff --git a/web/src/main/kotlin/world/phantasmal/web/core/dom/Dom.kt b/web/src/main/kotlin/world/phantasmal/web/core/dom/Dom.kt new file mode 100644 index 00000000..c2b17014 --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/core/dom/Dom.kt @@ -0,0 +1,13 @@ +package world.phantasmal.web.core.dom + +import org.w3c.dom.HTMLAnchorElement +import org.w3c.dom.Node +import world.phantasmal.webui.dom.appendHtmlEl + +fun Node.externalLink(href: String, block: HTMLAnchorElement.() -> Unit) = + appendHtmlEl("A") { + target = "_blank" + rel = "noopener noreferrer" + this.href = href + block() + } diff --git a/web/src/main/kotlin/world/phantasmal/web/core/loading/AssetLoader.kt b/web/src/main/kotlin/world/phantasmal/web/core/loading/AssetLoader.kt new file mode 100644 index 00000000..7744ab49 --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/core/loading/AssetLoader.kt @@ -0,0 +1,21 @@ +package world.phantasmal.web.core.loading + +import io.ktor.client.* +import io.ktor.client.request.* +import io.ktor.client.statement.* +import io.ktor.http.* +import org.khronos.webgl.ArrayBuffer + +class AssetLoader(val basePath: String, val httpClient: HttpClient) { + suspend inline fun load(path: String): T = + httpClient.get("$basePath$path") + + suspend fun loadArrayBuffer(path: String): ArrayBuffer { + val response = load(path) + val channel = response.content + val arrayBuffer = ArrayBuffer(response.contentLength()?.toInt() ?: channel.availableForRead) + channel.readFully(arrayBuffer, 0, arrayBuffer.byteLength) + check(channel.availableForRead == 0) { "Couldn't read all data." } + return arrayBuffer + } +} diff --git a/web/src/main/kotlin/world/phantasmal/web/core/rendering/Renderer.kt b/web/src/main/kotlin/world/phantasmal/web/core/rendering/Renderer.kt index 0f8155fd..15c8b4a6 100644 --- a/web/src/main/kotlin/world/phantasmal/web/core/rendering/Renderer.kt +++ b/web/src/main/kotlin/world/phantasmal/web/core/rendering/Renderer.kt @@ -2,25 +2,28 @@ package world.phantasmal.web.core.rendering import org.w3c.dom.HTMLCanvasElement import world.phantasmal.core.disposable.TrackedDisposable -import world.phantasmal.web.externals.Engine -import world.phantasmal.web.externals.Scene +import world.phantasmal.web.externals.babylon.Engine +import world.phantasmal.web.externals.babylon.Scene abstract class Renderer( protected val canvas: HTMLCanvasElement, - createEngine: (HTMLCanvasElement) -> Engine, + protected val engine: Engine, ) : TrackedDisposable() { - protected val engine = createEngine(canvas) protected val scene = Scene(engine) init { - println(engine.description) - engine.runRenderLoop { scene.render() } } override fun internalDispose() { - // TODO: Clean up Babylon resources. + scene.dispose() + engine.dispose() + super.internalDispose() + } + + fun scheduleRender() { + // TODO: Remove scheduleRender? } } diff --git a/web/src/main/kotlin/world/phantasmal/web/core/rendering/conversion/Conversion.kt b/web/src/main/kotlin/world/phantasmal/web/core/rendering/conversion/Conversion.kt new file mode 100644 index 00000000..7a88b87d --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/core/rendering/conversion/Conversion.kt @@ -0,0 +1,10 @@ +package world.phantasmal.web.core.rendering.conversion + +import world.phantasmal.lib.fileFormats.Vec2 +import world.phantasmal.lib.fileFormats.Vec3 +import world.phantasmal.web.externals.babylon.Vector2 +import world.phantasmal.web.externals.babylon.Vector3 + +fun vec2ToBabylon(v: Vec2): Vector2 = Vector2(v.x.toDouble(), v.y.toDouble()) + +fun vec3ToBabylon(v: Vec3): Vector3 = Vector3(v.x.toDouble(), v.y.toDouble(), v.z.toDouble()) diff --git a/web/src/main/kotlin/world/phantasmal/web/core/rendering/conversion/NinjaGeometryConversion.kt b/web/src/main/kotlin/world/phantasmal/web/core/rendering/conversion/NinjaGeometryConversion.kt new file mode 100644 index 00000000..82ede735 --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/core/rendering/conversion/NinjaGeometryConversion.kt @@ -0,0 +1,212 @@ +package world.phantasmal.web.core.rendering.conversion + +import mu.KotlinLogging +import world.phantasmal.lib.fileFormats.Vec3 +import world.phantasmal.lib.fileFormats.ninja.NinjaModel +import world.phantasmal.lib.fileFormats.ninja.NinjaObject +import world.phantasmal.lib.fileFormats.ninja.NjcmModel +import world.phantasmal.lib.fileFormats.ninja.XjModel +import world.phantasmal.web.externals.babylon.* +import kotlin.math.cos +import kotlin.math.sin + +private val logger = KotlinLogging.logger {} + +private val DEFAULT_NORMAL = Vector3.Up() +private val DEFAULT_UV = Vector2.Zero() +private val NO_TRANSLATION = Vector3.Zero() +private val NO_ROTATION = Quaternion.Identity() +private val NO_SCALE = Vector3.One() + +// TODO: take into account different kinds of meshes/vertices (with or without normals, uv, etc.). +fun ninjaObjectToVertexData(ninjaObject: NinjaObject<*>): VertexData = + NinjaToVertexDataConverter(VertexDataBuilder()).convert(ninjaObject) + +private class NinjaToVertexDataConverter(private val builder: VertexDataBuilder) { + private val vertexHolder = VertexHolder() + private var boneIndex = 0 + + fun convert(ninjaObject: NinjaObject<*>): VertexData { + objectToVertexData(ninjaObject, Matrix.Identity()) + return builder.build() + } + + private fun objectToVertexData(obj: NinjaObject<*>, parentMatrix: Matrix) { + val ef = obj.evaluationFlags + + val matrix = Matrix.Compose( + if (ef.noScale) NO_SCALE else vec3ToBabylon(obj.scale), + if (ef.noRotate) NO_ROTATION else eulerToQuat(obj.rotation, ef.zxyRotationOrder), + if (ef.noTranslate) NO_TRANSLATION else vec3ToBabylon(obj.position), + ) + + parentMatrix.multiplyToRef(matrix, matrix) + + if (!ef.hidden) { + obj.model?.let { model -> + modelToVertexData(model, matrix) + } + } + + boneIndex++ + + if (!ef.breakChildTrace) { + obj.children.forEach { child -> + objectToVertexData(child, matrix) + } + } + } + + private fun modelToVertexData(model: NinjaModel, matrix: Matrix) = + when (model) { + is NjcmModel -> njcmModelToVertexData(model, matrix) + is XjModel -> xjModelToVertexData(model, matrix) + } + + private fun njcmModelToVertexData(model: NjcmModel, matrix: Matrix) { + val normalMatrix = Matrix.Identity() + matrix.toNormalMatrix(normalMatrix) + + val newVertices = model.vertices.map { vertex -> + vertex?.let { + val position = vec3ToBabylon(vertex.position) + val normal = vertex.normal?.let(::vec3ToBabylon) ?: Vector3.Up() + + Vector3.TransformCoordinatesToRef(position, matrix, position) + Vector3.TransformNormalToRef(normal, normalMatrix, normal) + + Vertex( + boneIndex, + position, + normal, + vertex.boneWeight, + vertex.boneWeightStatus, + vertex.calcContinue, + ) + } + } + + vertexHolder.add(newVertices) + + for (mesh in model.meshes) { + val startIndexCount = builder.indexCount + var i = 0 + + for (meshVertex in mesh.vertices) { + val vertices = vertexHolder.get(meshVertex.index.toInt()) + + if (vertices.isEmpty()) { + logger.debug { + "Mesh refers to nonexistent vertex with index ${meshVertex.index}." + } + } else { + val vertex = vertices.last() + val normal = + vertex.normal ?: meshVertex.normal?.let(::vec3ToBabylon) ?: DEFAULT_NORMAL + val index = builder.vertexCount + + builder.addVertex( + vertex.position, + normal, + meshVertex.texCoords?.let(::vec2ToBabylon) ?: DEFAULT_UV + ) + + if (i >= 2) { + if (i % 2 == if (mesh.clockwiseWinding) 1 else 0) { + builder.addIndex(index - 2) + builder.addIndex(index - 1) + builder.addIndex(index) + } else { + builder.addIndex(index - 2) + builder.addIndex(index) + builder.addIndex(index - 1) + } + } + + val boneIndices = IntArray(4) + val boneWeights = FloatArray(4) + + for (v in vertices) { + boneIndices[v.boneWeightStatus] = v.boneIndex + boneWeights[v.boneWeightStatus] = v.boneWeight + } + + val totalWeight = boneWeights.sum() + + for (j in boneIndices.indices) { + builder.addBoneWeight( + boneIndices[j], + if (totalWeight > 0f) boneWeights[j] / totalWeight else 0f + ) + } + + i++ + } + } + + // TODO: support multiple materials +// builder.addGroup( +// startIndexCount +// ) + } + } + + private fun xjModelToVertexData(model: XjModel, matrix: Matrix) {} +} + +private class Vertex( + val boneIndex: Int, + val position: Vector3, + val normal: Vector3?, + val boneWeight: Float, + val boneWeightStatus: Int, + val calcContinue: Boolean, +) + +private class VertexHolder { + private val stack = mutableListOf>() + + fun add(vertices: List) { + vertices.forEachIndexed { i, vertex -> + if (i >= stack.size) { + stack.add(mutableListOf()) + } + + if (vertex != null) { + stack[i].add(vertex) + } + } + } + + fun get(index: Int): List = stack[index] +} + +private fun eulerToQuat(angles: Vec3, zxyRotationOrder: Boolean): Quaternion { + val x = angles.x.toDouble() + val y = angles.y.toDouble() + val z = angles.z.toDouble() + + val c1 = cos(x / 2) + val c2 = cos(y / 2) + val c3 = cos(z / 2) + + val s1 = sin(x / 2) + val s2 = sin(y / 2) + val s3 = sin(z / 2) + + return if (zxyRotationOrder) { + Quaternion( + s1 * c2 * c3 - c1 * s2 * s3, + c1 * s2 * c3 + s1 * c2 * s3, + c1 * c2 * s3 + s1 * s2 * c3, + c1 * c2 * c3 - s1 * s2 * s3, + ) + } else { + Quaternion( + s1 * c2 * c3 - c1 * s2 * s3, + c1 * s2 * c3 + s1 * c2 * s3, + c1 * c2 * s3 - s1 * s2 * c3, + c1 * c2 * c3 + s1 * s2 * s3, + ) + } +} diff --git a/web/src/main/kotlin/world/phantasmal/web/core/rendering/conversion/VertexDataBuilder.kt b/web/src/main/kotlin/world/phantasmal/web/core/rendering/conversion/VertexDataBuilder.kt new file mode 100644 index 00000000..8a9b195e --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/core/rendering/conversion/VertexDataBuilder.kt @@ -0,0 +1,78 @@ +package world.phantasmal.web.core.rendering.conversion + +import org.khronos.webgl.Float32Array +import org.khronos.webgl.Uint16Array +import org.khronos.webgl.set +import world.phantasmal.web.externals.babylon.Vector2 +import world.phantasmal.web.externals.babylon.Vector3 +import world.phantasmal.web.externals.babylon.VertexData + +class VertexDataBuilder { + private val positions = mutableListOf() + private val normals = mutableListOf() + private val uvs = mutableListOf() + private val indices = mutableListOf() + private val boneIndices = mutableListOf() + private val boneWeights = mutableListOf() + + val vertexCount: Int + get() = positions.size + + val indexCount: Int + get() = indices.size + + fun addVertex(position: Vector3, normal: Vector3, uv: Vector2) { + positions.add(position) + normals.add(normal) + uvs.add(uv) + } + + fun addIndex(index: Int) { + indices.add(index.toShort()) + } + + fun addBoneWeight(index: Int, weight: Float) { + boneIndices.add(index.toShort()) + boneWeights.add(weight) + } + + // TODO: support multiple materials +// fun addGroup( +// offset: Int, +// size: Int, +// textureId: Int?, +// alpha: Boolean = false, +// additiveBlending: Boolean = false, +// ) { +// +// } + + fun build(): VertexData { + val positions = Float32Array(3 * positions.size) + val normals = Float32Array(3 * normals.size) + val uvs = Float32Array(2 * uvs.size) + + for (i in this.positions.indices) { + val pos = this.positions[i] + positions[3 * i] = pos.x.toFloat() + positions[3 * i + 1] = pos.y.toFloat() + positions[3 * i + 2] = pos.z.toFloat() + + val normal = this.normals[i] + normals[3 * i] = normal.x.toFloat() + normals[3 * i + 1] = normal.y.toFloat() + normals[3 * i + 2] = normal.z.toFloat() + + val uv = this.uvs[i] + uvs[2 * i] = uv.x.toFloat() + uvs[2 * i + 1] = uv.y.toFloat() + } + + val data = VertexData() + data.positions = positions + data.normals = normals + data.uvs = uvs + data.indices = Uint16Array(indices.toTypedArray()) + return data + } +} diff --git a/web/src/main/kotlin/world/phantasmal/web/core/widgets/DockWidget.kt b/web/src/main/kotlin/world/phantasmal/web/core/widgets/DockWidget.kt index cb95cce0..c5a6a938 100644 --- a/web/src/main/kotlin/world/phantasmal/web/core/widgets/DockWidget.kt +++ b/web/src/main/kotlin/world/phantasmal/web/core/widgets/DockWidget.kt @@ -4,8 +4,8 @@ import kotlinx.coroutines.CoroutineScope import org.w3c.dom.Node import world.phantasmal.observable.value.Val import world.phantasmal.observable.value.falseVal -import world.phantasmal.webui.newJsObject -import world.phantasmal.web.externals.GoldenLayout +import world.phantasmal.webui.obj +import world.phantasmal.web.externals.goldenLayout.GoldenLayout import world.phantasmal.webui.dom.div import world.phantasmal.webui.widgets.Widget @@ -61,13 +61,13 @@ class DockWidget( val idToCreate = mutableMapOf Widget>() - val config = newJsObject { - settings = newJsObject { + val config = obj { + settings = obj { showPopoutIcon = false showMaximiseIcon = false showCloseIcon = false } - dimensions = newJsObject { + dimensions = obj { headerHeight = HEADER_HEIGHT } content = arrayOf( @@ -120,7 +120,7 @@ class DockWidget( is DockedWidget -> { idToCreate[item.id] = item.createWidget - newJsObject { + obj { title = item.title type = "component" componentName = item.id @@ -134,7 +134,7 @@ class DockWidget( } is DockedContainer -> - newJsObject { + obj { type = itemType content = Array(item.items.size) { toConfigContent(item.items[it], idToCreate) } diff --git a/web/src/main/kotlin/world/phantasmal/web/core/widgets/RendererWidget.kt b/web/src/main/kotlin/world/phantasmal/web/core/widgets/RendererWidget.kt index e341d0b8..a424273b 100644 --- a/web/src/main/kotlin/world/phantasmal/web/core/widgets/RendererWidget.kt +++ b/web/src/main/kotlin/world/phantasmal/web/core/widgets/RendererWidget.kt @@ -3,22 +3,22 @@ package world.phantasmal.web.core.widgets import kotlinx.coroutines.CoroutineScope import org.w3c.dom.HTMLCanvasElement import org.w3c.dom.Node -import world.phantasmal.web.externals.Engine -import world.phantasmal.web.questEditor.rendering.QuestRenderer +import world.phantasmal.web.core.rendering.Renderer import world.phantasmal.webui.dom.canvas import world.phantasmal.webui.widgets.Widget import kotlin.math.floor class RendererWidget( scope: CoroutineScope, - private val createEngine: (HTMLCanvasElement) -> Engine, + private val createRenderer: (HTMLCanvasElement) -> Renderer, ) : Widget(scope) { override fun Node.createElement() = canvas { className = "pw-core-renderer" + tabIndex = -1 observeResize() - addDisposable(QuestRenderer(this, createEngine)) + addDisposable(createRenderer(this)) } override fun resized(width: Double, height: Double) { @@ -35,6 +35,7 @@ class RendererWidget( .pw-core-renderer { width: 100%; height: 100%; + outline: none; } """.trimIndent()) } diff --git a/web/src/main/kotlin/world/phantasmal/web/externals/babylon.kt b/web/src/main/kotlin/world/phantasmal/web/externals/babylon.kt deleted file mode 100644 index 77b17611..00000000 --- a/web/src/main/kotlin/world/phantasmal/web/externals/babylon.kt +++ /dev/null @@ -1,105 +0,0 @@ -@file:JsModule("@babylonjs/core") -@file:JsNonModule - -package world.phantasmal.web.externals - -import org.w3c.dom.HTMLCanvasElement -import org.w3c.dom.HTMLElement - -external class Vector3(x: Double, y: Double, z: Double) { - var x: Double - var y: Double - var z: Double - - fun toQuaternion(): Quaternion - - fun addInPlace(otherVector: Vector3): Vector3 - - fun addInPlaceFromFloats(x: Double, y: Double, z: Double): Vector3 - - companion object { - fun Zero(): Vector3 - } -} - -external class Quaternion - -open external class ThinEngine { - val description: String - - /** - * Register and execute a render loop. The engine can have more than one render function - * @param renderFunction defines the function to continuously execute - */ - fun runRenderLoop(renderFunction: () -> Unit) -} - -external class Engine( - canvasOrContext: HTMLCanvasElement?, - antialias: Boolean = definedExternally, -) : ThinEngine - -external class Scene(engine: Engine) { - fun render() -} - -open external class Node { - /** - * Releases resources associated with this node. - * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default) - * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default) - */ - fun dispose( - doNotRecurse: Boolean = definedExternally, - disposeMaterialAndTextures: Boolean = definedExternally, - ) -} - -open external class Camera : Node { - fun attachControl(element: HTMLElement, noPreventDefault: Boolean = definedExternally) -} - -open external class TargetCamera : Camera - -/** - * @param setActiveOnSceneIfNoneActive default true - */ -external class ArcRotateCamera( - name: String, - alpha: Double, - beta: Double, - radius: Double, - target: Vector3, - scene: Scene, - setActiveOnSceneIfNoneActive: Boolean = definedExternally, -) : TargetCamera - -abstract external class Light : Node - -external class HemisphericLight(name: String, direction: Vector3, scene: Scene) : Light - -open external class TransformNode : Node - -abstract external class AbstractMesh : TransformNode - -external class Mesh : AbstractMesh - -external class MeshBuilder { - companion object { - interface CreateCylinderOptions { - var height: Double - var diameterTop: Double - var diameterBottom: Double - var diameter: Double - var tessellation: Double - var subdivisions: Double - var arc: Double - } - - fun CreateCylinder( - name: String, - options: CreateCylinderOptions, - scene: Scene? = definedExternally, - ): Mesh - } -} diff --git a/web/src/main/kotlin/world/phantasmal/web/externals/babylon/babylon.kt b/web/src/main/kotlin/world/phantasmal/web/externals/babylon/babylon.kt new file mode 100644 index 00000000..9c502c7f --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/externals/babylon/babylon.kt @@ -0,0 +1,228 @@ +@file:JsModule("@babylonjs/core") +@file:JsNonModule +@file:Suppress("FunctionName", "unused") + +package world.phantasmal.web.externals.babylon + +import org.khronos.webgl.Float32Array +import org.khronos.webgl.Uint16Array +import org.w3c.dom.HTMLCanvasElement + +external class Vector2(x: Double, y: Double) { + var x: Double + var y: Double + + fun addInPlace(otherVector: Vector2): Vector2 + + fun addInPlaceFromFloats(x: Double, y: Double): Vector2 + + fun copyFrom(source: Vector2): Vector2 + + companion object { + fun Zero(): Vector2 + } +} + +external class Vector3(x: Double, y: Double, z: Double) { + var x: Double + var y: Double + var z: Double + + fun toQuaternion(): Quaternion + + fun addInPlace(otherVector: Vector3): Vector3 + + fun addInPlaceFromFloats(x: Double, y: Double, z: Double): Vector3 + + fun copyFrom(source: Vector3): Vector3 + + companion object { + fun One(): Vector3 + fun Up(): Vector3 + fun Zero(): Vector3 + fun TransformCoordinates(vector: Vector3, transformation: Matrix): Vector3 + fun TransformCoordinatesToRef(vector: Vector3, transformation: Matrix, result: Vector3) + fun TransformNormal(vector: Vector3, transformation: Matrix): Vector3 + fun TransformNormalToRef(vector: Vector3, transformation: Matrix, result: Vector3) + } +} + +external class Quaternion( + x: Double = definedExternally, + y: Double = definedExternally, + z: Double = definedExternally, + w: Double = definedExternally, +) { + /** + * Multiplies two quaternions + * @return a new quaternion set as the multiplication result of the current one with the given one "q1" + */ + fun multiply(q1: Quaternion): Quaternion + + /** + * Updates the current quaternion with the multiplication of itself with the given one "q1" + * @return the current, updated quaternion + */ + fun multiplyInPlace(q1: Quaternion): Quaternion + + /** + * Sets the given "result" as the the multiplication result of the current one with the given one "q1" + * @return the current quaternion + */ + fun multiplyToRef(q1: Quaternion, result: Quaternion): Quaternion + + companion object { + fun Identity(): Quaternion + fun FromEulerAngles(x: Double, y: Double, z: Double): Quaternion + fun RotationYawPitchRoll(yaw: Double, pitch: Double, roll: Double): Quaternion + } +} + +external class Matrix { + fun multiply(other: Matrix): Matrix + fun multiplyToRef(other: Matrix, result: Matrix): Matrix + fun toNormalMatrix(ref: Matrix) + + companion object { + fun Identity(): Matrix + fun Compose(scale: Vector3, rotation: Quaternion, translation: Vector3): Matrix + } +} + +open external class ThinEngine { + val description: String + + /** + * Register and execute a render loop. The engine can have more than one render function + * @param renderFunction defines the function to continuously execute + */ + fun runRenderLoop(renderFunction: () -> Unit) + + fun dispose() +} + +external class Engine( + canvasOrContext: HTMLCanvasElement?, + antialias: Boolean = definedExternally, +) : ThinEngine + +external class Scene(engine: Engine) { + fun render() + fun addLight(light: Light) + fun addMesh(newMesh: AbstractMesh, recursive: Boolean? = definedExternally) + fun addTransformNode(newTransformNode: TransformNode) + fun removeLight(toRemove: Light) + fun removeMesh(toRemove: TransformNode, recursive: Boolean? = definedExternally) + fun removeTransformNode(toRemove: TransformNode) + fun dispose() +} + +open external class Node { + var metadata: Any? + var parent: Node? + var position: Vector3 + var rotation: Vector3 + var scaling: Vector3 + + fun setEnabled(value: Boolean) + + /** + * Releases resources associated with this node. + * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default) + * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default) + */ + fun dispose( + doNotRecurse: Boolean = definedExternally, + disposeMaterialAndTextures: Boolean = definedExternally, + ) +} + +open external class Camera : Node { + fun attachControl(noPreventDefault: Boolean = definedExternally) +} + +open external class TargetCamera : Camera + +/** + * @param setActiveOnSceneIfNoneActive default true + */ +external class ArcRotateCamera( + name: String, + alpha: Double, + beta: Double, + radius: Double, + target: Vector3, + scene: Scene, + setActiveOnSceneIfNoneActive: Boolean = definedExternally, +) : TargetCamera { + var inertia: Double + var angularSensibilityX: Double + var angularSensibilityY: Double + var panningInertia: Double + var panningSensibility: Double + var panningAxis: Vector3 + var pinchDeltaPercentage: Double + var wheelDeltaPercentage: Double + + fun attachControl( + element: HTMLCanvasElement, + noPreventDefault: Boolean, + useCtrlForPanning: Boolean, + panningMouseButton: Int, + ) +} + +abstract external class Light : Node + +external class HemisphericLight(name: String, direction: Vector3, scene: Scene) : Light + +open external class TransformNode( + name: String, + scene: Scene? = definedExternally, + isPure: Boolean = definedExternally, +) : Node { +} + +abstract external class AbstractMesh : TransformNode + +external class Mesh( + name: String, + scene: Scene? = definedExternally, + parent: Node? = definedExternally, + source: Mesh? = definedExternally, + doNotCloneChildren: Boolean = definedExternally, + clonePhysicsImpostor: Boolean = definedExternally, +) : AbstractMesh { + fun createInstance(name: String): InstancedMesh +} + +external class InstancedMesh : AbstractMesh + +external class MeshBuilder { + companion object { + interface CreateCylinderOptions { + var height: Double + var diameterTop: Double + var diameterBottom: Double + var diameter: Double + var tessellation: Double + var subdivisions: Double + var arc: Double + } + + fun CreateCylinder( + name: String, + options: CreateCylinderOptions, + scene: Scene? = definedExternally, + ): Mesh + } +} + +external class VertexData { + var positions: Float32Array? // number[] | Float32Array + var normals: Float32Array? // number[] | Float32Array + var uvs: Float32Array? // number[] | Float32Array + var indices: Uint16Array? // number[] | Int32Array | Uint32Array | Uint16Array + + fun applyToMesh(mesh: Mesh, updatable: Boolean = definedExternally): VertexData +} diff --git a/web/src/main/kotlin/world/phantasmal/web/externals/golden-layout.kt b/web/src/main/kotlin/world/phantasmal/web/externals/goldenLayout/goldenLayout.kt similarity index 98% rename from web/src/main/kotlin/world/phantasmal/web/externals/golden-layout.kt rename to web/src/main/kotlin/world/phantasmal/web/externals/goldenLayout/goldenLayout.kt index 80c7186a..ea139e92 100644 --- a/web/src/main/kotlin/world/phantasmal/web/externals/golden-layout.kt +++ b/web/src/main/kotlin/world/phantasmal/web/externals/goldenLayout/goldenLayout.kt @@ -1,4 +1,6 @@ -package world.phantasmal.web.externals +@file:Suppress("unused") + +package world.phantasmal.web.externals.goldenLayout import org.w3c.dom.Element diff --git a/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/HuntOptimizer.kt b/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/HuntOptimizer.kt index 2e6b9084..540342da 100644 --- a/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/HuntOptimizer.kt +++ b/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/HuntOptimizer.kt @@ -1,7 +1,7 @@ package world.phantasmal.web.huntOptimizer import kotlinx.coroutines.CoroutineScope -import world.phantasmal.web.core.AssetLoader +import world.phantasmal.web.core.loading.AssetLoader import world.phantasmal.web.core.stores.UiStore import world.phantasmal.web.huntOptimizer.controllers.HuntOptimizerController import world.phantasmal.web.huntOptimizer.controllers.MethodsController diff --git a/web/src/main/kotlin/world/phantasmal/web/core/dto/QuestDto.kt b/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/dto/QuestDto.kt similarity index 79% rename from web/src/main/kotlin/world/phantasmal/web/core/dto/QuestDto.kt rename to web/src/main/kotlin/world/phantasmal/web/huntOptimizer/dto/QuestDto.kt index 8690e8d8..53226840 100644 --- a/web/src/main/kotlin/world/phantasmal/web/core/dto/QuestDto.kt +++ b/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/dto/QuestDto.kt @@ -1,4 +1,4 @@ -package world.phantasmal.web.core.dto +package world.phantasmal.web.huntOptimizer.dto import kotlinx.serialization.Serializable diff --git a/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/stores/HuntMethodStore.kt b/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/stores/HuntMethodStore.kt index c1dcabf2..4c5bc927 100644 --- a/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/stores/HuntMethodStore.kt +++ b/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/stores/HuntMethodStore.kt @@ -7,11 +7,12 @@ import world.phantasmal.lib.fileFormats.quest.Episode import world.phantasmal.lib.fileFormats.quest.NpcType import world.phantasmal.observable.value.list.ListVal import world.phantasmal.observable.value.list.mutableListVal -import world.phantasmal.web.core.AssetLoader import world.phantasmal.web.core.IoDispatcher import world.phantasmal.web.core.UiDispatcher +import world.phantasmal.web.core.loading.AssetLoader import world.phantasmal.web.core.models.Server import world.phantasmal.web.core.stores.UiStore +import world.phantasmal.web.huntOptimizer.dto.QuestDto import world.phantasmal.web.huntOptimizer.models.HuntMethodModel import world.phantasmal.web.huntOptimizer.models.SimpleQuestModel import world.phantasmal.webui.stores.Store @@ -34,9 +35,10 @@ class HuntMethodStore( private fun loadMethods(server: Server) { launch(IoDispatcher) { - val quests = assetLoader.getQuests(server) + val quests = assetLoader.load>("/quests.${server.slug}.json") - val methods = quests.asSequence() + val methods = quests + .asSequence() .filter { when (it.id) { // The following quests are left out because their enemies don't drop diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/QuestEditor.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/QuestEditor.kt index 9419dcc6..1bfc1c98 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/QuestEditor.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/QuestEditor.kt @@ -2,10 +2,13 @@ package world.phantasmal.web.questEditor import kotlinx.coroutines.CoroutineScope import org.w3c.dom.HTMLCanvasElement -import world.phantasmal.web.core.stores.UiStore -import world.phantasmal.web.externals.Engine +import world.phantasmal.web.core.loading.AssetLoader +import world.phantasmal.web.externals.babylon.Engine import world.phantasmal.web.questEditor.controllers.QuestEditorToolbarController import world.phantasmal.web.questEditor.controllers.QuestInfoController +import world.phantasmal.web.questEditor.loading.EntityAssetLoader +import world.phantasmal.web.questEditor.rendering.QuestEditorMeshManager +import world.phantasmal.web.questEditor.rendering.QuestRenderer import world.phantasmal.web.questEditor.stores.QuestEditorStore import world.phantasmal.web.questEditor.widgets.QuestEditorRendererWidget import world.phantasmal.web.questEditor.widgets.QuestEditorToolbar @@ -16,11 +19,13 @@ import world.phantasmal.webui.widgets.Widget class QuestEditor( private val scope: CoroutineScope, - uiStore: UiStore, + private val assetLoader: AssetLoader, private val createEngine: (HTMLCanvasElement) -> Engine, ) : DisposableContainer() { + // Stores private val questEditorStore = addDisposable(QuestEditorStore(scope)) + // Controllers private val toolbarController = addDisposable(QuestEditorToolbarController(scope, questEditorStore)) private val questInfoController = addDisposable(QuestInfoController(scope, questEditorStore)) @@ -30,6 +35,18 @@ class QuestEditor( scope, QuestEditorToolbar(scope, toolbarController), { scope -> QuestInfoWidget(scope, questInfoController) }, - { scope -> QuestEditorRendererWidget(scope, createEngine) } + { scope -> QuestEditorRendererWidget(scope, ::createQuestEditorRenderer) } ) + + private fun createQuestEditorRenderer(canvas: HTMLCanvasElement): QuestRenderer = + QuestRenderer(canvas, createEngine(canvas)) { renderer, scene -> + QuestEditorMeshManager( + scope, + questEditorStore.currentQuest, + questEditorStore.currentArea, + questEditorStore.selectedWave, + renderer, + EntityAssetLoader(scope, assetLoader, scene) + ) + } } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/loading/EntityAssetLoader.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/loading/EntityAssetLoader.kt new file mode 100644 index 00000000..0787f186 --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/loading/EntityAssetLoader.kt @@ -0,0 +1,392 @@ +package world.phantasmal.web.questEditor.loading + +import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.async +import mu.KotlinLogging +import org.khronos.webgl.ArrayBuffer +import world.phantasmal.core.PwResult +import world.phantasmal.core.Success +import world.phantasmal.lib.Endianness +import world.phantasmal.lib.cursor.Cursor +import world.phantasmal.lib.cursor.cursor +import world.phantasmal.lib.fileFormats.ninja.NinjaModel +import world.phantasmal.lib.fileFormats.ninja.NinjaObject +import world.phantasmal.lib.fileFormats.ninja.parseNj +import world.phantasmal.lib.fileFormats.ninja.parseXj +import world.phantasmal.lib.fileFormats.quest.EntityType +import world.phantasmal.lib.fileFormats.quest.NpcType +import world.phantasmal.lib.fileFormats.quest.ObjectType +import world.phantasmal.web.core.loading.AssetLoader +import world.phantasmal.web.core.rendering.conversion.ninjaObjectToVertexData +import world.phantasmal.web.externals.babylon.* +import world.phantasmal.webui.DisposableContainer +import world.phantasmal.webui.obj + +private val logger = KotlinLogging.logger {} + +class EntityAssetLoader( + private val scope: CoroutineScope, + private val assetLoader: AssetLoader, + private val scene: Scene, +) : DisposableContainer() { + private val defaultMesh = MeshBuilder.CreateCylinder("Entity", obj { + diameter = 6.0 + height = 20.0 + }, scene).apply { + setEnabled(false) + position = Vector3(0.0, 10.0, 0.0) + } + + private val meshCache = addDisposable(LoadingCache, Mesh>()) + + suspend fun loadMesh(type: EntityType, model: Int?): Mesh { + return meshCache.getOrPut(Pair(type, model)) { + scope.async { + try { + loadGeometry(type, model)?.let { vertexData -> + // TODO: Remove this check when XJ models are parsed. + if (vertexData.indices == null || vertexData.indices!!.length == 0) { + defaultMesh + } else { + val mesh = Mesh("${type.uniqueName}${model?.let { "-$it" }}", scene) + mesh.setEnabled(false) + vertexData.applyToMesh(mesh) + mesh + } + } ?: defaultMesh + } catch (e: Throwable) { + logger.error(e) { "Couldn't load mesh for $type (model: $model)." } + defaultMesh + } + } + }.await() + } + + private suspend fun loadGeometry(type: EntityType, model: Int?): VertexData? { + val geomFormat = entityTypeToGeometryFormat(type) + + val geomParts = geometryParts(type).mapNotNull { suffix -> + entityTypeToPath(type, AssetType.Geometry, suffix, model, geomFormat)?.let { path -> + val data = assetLoader.loadArrayBuffer(path) + Pair(path, data) + } + } + + return when (geomFormat) { + GeomFormat.Nj -> parseGeometry(type, geomParts, ::parseNj) + GeomFormat.Xj -> parseGeometry(type, geomParts, ::parseXj) + } + } + + private fun parseGeometry( + type: EntityType, + parts: List>, + parse: (Cursor) -> PwResult>>, + ): VertexData? { + val njObjects = parts.flatMap { (path, data) -> + val njObjects = parse(data.cursor(Endianness.Little)) + + if (njObjects is Success && njObjects.value.isNotEmpty()) { + njObjects.value + } else { + logger.warn { "Couldn't parse $path for $type." } + emptyList() + } + } + + if (njObjects.isEmpty()) { + return null + } + + val njObject = njObjects.first() + njObject.evaluationFlags.breakChildTrace = false + + for (njObj in njObjects.drop(1)) { + njObject.addChild(njObj) + } + + return ninjaObjectToVertexData(njObject) + } +} + +private enum class AssetType { + Geometry, Texture +} + +private enum class GeomFormat { + Nj, Xj +} + +/** + * Returns the suffix of each geometry part. + */ +private fun geometryParts(type: EntityType): List = + when (type) { + ObjectType.Teleporter -> listOf("", "-2") + ObjectType.Warp -> listOf("", "-2") + ObjectType.BossTeleporter -> listOf("", "-2") + ObjectType.QuestWarp -> listOf("", "-2") + ObjectType.Epilogue -> listOf("", "-2") + ObjectType.MainRagolTeleporter -> listOf("", "-2") + ObjectType.PrincipalWarp -> listOf("", "-2") + ObjectType.TeleporterDoor -> listOf("", "-2") + ObjectType.EasterEgg -> listOf("", "-2") + ObjectType.ValentinesHeart -> listOf("", "-2", "-3") + ObjectType.ChristmasTree -> listOf("", "-2", "-3", "-4") + ObjectType.TwentyFirstCentury -> listOf("", "-2") + ObjectType.WelcomeBoard -> listOf("") // TODO: position part 2 correctly. + ObjectType.ForestDoor -> listOf("", "-2", "-3", "-4", "-5") + ObjectType.ForestSwitch -> listOf("", "-2", "-3") + ObjectType.LaserFence -> listOf("", "-2") + ObjectType.LaserSquareFence -> listOf("", "-2") + ObjectType.ForestLaserFenceSwitch -> listOf("", "-2", "-3") + ObjectType.Probe -> listOf("-0") // TODO: use correct part. + ObjectType.RandomTypeBox1 -> listOf("-2") // What are the other two parts for? + ObjectType.BlackSlidingDoor -> listOf("", "-2") + ObjectType.EnergyBarrier -> listOf("", "-2") + ObjectType.SwitchNoneDoor -> listOf("", "-2") + ObjectType.EnemyBoxGrey -> listOf("-2") // What are the other two parts for? + ObjectType.FixedTypeBox -> listOf("-3") // What are the other three parts for? + ObjectType.EnemyBoxBrown -> listOf("-3") // What are the other three parts for? + ObjectType.LaserFenceEx -> listOf("", "-2") + ObjectType.LaserSquareFenceEx -> listOf("", "-2") + ObjectType.CavesSmashingPillar -> listOf("", "-3") // What's part 2 for? + ObjectType.RobotRechargeStation -> listOf("", "-2") + ObjectType.RuinsTeleporter -> listOf("", "-2", "-3", "-4") + ObjectType.RuinsWarpSiteToSite -> listOf("", "-2") + ObjectType.RuinsSwitch -> listOf("", "-2") + ObjectType.RuinsPillarTrap -> listOf("", "-2", "-3", "-4") + ObjectType.RuinsCrystal -> listOf("", "-2", "-3") + ObjectType.FloatingRocks -> listOf("-0") + ObjectType.ItemBoxCca -> listOf("", "-3") // What are the other two parts for? + ObjectType.TeleporterEp2 -> listOf("", "-2") + ObjectType.CcaDoor -> listOf("", "-2") + ObjectType.SpecialBoxCca -> listOf("", "-4") // What are the other two parts for? + ObjectType.BigCcaDoor -> listOf("", "-2", "-3", "-4") + ObjectType.BigCcaDoorSwitch -> listOf("", "-2") + ObjectType.LaserDetect -> listOf("", "-2") // TODO: use correct part. + ObjectType.LabCeilingWarp -> listOf("", "-2") + ObjectType.BigBrownRock -> listOf("-0") // TODO: use correct part. + ObjectType.BigBlackRocks -> listOf("") + ObjectType.BeeHive -> listOf("", "-0", "-1") + else -> listOf(null) + } + +private fun entityTypeToGeometryFormat(type: EntityType): GeomFormat = + when (type) { + is NpcType -> { + when (type) { + NpcType.Dubswitch -> GeomFormat.Xj + else -> GeomFormat.Nj + } + } + is ObjectType -> { + when (type) { + ObjectType.EasterEgg, + ObjectType.ChristmasTree, + ObjectType.ChristmasWreath, + ObjectType.TwentyFirstCentury, + ObjectType.Sonic, + ObjectType.WelcomeBoard, + ObjectType.FloatingJellyfish, + ObjectType.RuinsSeal, + ObjectType.Dolphin, + ObjectType.Cactus, + ObjectType.BigBrownRock, + ObjectType.PoisonPlant, + ObjectType.BigBlackRocks, + ObjectType.FallingRock, + ObjectType.DesertFixedTypeBoxBreakableCrystals, + ObjectType.BeeHive, + -> GeomFormat.Nj + + else -> GeomFormat.Xj + } + } + else -> { + error("$type not supported.") + } + } + +private fun entityTypeToPath( + type: EntityType, + assetType: AssetType, + suffix: String?, + model: Int?, + geomFormat: GeomFormat, +): String? { + val fullSuffix = when { + suffix != null -> suffix + model != null -> "-$model" + else -> "" + } + + val extension = when (assetType) { + AssetType.Geometry -> when (geomFormat) { + GeomFormat.Nj -> "nj" + GeomFormat.Xj -> "xj" + } + AssetType.Texture -> "xvm" + } + + return when (type) { + is NpcType -> { + when (type) { + // We don't have a model for these NPCs. + NpcType.Unknown, + NpcType.Migium, + NpcType.Hidoom, + NpcType.VolOptPart1, + NpcType.DeathGunner, + NpcType.StRappy, + NpcType.HalloRappy, + NpcType.EggRappy, + NpcType.Migium2, + NpcType.Hidoom2, + NpcType.Recon, + -> null + + // Episode II VR Temple + + NpcType.Hildebear2 -> + entityTypeToPath(NpcType.Hildebear, assetType, suffix, model, geomFormat) + NpcType.Hildeblue2 -> + entityTypeToPath(NpcType.Hildeblue, assetType, suffix, model, geomFormat) + NpcType.RagRappy2 -> + entityTypeToPath(NpcType.RagRappy, assetType, suffix, model, geomFormat) + NpcType.Monest2 -> + entityTypeToPath(NpcType.Monest, assetType, suffix, model, geomFormat) + NpcType.Mothmant2 -> + entityTypeToPath(NpcType.Mothmant, assetType, suffix, model, geomFormat) + NpcType.PoisonLily2 -> + entityTypeToPath(NpcType.PoisonLily, assetType, suffix, model, geomFormat) + NpcType.NarLily2 -> + entityTypeToPath(NpcType.NarLily, assetType, suffix, model, geomFormat) + NpcType.GrassAssassin2 -> + entityTypeToPath(NpcType.GrassAssassin, assetType, suffix, model, geomFormat) + NpcType.Dimenian2 -> + entityTypeToPath(NpcType.Dimenian, assetType, suffix, model, geomFormat) + NpcType.LaDimenian2 -> + entityTypeToPath(NpcType.LaDimenian, assetType, suffix, model, geomFormat) + NpcType.SoDimenian2 -> + entityTypeToPath(NpcType.SoDimenian, assetType, suffix, model, geomFormat) + NpcType.DarkBelra2 -> + entityTypeToPath(NpcType.DarkBelra, assetType, suffix, model, geomFormat) + + // Episode II VR Spaceship + + NpcType.SavageWolf2 -> + entityTypeToPath(NpcType.SavageWolf, assetType, suffix, model, geomFormat) + NpcType.BarbarousWolf2 -> + entityTypeToPath(NpcType.BarbarousWolf, assetType, suffix, model, geomFormat) + NpcType.PanArms2 -> + entityTypeToPath(NpcType.PanArms, assetType, suffix, model, geomFormat) + NpcType.Dubchic2 -> + entityTypeToPath(NpcType.Dubchic, assetType, suffix, model, geomFormat) + NpcType.Gilchic2 -> + entityTypeToPath(NpcType.Gilchic, assetType, suffix, model, geomFormat) + NpcType.Garanz2 -> + entityTypeToPath(NpcType.Garanz, assetType, suffix, model, geomFormat) + NpcType.Dubswitch2 -> + entityTypeToPath(NpcType.Dubswitch, assetType, suffix, model, geomFormat) + NpcType.Delsaber2 -> + entityTypeToPath(NpcType.Delsaber, assetType, suffix, model, geomFormat) + NpcType.ChaosSorcerer2 -> + entityTypeToPath(NpcType.ChaosSorcerer, assetType, suffix, model, geomFormat) + + else -> "/npcs/${type.name}${fullSuffix}.$extension" + } + } + is ObjectType -> { + when (type) { + // We don't have a model for these objects. + ObjectType.Unknown, + ObjectType.PlayerSet, + ObjectType.Particle, + ObjectType.LightCollision, + ObjectType.EnvSound, + ObjectType.FogCollision, + ObjectType.EventCollision, + ObjectType.CharaCollision, + ObjectType.ObjRoomID, + ObjectType.LensFlare, + ObjectType.ScriptCollision, + ObjectType.MapCollision, + ObjectType.ScriptCollisionA, + ObjectType.ItemLight, + ObjectType.RadarCollision, + ObjectType.FogCollisionSW, + ObjectType.ImageBoard, + ObjectType.UnknownItem29, + ObjectType.UnknownItem30, + ObjectType.UnknownItem31, + ObjectType.MenuActivation, + ObjectType.BoxDetectObject, + ObjectType.SymbolChatObject, + ObjectType.TouchPlateObject, + ObjectType.TargetableObject, + ObjectType.EffectObject, + ObjectType.CountDownObject, + ObjectType.UnknownItem38, + ObjectType.UnknownItem39, + ObjectType.UnknownItem40, + ObjectType.UnknownItem41, + ObjectType.TelepipeLocation, + ObjectType.BGMCollision, + ObjectType.Pioneer2InvisibleTouchplate, + ObjectType.TempleMapDetect, + ObjectType.Firework, + ObjectType.MainRagolTeleporterBattleInNextArea, + ObjectType.Rainbow, + ObjectType.FloatingBlueLight, + ObjectType.PopupTrapNoTech, + ObjectType.Poison, + ObjectType.EnemyTypeBoxYellow, + ObjectType.EnemyTypeBoxBlue, + ObjectType.EmptyTypeBoxBlue, + ObjectType.FloatingSoul, + ObjectType.Butterfly, + ObjectType.UnknownItem400, + ObjectType.CcaAreaTeleporter, + ObjectType.UnknownItem523, + ObjectType.WhiteBird, + ObjectType.OrangeBird, + ObjectType.UnknownItem529, + ObjectType.UnknownItem530, + ObjectType.Seagull, + ObjectType.UnknownItem576, + ObjectType.WarpInBarbaRayRoom, + ObjectType.UnknownItem672, + ObjectType.InstaWarp, + ObjectType.LabInvisibleObject, + ObjectType.UnknownItem700, + ObjectType.Ep4LightSource, + ObjectType.BreakableBrownRock, + ObjectType.UnknownItem897, + ObjectType.UnknownItem898, + ObjectType.OozingDesertPlant, + ObjectType.UnknownItem901, + ObjectType.UnknownItem903, + ObjectType.UnknownItem904, + ObjectType.UnknownItem905, + ObjectType.UnknownItem906, + ObjectType.DesertPlantHasCollision, + ObjectType.UnknownItem910, + ObjectType.UnknownItem912, + ObjectType.Heat, + ObjectType.TopOfSaintMillionEgg, + ObjectType.UnknownItem961, + -> null + + else -> { + type.typeId?.let { typeId -> + "/objects/${typeId}${fullSuffix}.$extension" + } + } + } + } + else -> { + error("$type not supported.") + } + } +} diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/loading/LoadingCache.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/loading/LoadingCache.kt new file mode 100644 index 00000000..29bb2847 --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/loading/LoadingCache.kt @@ -0,0 +1,20 @@ +package world.phantasmal.web.questEditor.loading + +import kotlinx.coroutines.Deferred +import world.phantasmal.core.disposable.TrackedDisposable + +class LoadingCache : TrackedDisposable() { + private val map = mutableMapOf>() + + operator fun set(key: K, value: Deferred) { + map[key] = value + } + + fun getOrPut(key: K, defaultValue: () -> Deferred): Deferred = + map.getOrPut(key, defaultValue) + + override fun internalDispose() { + map.values.forEach { it.cancel() } + super.internalDispose() + } +} diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/models/AreaModel.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/AreaModel.kt new file mode 100644 index 00000000..02e8123b --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/AreaModel.kt @@ -0,0 +1,3 @@ +package world.phantasmal.web.questEditor.models + +class AreaModel diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/models/AreaVariantModel.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/AreaVariantModel.kt new file mode 100644 index 00000000..a690dd29 --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/AreaVariantModel.kt @@ -0,0 +1,3 @@ +package world.phantasmal.web.questEditor.models + +class AreaVariantModel diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestEntityModel.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestEntityModel.kt new file mode 100644 index 00000000..fb7f8230 --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestEntityModel.kt @@ -0,0 +1,27 @@ +package world.phantasmal.web.questEditor.models + +import world.phantasmal.lib.fileFormats.quest.EntityType +import world.phantasmal.lib.fileFormats.quest.QuestEntity +import world.phantasmal.observable.value.Val +import world.phantasmal.observable.value.mutableVal +import world.phantasmal.web.core.rendering.conversion.vec3ToBabylon +import world.phantasmal.web.externals.babylon.Vector3 + +abstract class QuestEntityModel>( + private val entity: Entity, +) { + private val _position = mutableVal(vec3ToBabylon(entity.position)) + private val _worldPosition = mutableVal(_position.value) + + val type: Type get() = entity.type + + /** + * Section-relative position + */ + val position: Val = _position + + /** + * World position + */ + val worldPosition: Val = _worldPosition +} diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestModel.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestModel.kt index 1d422b7e..a25638fc 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestModel.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestModel.kt @@ -2,6 +2,8 @@ package world.phantasmal.web.questEditor.models import world.phantasmal.lib.fileFormats.quest.Episode import world.phantasmal.observable.value.Val +import world.phantasmal.observable.value.list.ListVal +import world.phantasmal.observable.value.list.mutableListVal import world.phantasmal.observable.value.mutableVal class QuestModel( @@ -11,18 +13,24 @@ class QuestModel( shortDescription: String, longDescription: String, val episode: Episode, + npcs: MutableList, + objects: MutableList, ) { private val _id = mutableVal(0) private val _language = mutableVal(0) private val _name = mutableVal("") private val _shortDescription = mutableVal("") private val _longDescription = mutableVal("") + private val _npcs = mutableListVal(npcs) + private val _objects = mutableListVal(objects) val id: Val = _id val language: Val = _language val name: Val = _name val shortDescription: Val = _shortDescription val longDescription: Val = _longDescription + val npcs: ListVal = _npcs + val objects: ListVal = _objects init { setId(id) diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestNpcModel.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestNpcModel.kt new file mode 100644 index 00000000..b9d8e528 --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestNpcModel.kt @@ -0,0 +1,12 @@ +package world.phantasmal.web.questEditor.models + +import world.phantasmal.lib.fileFormats.quest.NpcType +import world.phantasmal.lib.fileFormats.quest.QuestNpc +import world.phantasmal.observable.value.Val +import world.phantasmal.observable.value.mutableVal + +class QuestNpcModel(npc: QuestNpc, wave: WaveModel?) : QuestEntityModel(npc) { + private val _wave = mutableVal(wave) + + val wave: Val = _wave +} diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestObjectModel.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestObjectModel.kt new file mode 100644 index 00000000..2c5d8caa --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestObjectModel.kt @@ -0,0 +1,6 @@ +package world.phantasmal.web.questEditor.models + +import world.phantasmal.lib.fileFormats.quest.ObjectType +import world.phantasmal.lib.fileFormats.quest.QuestObject + +class QuestObjectModel(obj: QuestObject) : QuestEntityModel(obj) diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/models/WaveModel.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/WaveModel.kt new file mode 100644 index 00000000..63cfd407 --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/WaveModel.kt @@ -0,0 +1,3 @@ +package world.phantasmal.web.questEditor.models + +class WaveModel diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/EntityMeshManager.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/EntityMeshManager.kt new file mode 100644 index 00000000..480699ee --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/EntityMeshManager.kt @@ -0,0 +1,136 @@ +package world.phantasmal.web.questEditor.rendering + +import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.launch +import mu.KotlinLogging +import world.phantasmal.core.disposable.Disposer +import world.phantasmal.core.disposable.TrackedDisposable +import world.phantasmal.observable.value.Val +import world.phantasmal.web.externals.babylon.AbstractMesh +import world.phantasmal.web.questEditor.loading.EntityAssetLoader +import world.phantasmal.web.questEditor.models.QuestEntityModel +import world.phantasmal.web.questEditor.models.QuestNpcModel +import world.phantasmal.web.questEditor.models.WaveModel +import world.phantasmal.web.questEditor.rendering.conversion.EntityMetadata + +private val logger = KotlinLogging.logger {} + +private class LoadedEntity(val entity: QuestEntityModel<*, *>, val disposer: Disposer) + +class EntityMeshManager( + private val scope: CoroutineScope, + private val selectedWave: Val, + private val renderer: QuestRenderer, + private val entityAssetLoader: EntityAssetLoader, +) : TrackedDisposable() { + private val queue: MutableList> = mutableListOf() + private val loadedEntities: MutableList = mutableListOf() + private var loading = false + + override fun internalDispose() { + removeAll() + super.internalDispose() + } + + fun add(entities: List>) { + queue.addAll(entities) + + if (!loading) { + scope.launch { + try { + loading = true + + while (queue.isNotEmpty()) { + val entity = queue.first() + + try { + load(entity) + } catch (e: Error) { + logger.error(e) { + "Couldn't load model for entity of type ${entity.type}." + } + queue.remove(entity) + } + } + } finally { + loading = false + } + } + } + } + + fun remove(entities: List>) { + for (entity in entities) { + queue.remove(entity) + + val loadedIndex = loadedEntities.indexOfFirst { it.entity == entity } + + if (loadedIndex != -1) { + val loaded = loadedEntities.removeAt(loadedIndex) + + renderer.removeEntityMesh(loaded.entity) + loaded.disposer.dispose() + } + } + } + + fun removeAll() { + for (loaded in loadedEntities) { + loaded.disposer.dispose() + } + + loadedEntities.clear() + queue.clear() + } + + private suspend fun load(entity: QuestEntityModel<*, *>) { + // TODO + val mesh = entityAssetLoader.loadMesh(entity.type, model = null) + + // Only add an instance of this mesh if the entity is still in the queue at this point. + if (queue.remove(entity)) { + val instance = mesh.createInstance(entity.type.uniqueName) + instance.metadata = EntityMetadata(entity) + instance.position = entity.worldPosition.value + updateEntityMesh(entity, instance) + } + } + + private fun updateEntityMesh(entity: QuestEntityModel<*, *>, mesh: AbstractMesh) { + renderer.addEntityMesh(mesh) + + val disposer = Disposer( + entity.worldPosition.observe { (pos) -> + mesh.position = pos + renderer.scheduleRender() + }, + + // TODO: Rotation. +// entity.worldRotation.observe { (value) -> +// mesh.rotation.copy(value) +// renderer.schedule_render() +// }, + + // TODO: Model. +// entity.model.observe { +// remove(listOf(entity)) +// add(listOf(entity)) +// }, + ) + + if (entity is QuestNpcModel) { + disposer.add( + selectedWave + .map(entity.wave) { selectedWave, entityWave -> + selectedWave == null || selectedWave == entityWave + } + .observe(callNow = true) { (visible) -> + mesh.setEnabled(visible) + renderer.scheduleRender() + }, + ) + } + + loadedEntities.add(LoadedEntity(entity, disposer)) + } +} diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestEditorMeshManager.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestEditorMeshManager.kt new file mode 100644 index 00000000..e9209f41 --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestEditorMeshManager.kt @@ -0,0 +1,46 @@ +package world.phantasmal.web.questEditor.rendering + +import kotlinx.coroutines.CoroutineScope +import world.phantasmal.observable.value.Val +import world.phantasmal.observable.value.list.ListVal +import world.phantasmal.observable.value.list.listVal +import world.phantasmal.web.questEditor.loading.EntityAssetLoader +import world.phantasmal.web.questEditor.models.* + +class QuestEditorMeshManager( + scope: CoroutineScope, + private val currentQuest: Val, + private val currentArea: Val, + selectedWave: Val, + renderer: QuestRenderer, + entityAssetLoader: EntityAssetLoader, +) : QuestMeshManager(scope, selectedWave, renderer, entityAssetLoader) { + init { + disposer.addAll( + currentQuest.observe { areaVariantChanged() }, + currentArea.observe { areaVariantChanged() }, + ) + } + + override fun getAreaVariantDetails(): AreaVariantDetails { + val quest = currentQuest.value + val area = currentArea.value + + val areaVariant: AreaVariantModel? + val npcs: ListVal + val objects: ListVal + + if (quest != null /*&& area != null*/) { + // TODO: Set areaVariant. + areaVariant = null + npcs = quest.npcs // TODO: Filter NPCs. + objects = listVal() // TODO: Filter objects. + } else { + areaVariant = null + npcs = listVal() + objects = listVal() + } + + return AreaVariantDetails(quest?.episode, areaVariant, npcs, objects) + } +} diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestMeshManager.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestMeshManager.kt new file mode 100644 index 00000000..7d630f5d --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestMeshManager.kt @@ -0,0 +1,78 @@ +package world.phantasmal.web.questEditor.rendering + +import kotlinx.coroutines.CoroutineScope +import world.phantasmal.core.disposable.Disposer +import world.phantasmal.core.disposable.TrackedDisposable +import world.phantasmal.lib.fileFormats.quest.Episode +import world.phantasmal.observable.value.Val +import world.phantasmal.observable.value.list.ListVal +import world.phantasmal.observable.value.list.ListValChangeEvent +import world.phantasmal.web.questEditor.loading.EntityAssetLoader +import world.phantasmal.web.questEditor.models.AreaVariantModel +import world.phantasmal.web.questEditor.models.QuestNpcModel +import world.phantasmal.web.questEditor.models.QuestObjectModel +import world.phantasmal.web.questEditor.models.WaveModel + +/** + * Loads the necessary area and entity 3D models into [QuestRenderer]. + */ +abstract class QuestMeshManager protected constructor( + scope: CoroutineScope, + selectedWave: Val, + private val renderer: QuestRenderer, + entityAssetLoader: EntityAssetLoader, +) : TrackedDisposable() { + protected val disposer = Disposer() + + private val areaDisposer = disposer.add(Disposer()) + private val npcMeshManager = disposer.add( + EntityMeshManager(scope, selectedWave, renderer, entityAssetLoader) + ) + private val objectMeshManager = disposer.add( + EntityMeshManager(scope, selectedWave, renderer, entityAssetLoader) + ) + + protected abstract fun getAreaVariantDetails(): AreaVariantDetails + + protected fun areaVariantChanged() { + val details = getAreaVariantDetails() + + // TODO: Load area mesh. + + areaDisposer.disposeAll() + npcMeshManager.removeAll() + renderer.resetEntityMeshes() + + // Load entity meshes. + areaDisposer.addAll( + details.npcs.observeList(callNow = true, ::npcsChanged), + details.objects.observeList(callNow = true, ::objectsChanged), + ) + } + + override fun internalDispose() { + disposer.dispose() + super.internalDispose() + } + + private fun npcsChanged(change: ListValChangeEvent) { + if (change is ListValChangeEvent.Change) { + npcMeshManager.remove(change.removed) + npcMeshManager.add(change.inserted) + } + } + + private fun objectsChanged(change: ListValChangeEvent) { + if (change is ListValChangeEvent.Change) { + objectMeshManager.remove(change.removed) + objectMeshManager.add(change.inserted) + } + } +} + +class AreaVariantDetails( + val episode: Episode?, + val areaVariant: AreaVariantModel?, + val npcs: ListVal, + val objects: ListVal, +) diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestRenderer.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestRenderer.kt index 3799b455..bfddd63f 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestRenderer.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestRenderer.kt @@ -1,28 +1,83 @@ package world.phantasmal.web.questEditor.rendering import org.w3c.dom.HTMLCanvasElement -import world.phantasmal.webui.newJsObject import world.phantasmal.web.core.rendering.Renderer -import world.phantasmal.web.externals.* +import world.phantasmal.web.externals.babylon.* +import world.phantasmal.web.questEditor.models.QuestEntityModel +import world.phantasmal.web.questEditor.rendering.conversion.EntityMetadata import kotlin.math.PI class QuestRenderer( canvas: HTMLCanvasElement, - createEngine: (HTMLCanvasElement) -> Engine, -) : Renderer(canvas, createEngine) { - private val camera = ArcRotateCamera("Camera", PI / 2, PI / 2, 2.0, Vector3.Zero(), scene) + engine: Engine, + createMeshManager: (QuestRenderer, Scene) -> QuestMeshManager, +) : Renderer(canvas, engine) { + private val meshManager = createMeshManager(this, scene) + private var entityMeshes = TransformNode("Entities", scene) + private val entityToMesh = mutableMapOf, AbstractMesh>() + private val camera = ArcRotateCamera("Camera", 0.0, PI / 6, 500.0, Vector3.Zero(), scene) private val light = HemisphericLight("Light", Vector3(1.0, 1.0, 0.0), scene) - private val cylinder = - MeshBuilder.CreateCylinder("Cylinder", newJsObject { diameter = 1.0 }, scene) init { - camera.attachControl(canvas, noPreventDefault = true) + with(camera) { + attachControl( + canvas, + noPreventDefault = false, + useCtrlForPanning = false, + panningMouseButton = 0 + ) + inertia = 0.0 + angularSensibilityX = 200.0 + angularSensibilityY = 200.0 + panningInertia = 0.0 + panningSensibility = 3.0 + panningAxis = Vector3(1.0, 0.0, 1.0) + pinchDeltaPercentage = 0.1 + wheelDeltaPercentage = 0.1 + } } override fun internalDispose() { + meshManager.dispose() + entityMeshes.dispose() + entityToMesh.clear() camera.dispose() light.dispose() - cylinder.dispose() super.internalDispose() } + + fun resetEntityMeshes() { + entityMeshes.dispose(false) + entityToMesh.clear() + + entityMeshes = TransformNode("Entities", scene) + scheduleRender() + } + + fun addEntityMesh(mesh: AbstractMesh) { + val entity = (mesh.metadata as EntityMetadata).entity + mesh.parent = entityMeshes + + entityToMesh[entity]?.let { prevMesh -> + prevMesh.parent = null + prevMesh.dispose() + } + + entityToMesh[entity] = mesh + + // TODO: Mark selected entity. +// if (entity === this.selected_entity) { +// this.mark_selected(model) +// } + + this.scheduleRender() + } + + fun removeEntityMesh(entity: QuestEntityModel<*, *>) { + entityToMesh.remove(entity)?.let { mesh -> + mesh.parent = null + mesh.dispose() + this.scheduleRender() + } + } } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/conversion/Entities.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/conversion/Entities.kt new file mode 100644 index 00000000..d813fe2f --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/conversion/Entities.kt @@ -0,0 +1,5 @@ +package world.phantasmal.web.questEditor.rendering.conversion + +import world.phantasmal.web.questEditor.models.QuestEntityModel + +class EntityMetadata(val entity: QuestEntityModel<*, *>) diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/ModelConversion.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/ModelConversion.kt index b5871e55..5279b20f 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/ModelConversion.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/ModelConversion.kt @@ -2,6 +2,8 @@ package world.phantasmal.web.questEditor.stores import world.phantasmal.lib.fileFormats.quest.Quest import world.phantasmal.web.questEditor.models.QuestModel +import world.phantasmal.web.questEditor.models.QuestNpcModel +import world.phantasmal.web.questEditor.models.QuestObjectModel fun convertQuestToModel(quest: Quest): QuestModel { return QuestModel( @@ -11,5 +13,8 @@ fun convertQuestToModel(quest: Quest): QuestModel { quest.shortDescription, quest.longDescription, quest.episode, + // TODO: Add WaveModel to QuestNpcModel + quest.npcs.mapTo(mutableListOf()) { QuestNpcModel(it, null) }, + quest.objects.mapTo(mutableListOf()) { QuestObjectModel(it) } ) } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/QuestEditorStore.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/QuestEditorStore.kt index 15688918..3be789ac 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/QuestEditorStore.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/QuestEditorStore.kt @@ -3,13 +3,19 @@ package world.phantasmal.web.questEditor.stores import kotlinx.coroutines.CoroutineScope import world.phantasmal.observable.value.Val import world.phantasmal.observable.value.mutableVal +import world.phantasmal.web.questEditor.models.AreaModel import world.phantasmal.web.questEditor.models.QuestModel +import world.phantasmal.web.questEditor.models.WaveModel import world.phantasmal.webui.stores.Store class QuestEditorStore(scope: CoroutineScope) : Store(scope) { private val _currentQuest = mutableVal(null) + private val _currentArea = mutableVal(null) + private val _selectedWave = mutableVal(null) val currentQuest: Val = _currentQuest + val currentArea: Val = _currentArea + val selectedWave: Val = _selectedWave // TODO: Take into account whether we're debugging or not. val questEditingDisabled: Val = currentQuest.map { it == null } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestEditorRendererWidget.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestEditorRendererWidget.kt index 7b3b60f2..9482da51 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestEditorRendererWidget.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestEditorRendererWidget.kt @@ -2,10 +2,9 @@ package world.phantasmal.web.questEditor.widgets import kotlinx.coroutines.CoroutineScope import org.w3c.dom.HTMLCanvasElement -import world.phantasmal.web.externals.Engine +import world.phantasmal.web.questEditor.rendering.QuestRenderer class QuestEditorRendererWidget( scope: CoroutineScope, - createEngine: (HTMLCanvasElement) -> Engine, -) : QuestRendererWidget(scope, createEngine) { -} + createRenderer: (HTMLCanvasElement) -> QuestRenderer, +) : QuestRendererWidget(scope, createRenderer) diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestEditorToolbar.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestEditorToolbar.kt index 11f106af..0e6f9192 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestEditorToolbar.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestEditorToolbar.kt @@ -3,6 +3,7 @@ package world.phantasmal.web.questEditor.widgets import kotlinx.coroutines.CoroutineScope import org.w3c.dom.Node import world.phantasmal.web.questEditor.controllers.QuestEditorToolbarController +import world.phantasmal.webui.dom.Icon import world.phantasmal.webui.dom.div import world.phantasmal.webui.widgets.FileButton import world.phantasmal.webui.widgets.Toolbar @@ -22,6 +23,7 @@ class QuestEditorToolbar( FileButton( scope, text = "Open file...", + iconLeft = Icon.File, accept = ".bin, .dat, .qst", multiple = true, filesSelected = ctrl::openFiles diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestEditorWidget.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestEditorWidget.kt index 732d06b7..24117fe0 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestEditorWidget.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestEditorWidget.kt @@ -17,7 +17,7 @@ private class TestWidget(scope: CoroutineScope) : Widget(scope) { } } -open class QuestEditorWidget( +class QuestEditorWidget( scope: CoroutineScope, private val toolbar: Widget, private val createQuestInfoWidget: (CoroutineScope) -> Widget, diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestRendererWidget.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestRendererWidget.kt index d5079a2d..036fbbff 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestRendererWidget.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestRendererWidget.kt @@ -4,19 +4,20 @@ import kotlinx.coroutines.CoroutineScope import org.w3c.dom.HTMLCanvasElement import org.w3c.dom.Node import world.phantasmal.web.core.widgets.RendererWidget -import world.phantasmal.web.externals.Engine +import world.phantasmal.web.questEditor.rendering.QuestRenderer import world.phantasmal.webui.dom.div import world.phantasmal.webui.widgets.Widget abstract class QuestRendererWidget( scope: CoroutineScope, - private val createEngine: (HTMLCanvasElement) -> Engine, + private val createRenderer: (HTMLCanvasElement) -> QuestRenderer, ) : Widget(scope) { override fun Node.createElement() = div { className = "pw-quest-editor-quest-renderer" + tabIndex = -1 - addChild(RendererWidget(scope, createEngine)) + addChild(RendererWidget(scope, createRenderer)) } companion object { diff --git a/web/src/test/kotlin/world/phantasmal/web/application/ApplicationTests.kt b/web/src/test/kotlin/world/phantasmal/web/application/ApplicationTests.kt index 3f9abafd..bad7ca7e 100644 --- a/web/src/test/kotlin/world/phantasmal/web/application/ApplicationTests.kt +++ b/web/src/test/kotlin/world/phantasmal/web/application/ApplicationTests.kt @@ -9,9 +9,9 @@ import world.phantasmal.core.disposable.Disposer import world.phantasmal.core.disposable.disposable import world.phantasmal.core.disposable.use import world.phantasmal.testUtils.TestSuite -import world.phantasmal.web.core.HttpAssetLoader +import world.phantasmal.web.core.loading.AssetLoader import world.phantasmal.web.core.stores.PwTool -import world.phantasmal.web.externals.Engine +import world.phantasmal.web.externals.babylon.Engine import world.phantasmal.web.test.TestApplicationUrl import kotlin.test.Test @@ -35,7 +35,7 @@ class ApplicationTests : TestSuite() { Application( scope, rootElement = document.body!!, - assetLoader = HttpAssetLoader(httpClient, basePath = ""), + assetLoader = AssetLoader(basePath = "", httpClient), applicationUrl = appUrl, createEngine = { Engine(it) } ) diff --git a/web/src/test/kotlin/world/phantasmal/web/huntOptimizer/HuntOptimizerTests.kt b/web/src/test/kotlin/world/phantasmal/web/huntOptimizer/HuntOptimizerTests.kt index db4894a0..831c6fde 100644 --- a/web/src/test/kotlin/world/phantasmal/web/huntOptimizer/HuntOptimizerTests.kt +++ b/web/src/test/kotlin/world/phantasmal/web/huntOptimizer/HuntOptimizerTests.kt @@ -6,7 +6,7 @@ import io.ktor.client.features.json.serializer.* import kotlinx.coroutines.cancel import world.phantasmal.core.disposable.disposable import world.phantasmal.testUtils.TestSuite -import world.phantasmal.web.core.HttpAssetLoader +import world.phantasmal.web.core.loading.AssetLoader import world.phantasmal.web.core.stores.PwTool import world.phantasmal.web.core.stores.UiStore import world.phantasmal.web.test.TestApplicationUrl @@ -29,7 +29,7 @@ class HuntOptimizerTests : TestSuite() { disposer.add( HuntOptimizer( scope, - assetLoader = HttpAssetLoader(httpClient, basePath = ""), + AssetLoader(basePath = "", httpClient), uiStore ) ) diff --git a/web/src/test/kotlin/world/phantasmal/web/questEditor/QuestEditorTests.kt b/web/src/test/kotlin/world/phantasmal/web/questEditor/QuestEditorTests.kt index 0e8cf40f..b190184e 100644 --- a/web/src/test/kotlin/world/phantasmal/web/questEditor/QuestEditorTests.kt +++ b/web/src/test/kotlin/world/phantasmal/web/questEditor/QuestEditorTests.kt @@ -6,10 +6,8 @@ import io.ktor.client.features.json.serializer.* import kotlinx.coroutines.cancel import world.phantasmal.core.disposable.disposable import world.phantasmal.testUtils.TestSuite -import world.phantasmal.web.core.stores.PwTool -import world.phantasmal.web.core.stores.UiStore -import world.phantasmal.web.externals.Engine -import world.phantasmal.web.test.TestApplicationUrl +import world.phantasmal.web.core.loading.AssetLoader +import world.phantasmal.web.externals.babylon.Engine import kotlin.test.Test class QuestEditorTests : TestSuite() { @@ -24,12 +22,10 @@ class QuestEditorTests : TestSuite() { } disposer.add(disposable { httpClient.cancel() }) - val uiStore = disposer.add(UiStore(scope, TestApplicationUrl("/${PwTool.QuestEditor}"))) - disposer.add( QuestEditor( scope, - uiStore, + AssetLoader(basePath = "", httpClient), createEngine = { Engine(it) } ) ) diff --git a/webui/build.gradle.kts b/webui/build.gradle.kts index 8b166ce0..7d631a7b 100644 --- a/webui/build.gradle.kts +++ b/webui/build.gradle.kts @@ -14,6 +14,7 @@ dependencies { api(project(":core")) api(project(":observable")) api("org.jetbrains.kotlinx:kotlinx-coroutines-core:$coroutinesVersion") + implementation(npm("@fortawesome/fontawesome-free", "^5.13.1")) testImplementation(kotlin("test-js")) testImplementation(project(":test-utils")) diff --git a/webui/src/main/kotlin/world/phantasmal/webui/Js.kt b/webui/src/main/kotlin/world/phantasmal/webui/Js.kt index 0d37ca5f..9188db64 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/Js.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/Js.kt @@ -1,4 +1,4 @@ package world.phantasmal.webui -fun newJsObject(block: T.() -> Unit): T = +fun obj(block: T.() -> Unit): T = js("{}").unsafeCast().apply(block) diff --git a/webui/src/main/kotlin/world/phantasmal/webui/dom/Dom.kt b/webui/src/main/kotlin/world/phantasmal/webui/dom/Dom.kt index f02555f7..60123138 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/dom/Dom.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/dom/Dom.kt @@ -5,6 +5,7 @@ import kotlinx.dom.appendText import org.w3c.dom.AddEventListenerOptions import org.w3c.dom.HTMLElement import org.w3c.dom.HTMLStyleElement +import org.w3c.dom.Node import org.w3c.dom.events.Event import org.w3c.dom.events.EventTarget import world.phantasmal.core.disposable.Disposable @@ -33,3 +34,51 @@ fun HTMLElement.root(): HTMLElement { id = "pw-root" return this } + +enum class Icon { + ArrowDown, + Eye, + File, + GitHub, + LevelDown, + LevelUp, + LongArrowRight, + NewFile, + Play, + Plus, + Redo, + Remove, + Save, + SquareArrowRight, + Stop, + TriangleDown, + TriangleUp, + Undo, +} + +fun Node.icon(icon: Icon): HTMLElement { + val iconStr = when (icon) { + Icon.ArrowDown -> "fas fa-arrow-down" + Icon.Eye -> "far fa-eye" + Icon.File -> "fas fa-file" + Icon.GitHub -> "fab fa-github" + Icon.LevelDown -> "fas fa-level-down-alt" + Icon.LevelUp -> "fas fa-level-up-alt" + Icon.LongArrowRight -> "fas fa-long-arrow-alt-right" + Icon.NewFile -> "fas fa-file-medical" + Icon.Play -> "fas fa-play" + Icon.Plus -> "fas fa-plus" + Icon.Redo -> "fas fa-redo" + Icon.Remove -> "fas fa-trash-alt" + Icon.Save -> "fas fa-save" + Icon.Stop -> "fas fa-stop" + Icon.SquareArrowRight -> "far fa-caret-square-right" + Icon.TriangleDown -> "fas fa-caret-down" + Icon.TriangleUp -> "fas fa-caret-up" + Icon.Undo -> "fas fa-undo" + } + + // Wrap the span in another span, because Font Awesome will replace the inner element. This way + // the returned element will stay valid. + return span { span { className = iconStr } } +} diff --git a/webui/src/main/kotlin/world/phantasmal/webui/dom/DomCreation.kt b/webui/src/main/kotlin/world/phantasmal/webui/dom/DomCreation.kt index 5716d3c2..97b2eab9 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/dom/DomCreation.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/dom/DomCreation.kt @@ -3,9 +3,6 @@ package world.phantasmal.webui.dom import kotlinx.browser.document import org.w3c.dom.* -fun Node.a(block: HTMLAnchorElement.() -> Unit = {}): HTMLAnchorElement = - appendHtmlEl("A", block) - fun Node.button(block: HTMLButtonElement.() -> Unit = {}): HTMLButtonElement = appendHtmlEl("BUTTON", block) diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Button.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Button.kt index cf68f1bc..4945edf0 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Button.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Button.kt @@ -6,7 +6,9 @@ import org.w3c.dom.events.KeyboardEvent import org.w3c.dom.events.MouseEvent import world.phantasmal.observable.value.Val import world.phantasmal.observable.value.falseVal +import world.phantasmal.webui.dom.Icon import world.phantasmal.webui.dom.button +import world.phantasmal.webui.dom.icon import world.phantasmal.webui.dom.span open class Button( @@ -15,6 +17,8 @@ open class Button( disabled: Val = falseVal(), private val text: String? = null, private val textVal: Val? = null, + private val iconLeft: Icon? = null, + private val iconRight: Icon? = null, private val onMouseDown: ((MouseEvent) -> Unit)? = null, private val onMouseUp: ((MouseEvent) -> Unit)? = null, private val onClick: ((MouseEvent) -> Unit)? = null, @@ -35,6 +39,13 @@ open class Button( span { className = "pw-button-inner" + iconLeft?.let { + span { + className = "pw-button-left" + icon(iconLeft) + } + } + span { className = "pw-button-center" @@ -49,6 +60,13 @@ open class Button( hidden = true } } + + iconRight?.let { + span { + className = "pw-button-right" + icon(iconRight) + } + } } } diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/FileButton.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/FileButton.kt index e1d174dd..ce0127b1 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/FileButton.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/FileButton.kt @@ -5,6 +5,7 @@ import org.w3c.dom.HTMLElement import org.w3c.files.File import world.phantasmal.observable.value.Val import world.phantasmal.observable.value.falseVal +import world.phantasmal.webui.dom.Icon import world.phantasmal.webui.openFiles class FileButton( @@ -13,10 +14,12 @@ class FileButton( disabled: Val = falseVal(), text: String? = null, textVal: Val? = null, + iconLeft: Icon? = null, + iconRight: Icon? = null, private val accept: String = "", private val multiple: Boolean = false, private val filesSelected: ((List) -> Unit)? = null, -) : Button(scope, hidden, disabled, text, textVal) { +) : Button(scope, hidden, disabled, text, textVal, iconLeft, iconRight) { override fun interceptElement(element: HTMLElement) { element.classList.add("pw-file-button") diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Menu.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Menu.kt index b1d80bce..afdf3662 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Menu.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Menu.kt @@ -11,7 +11,7 @@ import world.phantasmal.observable.value.falseVal import world.phantasmal.observable.value.value import world.phantasmal.webui.dom.disposableListener import world.phantasmal.webui.dom.div -import world.phantasmal.webui.newJsObject +import world.phantasmal.webui.obj class Menu( scope: CoroutineScope, @@ -173,7 +173,7 @@ class Menu( highlightedElement?.let { highlightedIndex = index it.classList.add("pw-menu-highlighted") - it.scrollIntoView(newJsObject { block = "nearest" }) + it.scrollIntoView(obj { block = "nearest" }) } } diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Select.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Select.kt index 6472a7c1..c69fa1ac 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Select.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Select.kt @@ -8,6 +8,7 @@ import world.phantasmal.observable.value.Val import world.phantasmal.observable.value.falseVal import world.phantasmal.observable.value.mutableVal import world.phantasmal.observable.value.value +import world.phantasmal.webui.dom.Icon import world.phantasmal.webui.dom.div class Select( @@ -51,6 +52,7 @@ class Select( scope, disabled = disabled, textVal = buttonText, + iconRight = Icon.TriangleDown, onMouseDown = ::onButtonMouseDown, onMouseUp = { onButtonMouseUp() }, onKeyDown = ::onButtonKeyDown, diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Widget.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Widget.kt index cc3925bc..71e93641 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Widget.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Widget.kt @@ -228,6 +228,13 @@ abstract class Widget( el } + init { + js("require('@fortawesome/fontawesome-free/js/fontawesome');") + js("require('@fortawesome/fontawesome-free/js/solid');") + js("require('@fortawesome/fontawesome-free/js/regular');") + js("require('@fortawesome/fontawesome-free/js/brands');") + } + protected fun style(style: String) { STYLE_EL.append(style) }