From db1149ddc04511fc4670fc71207085cf8953d507 Mon Sep 17 00:00:00 2001 From: Daan Vanden Bosch Date: Sun, 8 Nov 2020 16:07:08 +0100 Subject: [PATCH] Entities are now correctly rotated and positioned within their section. --- .../kotlin/world/phantasmal/core/math/Math.kt | 8 + .../lib/fileFormats/AreaGeometry.kt | 94 ++++++++++++ .../phantasmal/lib/fileFormats/ninja/Ninja.kt | 3 + .../lib/fileFormats/quest/QuestEntity.kt | 2 + .../lib/fileFormats/quest/QuestNpc.kt | 6 + .../lib/fileFormats/quest/QuestObject.kt | 6 + .../phantasmal/web/core/BabylonExtensions.kt | 9 ++ .../core/rendering/conversion/Conversion.kt | 2 + .../conversion/NinjaGeometryConversion.kt | 8 +- .../web/externals/babylon/babylon.kt | 5 +- .../QuestEditorToolbarController.kt | 2 +- .../questEditor/loading/AreaAssetLoader.kt | 141 +++++++++++++++++- .../questEditor/models/QuestEntityModel.kt | 91 ++++++++++- .../web/questEditor/models/SectionModel.kt | 4 + .../rendering/EntityMeshManager.kt | 10 +- .../questEditor/rendering/conversion/Areas.kt | 50 ------- .../web/questEditor/stores/AreaStore.kt | 16 +- .../questEditor/stores/QuestEditorStore.kt | 36 ++++- 18 files changed, 418 insertions(+), 75 deletions(-) create mode 100644 core/src/commonMain/kotlin/world/phantasmal/core/math/Math.kt create mode 100644 lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/AreaGeometry.kt delete mode 100644 web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/conversion/Areas.kt diff --git a/core/src/commonMain/kotlin/world/phantasmal/core/math/Math.kt b/core/src/commonMain/kotlin/world/phantasmal/core/math/Math.kt new file mode 100644 index 00000000..8517ee85 --- /dev/null +++ b/core/src/commonMain/kotlin/world/phantasmal/core/math/Math.kt @@ -0,0 +1,8 @@ +package world.phantasmal.core.math + +/** + * Returns the floored modulus of its arguments. The computed value will have the same sign as the + * [divisor]. + */ +fun floorMod(dividend: Double, divisor: Double): Double = + ((dividend % divisor) + divisor) % divisor diff --git a/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/AreaGeometry.kt b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/AreaGeometry.kt new file mode 100644 index 00000000..b4996ebd --- /dev/null +++ b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/AreaGeometry.kt @@ -0,0 +1,94 @@ +package world.phantasmal.lib.fileFormats + +import world.phantasmal.lib.cursor.Cursor +import world.phantasmal.lib.fileFormats.ninja.NinjaObject +import world.phantasmal.lib.fileFormats.ninja.XjModel +import world.phantasmal.lib.fileFormats.ninja.angleToRad +import world.phantasmal.lib.fileFormats.ninja.parseXjObject + +class RenderObject( + val sections: List, +) + +class RenderSection( + val id: Int, + val position: Vec3, + val rotation: Vec3, + val objects: List>, +) + +fun parseAreaGeometry(cursor: Cursor): RenderObject { + val sections = mutableListOf() + + cursor.seekEnd(16) + + val dataOffset = parseRel(cursor, parseIndex = false).dataOffset + cursor.seekStart(dataOffset) + cursor.seek(8) // Format "fmt2" in UTF-16. + val sectionCount = cursor.int() + cursor.seek(4) + val sectionTableOffset = cursor.int() + // val textureNameOffset = cursor.int() + + for (i in 0 until sectionCount) { + cursor.seekStart(sectionTableOffset + 52 * i) + + val sectionId = cursor.int() + val sectionPosition = cursor.vec3Float() + val sectionRotation = Vec3( + angleToRad(cursor.int()), + angleToRad(cursor.int()), + angleToRad(cursor.int()), + ) + + cursor.seek(4) + + val simpleGeometryOffsetTableOffset = cursor.int() +// val animatedGeometryOffsetTableOffset = cursor.int() + cursor.seek(4) + val simpleGeometryOffsetCount = cursor.int() +// val animatedGeometryOffsetCount = cursor.int() + // Ignore animatedGeometryOffsetCount and the last 4 bytes. + + val objects = parseGeometryTable( + cursor, + simpleGeometryOffsetTableOffset, + simpleGeometryOffsetCount, + ) + + sections.add(RenderSection( + sectionId, + sectionPosition, + sectionRotation, + objects, + )) + } + + return RenderObject(sections) +} + +// TODO: don't reparse the same objects multiple times. Create DAG instead of tree. +private fun parseGeometryTable( + cursor: Cursor, + tableOffset: Int, + tableEntryCount: Int, +): List> { + val objects = mutableListOf>() + + for (i in 0 until tableEntryCount) { + cursor.seekStart(tableOffset + 16 * i) + + var offset = cursor.int() + cursor.seek(8) + val flags = cursor.int() + + if (flags and 0b100 != 0) { + offset = cursor.seekStart(offset).int() + } + + cursor.seekStart(offset) + objects.addAll(parseXjObject(cursor)) + } + + return objects +} 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 8f948c77..a10982ae 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 @@ -16,6 +16,9 @@ fun parseNj(cursor: Cursor): PwResult>> = fun parseXj(cursor: Cursor): PwResult>> = parseNinja(cursor, { c, _ -> parseXjModel(c) }, Unit) +fun parseXjObject(cursor: Cursor): List> = + parseSiblingObjects(cursor, { c, _ -> parseXjModel(c) }, Unit) + private fun parseNinja( cursor: Cursor, parseModel: (cursor: Cursor, context: Context) -> Model, 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 index c23e080c..bc812f7e 100644 --- a/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/QuestEntity.kt +++ b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/QuestEntity.kt @@ -7,6 +7,8 @@ interface QuestEntity { var areaId: Int + var sectionId: Int + /** * Section-relative position. */ 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 b2888fd5..436b3d41 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 @@ -69,6 +69,12 @@ class QuestNpc( } } + override var sectionId: Int + get() = data.getUShort(12).toInt() + set(value) { + data.setUShort(12, value.toUShort()) + } + override var position: Vec3 get() = Vec3(data.getFloat(20), data.getFloat(24), data.getFloat(28)) set(value) { 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 b4109012..fb4c1363 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 @@ -19,6 +19,12 @@ class QuestObject(override var areaId: Int, val data: Buffer) : QuestEntity): VertexData = NinjaToVertexDataConverter(VertexDataBuilder()).convert(ninjaObject) +fun ninjaObjectToVertexDataBuilder( + ninjaObject: NinjaObject<*>, + builder: VertexDataBuilder, +): VertexData = + NinjaToVertexDataConverter(builder).convert(ninjaObject) + +// TODO: take into account different kinds of meshes/vertices (with or without normals, uv, etc.). private class NinjaToVertexDataConverter(private val builder: VertexDataBuilder) { private val vertexHolder = VertexHolder() private var boneIndex = 0 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 index 8c52aa4e..21d30e10 100644 --- a/web/src/main/kotlin/world/phantasmal/web/externals/babylon/babylon.kt +++ b/web/src/main/kotlin/world/phantasmal/web/externals/babylon/babylon.kt @@ -82,13 +82,16 @@ external class Quaternion( * @return the current quaternion */ fun multiplyToRef(q1: Quaternion, result: Quaternion): Quaternion - + fun toEulerAngles(): Vector3 + fun toEulerAnglesToRef(result: Vector3): Quaternion + fun rotateByQuaternionToRef(quaternion: Quaternion, result: Vector3): Vector3 fun clone(): Quaternion fun copyFrom(other: Quaternion): Quaternion companion object { fun Identity(): Quaternion fun FromEulerAngles(x: Double, y: Double, z: Double): Quaternion + fun FromEulerAnglesToRef(x: Double, y: Double, z: Double, result: Quaternion): Quaternion fun RotationYawPitchRoll(yaw: Double, pitch: Double, roll: Double): Quaternion } } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/QuestEditorToolbarController.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/QuestEditorToolbarController.kt index b7fa7a39..d3ec4581 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/QuestEditorToolbarController.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/QuestEditorToolbarController.kt @@ -81,7 +81,7 @@ class QuestEditorToolbarController( } } - private fun setCurrentQuest(quest: Quest) { + private suspend fun setCurrentQuest(quest: Quest) { questEditorStore.setCurrentQuest(convertQuestToModel(quest, areaStore::getVariant)) } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/loading/AreaAssetLoader.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/loading/AreaAssetLoader.kt index 695e1b7b..436b33ff 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/loading/AreaAssetLoader.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/loading/AreaAssetLoader.kt @@ -5,28 +5,65 @@ import kotlinx.coroutines.async import org.khronos.webgl.ArrayBuffer import world.phantasmal.lib.Endianness import world.phantasmal.lib.cursor.cursor +import world.phantasmal.lib.fileFormats.CollisionObject +import world.phantasmal.lib.fileFormats.RenderObject import world.phantasmal.lib.fileFormats.parseAreaCollisionGeometry +import world.phantasmal.lib.fileFormats.parseAreaGeometry import world.phantasmal.lib.fileFormats.quest.Episode import world.phantasmal.web.core.loading.AssetLoader +import world.phantasmal.web.core.rendering.conversion.VertexDataBuilder +import world.phantasmal.web.core.rendering.conversion.ninjaObjectToVertexDataBuilder +import world.phantasmal.web.core.rendering.conversion.vec3ToBabylon +import world.phantasmal.web.externals.babylon.Mesh import world.phantasmal.web.externals.babylon.Scene import world.phantasmal.web.externals.babylon.TransformNode import world.phantasmal.web.questEditor.models.AreaVariantModel -import world.phantasmal.web.questEditor.rendering.conversion.areaCollisionGeometryToTransformNode +import world.phantasmal.web.questEditor.models.SectionModel import world.phantasmal.webui.DisposableContainer +/** + * Loads and caches area assets. + */ class AreaAssetLoader( private val scope: CoroutineScope, private val assetLoader: AssetLoader, private val scene: Scene, ) : DisposableContainer() { - private val collisionObjectCache = - addDisposable(LoadingCache, TransformNode> { it.dispose() }) + /** + * This cache's values consist of a TransformNode containing area render meshes and a list of + * that area's sections. + */ + private val renderObjectCache = addDisposable( + LoadingCache>> { it.first.dispose() } + ) + + private val collisionObjectCache = addDisposable( + LoadingCache { it.dispose() } + ) + + suspend fun loadSections(episode: Episode, areaVariant: AreaVariantModel): List = + loadRenderGeometryAndSections(episode, areaVariant).second + + suspend fun loadRenderGeometry(episode: Episode, areaVariant: AreaVariantModel): TransformNode = + loadRenderGeometryAndSections(episode, areaVariant).first + + private suspend fun loadRenderGeometryAndSections( + episode: Episode, + areaVariant: AreaVariantModel, + ): Pair> = + renderObjectCache.getOrPut(CacheKey(episode, areaVariant.area.id, areaVariant.id)) { + scope.async { + val buffer = getAreaAsset(episode, areaVariant, AssetType.Render) + val obj = parseAreaGeometry(buffer.cursor(Endianness.Little)) + areaGeometryToTransformNodeAndSections(scene, obj, areaVariant) + } + }.await() suspend fun loadCollisionGeometry( episode: Episode, areaVariant: AreaVariantModel, ): TransformNode = - collisionObjectCache.getOrPut(Triple(episode, areaVariant.area.id, areaVariant.id)) { + collisionObjectCache.getOrPut(CacheKey(episode, areaVariant.area.id, areaVariant.id)) { scope.async { val buffer = getAreaAsset(episode, areaVariant, AssetType.Collision) val obj = parseAreaCollisionGeometry(buffer.cursor(Endianness.Little)) @@ -47,11 +84,21 @@ class AreaAssetLoader( return assetLoader.loadArrayBuffer(baseUrl + suffix) } - enum class AssetType { + private data class CacheKey( + val episode: Episode, + val areaId: Int, + val areaVariantId: Int, + ) + + private enum class AssetType { Render, Collision } } +class AreaMetadata( + val section: SectionModel?, +) + private val AREA_BASE_NAMES: Map>> = mapOf( Episode.I to listOf( Pair("city00_00", 1), @@ -135,3 +182,87 @@ private fun areaVersionToBaseUrl(episode: Episode, areaVariant: AreaVariantModel return "/maps/map_${base_name}${variant}" } + +private fun areaGeometryToTransformNodeAndSections( + scene: Scene, + renderObject: RenderObject, + areaVariant: AreaVariantModel, +): Pair> { + val sections = mutableListOf() + val node = TransformNode("Render Geometry", scene) + node.setEnabled(false) + + for (section in renderObject.sections) { + val builder = VertexDataBuilder() + + for (obj in section.objects) { + ninjaObjectToVertexDataBuilder(obj, builder) + } + + val vertexData = builder.build() + val mesh = Mesh("Render Geometry", scene, node) + vertexData.applyToMesh(mesh) + // TODO: Material. + + mesh.position = vec3ToBabylon(section.position) + mesh.rotation = vec3ToBabylon(section.rotation) + + if (section.id >= 0) { + val sec = SectionModel( + section.id, + vec3ToBabylon(section.position), + vec3ToBabylon(section.rotation), + areaVariant, + ) + sections.add(sec) + mesh.metadata = AreaMetadata(sec) + } + } + + return Pair(node, sections) +} + +private fun areaCollisionGeometryToTransformNode( + scene: Scene, + obj: CollisionObject, +): TransformNode { + val node = TransformNode("Collision Geometry", scene) + + for (collisionMesh in obj.meshes) { + val builder = VertexDataBuilder() + + for (triangle in collisionMesh.triangles) { + val isSectionTransition = (triangle.flags and 0b1000000) != 0 + val isVegetation = (triangle.flags and 0b10000) != 0 + val isGround = (triangle.flags and 0b1) != 0 + val colorIndex = when { + isSectionTransition -> 3 + isVegetation -> 2 + isGround -> 1 + else -> 0 + } + + // Filter out walls. + if (colorIndex != 0) { + val p1 = vec3ToBabylon(collisionMesh.vertices[triangle.index1]) + val p2 = vec3ToBabylon(collisionMesh.vertices[triangle.index2]) + val p3 = vec3ToBabylon(collisionMesh.vertices[triangle.index3]) + val n = vec3ToBabylon(triangle.normal) + + builder.addIndex(builder.vertexCount) + builder.addVertex(p1, n) + builder.addIndex(builder.vertexCount) + builder.addVertex(p3, n) + builder.addIndex(builder.vertexCount) + builder.addVertex(p2, n) + } + } + + if (builder.vertexCount > 0) { + val mesh = Mesh("Collision Geometry", scene, parent = node) + builder.build().applyToMesh(mesh) + } + } + + return node +} 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 index c70c0f47..67d8225f 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestEntityModel.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestEntityModel.kt @@ -1,29 +1,114 @@ package world.phantasmal.web.questEditor.models +import world.phantasmal.core.math.floorMod 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.plusAssign +import world.phantasmal.web.core.rendering.conversion.babylonToVec3 import world.phantasmal.web.core.rendering.conversion.vec3ToBabylon +import world.phantasmal.web.core.timesAssign +import world.phantasmal.web.externals.babylon.Quaternion import world.phantasmal.web.externals.babylon.Vector3 +import kotlin.math.PI abstract class QuestEntityModel>( private val entity: Entity, ) { + private val _sectionId = mutableVal(entity.sectionId) + private val _section = mutableVal(null) private val _position = mutableVal(vec3ToBabylon(entity.position)) private val _worldPosition = mutableVal(_position.value) + private val _rotation = mutableVal(vec3ToBabylon(entity.rotation)) + private val _worldRotation = mutableVal(_rotation.value) val type: Type get() = entity.type val areaId: Int get() = entity.areaId + val sectionId: Val = _sectionId + + val section: Val = _section + /** * Section-relative position */ val position: Val = _position - /** - * World position - */ val worldPosition: Val = _worldPosition + + /** + * Section-relative rotation + */ + val rotation: Val = _rotation + + val worldRotation: Val = _worldRotation + + fun setSection(section: SectionModel) { + require(section.areaVariant.area.id == areaId) { + "Quest entities can't be moved across areas." + } + + entity.sectionId = section.id + + _section.value = section + _sectionId.value = section.id + + setPosition(position.value) + setRotation(rotation.value) + } + + fun setPosition(pos: Vector3) { + entity.position = babylonToVec3(pos) + + _position.value = pos + + val section = section.value + + _worldPosition.value = + if (section == null) pos + else Vector3.Zero().also { worldPos -> + pos.rotateByQuaternionToRef(section.rotationQuaternion, worldPos) + worldPos += section.position + } + } + + fun setRotation(rot: Vector3) { + floorModEuler(rot) + + entity.rotation = babylonToVec3(rot) + + val section = section.value + + if (section == null) { + _worldRotation.value = rot + } else { + Quaternion.FromEulerAnglesToRef(rot.x, rot.y, rot.z, q1) + Quaternion.FromEulerAnglesToRef( + section.rotation.x, + section.rotation.y, + section.rotation.z, + q2 + ) + q1 *= q2 + val worldRot = q1.toEulerAngles() + floorModEuler(worldRot) + _worldRotation.value = worldRot + } + } + + private fun floorModEuler(euler: Vector3) { + euler.set( + floorMod(euler.x, 2 * PI), + floorMod(euler.y, 2 * PI), + floorMod(euler.z, 2 * PI), + ) + } + + companion object { + // These quaternions are used as temporary variables to avoid memory allocation. + private val q1 = Quaternion() + private val q2 = Quaternion() + } } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/models/SectionModel.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/SectionModel.kt index 7c368ff4..43bb1dec 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/models/SectionModel.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/SectionModel.kt @@ -1,5 +1,6 @@ package world.phantasmal.web.questEditor.models +import world.phantasmal.web.externals.babylon.Quaternion import world.phantasmal.web.externals.babylon.Vector3 class SectionModel( @@ -13,4 +14,7 @@ class SectionModel( "id should be greater than or equal to -1 but was $id." } } + + val rotationQuaternion: Quaternion = + Quaternion.FromEulerAngles(rotation.x, rotation.y, rotation.z) } 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 index 7abc490b..53bf492c 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/EntityMeshManager.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/EntityMeshManager.kt @@ -146,13 +146,11 @@ private class LoadedEntity( mesh.position = pos } - addDisposables( - // TODO: Rotation. -// entity.worldRotation.observe { (value) -> -// mesh.rotation.copy(value) -// renderer.schedule_render() -// }, + observe(entity.worldRotation) { rot -> + mesh.rotation = rot + } + addDisposables( // TODO: Model. // entity.model.observe { // remove(listOf(entity)) diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/conversion/Areas.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/conversion/Areas.kt deleted file mode 100644 index f2f69b52..00000000 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/conversion/Areas.kt +++ /dev/null @@ -1,50 +0,0 @@ -package world.phantasmal.web.questEditor.rendering.conversion - -import world.phantasmal.lib.fileFormats.CollisionObject -import world.phantasmal.web.core.rendering.conversion.VertexDataBuilder -import world.phantasmal.web.core.rendering.conversion.vec3ToBabylon -import world.phantasmal.web.externals.babylon.Mesh -import world.phantasmal.web.externals.babylon.Scene -import world.phantasmal.web.externals.babylon.TransformNode - -fun areaCollisionGeometryToTransformNode(scene: Scene, obj: CollisionObject): TransformNode { - val node = TransformNode("", scene) - - for (collisionMesh in obj.meshes) { - val builder = VertexDataBuilder() - - for (triangle in collisionMesh.triangles) { - val isSectionTransition = (triangle.flags and 0b1000000) != 0 - val isVegetation = (triangle.flags and 0b10000) != 0 - val isGround = (triangle.flags and 0b1) != 0 - val colorIndex = when { - isSectionTransition -> 3 - isVegetation -> 2 - isGround -> 1 - else -> 0 - } - - // Filter out walls. - if (colorIndex != 0) { - val p1 = vec3ToBabylon(collisionMesh.vertices[triangle.index1]) - val p2 = vec3ToBabylon(collisionMesh.vertices[triangle.index2]) - val p3 = vec3ToBabylon(collisionMesh.vertices[triangle.index3]) - val n = vec3ToBabylon(triangle.normal) - - builder.addIndex(builder.vertexCount) - builder.addVertex(p1, n) - builder.addIndex(builder.vertexCount) - builder.addVertex(p3, n) - builder.addIndex(builder.vertexCount) - builder.addVertex(p2, n) - } - } - - if (builder.vertexCount > 0) { - val mesh = Mesh("Collision Geometry", scene, parent = node) - builder.build().applyToMesh(mesh) - } - } - - return node -} diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/AreaStore.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/AreaStore.kt index a784f787..8f0886aa 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/AreaStore.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/AreaStore.kt @@ -2,19 +2,23 @@ package world.phantasmal.web.questEditor.stores import kotlinx.coroutines.CoroutineScope import world.phantasmal.lib.fileFormats.quest.Episode -import world.phantasmal.lib.fileFormats.quest.getAreasForEpisode import world.phantasmal.web.questEditor.loading.AreaAssetLoader import world.phantasmal.web.questEditor.models.AreaModel import world.phantasmal.web.questEditor.models.AreaVariantModel +import world.phantasmal.web.questEditor.models.SectionModel import world.phantasmal.webui.stores.Store +import world.phantasmal.lib.fileFormats.quest.getAreasForEpisode as getAreasForEpisodeLib -class AreaStore(scope: CoroutineScope, areaAssetLoader: AreaAssetLoader) : Store(scope) { +class AreaStore( + scope: CoroutineScope, + private val areaAssetLoader: AreaAssetLoader, +) : Store(scope) { private val areas: Map> init { areas = Episode.values() .map { episode -> - episode to getAreasForEpisode(episode).map { area -> + episode to getAreasForEpisodeLib(episode).map { area -> val variants = mutableListOf() val areaModel = AreaModel(area.id, area.name, area.order, variants) @@ -28,9 +32,15 @@ class AreaStore(scope: CoroutineScope, areaAssetLoader: AreaAssetLoader) : Store .toMap() } + fun getAreasForEpisode(episode: Episode): List = + areas.getValue(episode) + fun getArea(episode: Episode, areaId: Int): AreaModel? = areas.getValue(episode).find { it.id == areaId } fun getVariant(episode: Episode, areaId: Int, variantId: Int): AreaVariantModel? = getArea(episode, areaId)?.areaVariants?.getOrNull(variantId) + + suspend fun getSections(episode: Episode, variant: AreaVariantModel): List = + areaAssetLoader.loadSections(episode, variant) } 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 56785f71..ab65322e 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 @@ -1,14 +1,14 @@ package world.phantasmal.web.questEditor.stores import kotlinx.coroutines.CoroutineScope +import mu.KotlinLogging 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.QuestEntityModel -import world.phantasmal.web.questEditor.models.QuestModel -import world.phantasmal.web.questEditor.models.WaveModel +import world.phantasmal.web.questEditor.models.* import world.phantasmal.webui.stores.Store +private val logger = KotlinLogging.logger {} + class QuestEditorStore(scope: CoroutineScope, private val areaStore: AreaStore) : Store(scope) { private val _currentQuest = mutableVal(null) private val _currentArea = mutableVal(null) @@ -23,12 +23,38 @@ class QuestEditorStore(scope: CoroutineScope, private val areaStore: AreaStore) // TODO: Take into account whether we're debugging or not. val questEditingDisabled: Val = currentQuest.map { it == null } - fun setCurrentQuest(quest: QuestModel?) { + suspend fun setCurrentQuest(quest: QuestModel?) { _currentArea.value = null _currentQuest.value = quest quest?.let { _currentArea.value = areaStore.getArea(quest.episode, 0) + + // Load section data. + quest.areaVariants.value.forEach { variant -> + val sections = areaStore.getSections(quest.episode, variant) + variant.setSections(sections) + setSectionOnQuestEntities(quest.npcs.value, variant, sections) + setSectionOnQuestEntities(quest.objects.value, variant, sections) + } + } + } + + private fun setSectionOnQuestEntities( + entities: List>, + variant: AreaVariantModel, + sections: List, + ) { + entities.forEach { entity -> + if (entity.areaId == variant.area.id) { + val section = sections.find { it.id == entity.sectionId.value } + + if (section == null) { + logger.warn { "Section ${entity.sectionId.value} not found." } + } else { + entity.setSection(section) + } + } } }