mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-04 06:28:28 +08:00
Entities are now correctly rotated and positioned within their section.
This commit is contained in:
parent
346a2cb4f9
commit
db1149ddc0
@ -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
|
@ -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<RenderSection>,
|
||||
)
|
||||
|
||||
class RenderSection(
|
||||
val id: Int,
|
||||
val position: Vec3,
|
||||
val rotation: Vec3,
|
||||
val objects: List<NinjaObject<XjModel>>,
|
||||
)
|
||||
|
||||
fun parseAreaGeometry(cursor: Cursor): RenderObject {
|
||||
val sections = mutableListOf<RenderSection>()
|
||||
|
||||
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<NinjaObject<XjModel>> {
|
||||
val objects = mutableListOf<NinjaObject<XjModel>>()
|
||||
|
||||
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
|
||||
}
|
@ -16,6 +16,9 @@ fun parseNj(cursor: Cursor): PwResult<List<NinjaObject<NjModel>>> =
|
||||
fun parseXj(cursor: Cursor): PwResult<List<NinjaObject<XjModel>>> =
|
||||
parseNinja(cursor, { c, _ -> parseXjModel(c) }, Unit)
|
||||
|
||||
fun parseXjObject(cursor: Cursor): List<NinjaObject<XjModel>> =
|
||||
parseSiblingObjects(cursor, { c, _ -> parseXjModel(c) }, Unit)
|
||||
|
||||
private fun <Model : NinjaModel, Context> parseNinja(
|
||||
cursor: Cursor,
|
||||
parseModel: (cursor: Cursor, context: Context) -> Model,
|
||||
|
@ -7,6 +7,8 @@ interface QuestEntity<Type : EntityType> {
|
||||
|
||||
var areaId: Int
|
||||
|
||||
var sectionId: Int
|
||||
|
||||
/**
|
||||
* Section-relative position.
|
||||
*/
|
||||
|
@ -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) {
|
||||
|
@ -19,6 +19,12 @@ class QuestObject(override var areaId: Int, val data: Buffer) : QuestEntity<Obje
|
||||
typeId = value.typeId ?: -1
|
||||
}
|
||||
|
||||
override var sectionId: Int
|
||||
get() = data.getUShort(12).toInt()
|
||||
set(value) {
|
||||
data.setUShort(12, value.toUShort())
|
||||
}
|
||||
|
||||
override var position: Vec3
|
||||
get() = Vec3(data.getFloat(16), data.getFloat(20), data.getFloat(24))
|
||||
set(value) {
|
||||
|
@ -1,8 +1,13 @@
|
||||
package world.phantasmal.web.core
|
||||
|
||||
import world.phantasmal.web.externals.babylon.Matrix
|
||||
import world.phantasmal.web.externals.babylon.Quaternion
|
||||
import world.phantasmal.web.externals.babylon.Vector3
|
||||
|
||||
operator fun Vector3.plusAssign(other: Vector3) {
|
||||
addInPlace(other)
|
||||
}
|
||||
|
||||
operator fun Vector3.minus(other: Vector3): Vector3 =
|
||||
subtract(other)
|
||||
|
||||
@ -32,3 +37,7 @@ fun Matrix.multiply(v: Vector3) {
|
||||
fun Matrix.multiply3x3(v: Vector3) {
|
||||
Vector3.TransformNormalToRef(v, this, v)
|
||||
}
|
||||
|
||||
operator fun Quaternion.timesAssign(other: Quaternion) {
|
||||
multiplyInPlace(other)
|
||||
}
|
||||
|
@ -8,3 +8,5 @@ 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())
|
||||
|
||||
fun babylonToVec3(v: Vector3): Vec3 = Vec3(v.x.toFloat(), v.y.toFloat(), v.z.toFloat())
|
||||
|
@ -19,10 +19,16 @@ 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)
|
||||
|
||||
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
|
||||
|
@ -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
|
||||
}
|
||||
}
|
||||
|
@ -81,7 +81,7 @@ class QuestEditorToolbarController(
|
||||
}
|
||||
}
|
||||
|
||||
private fun setCurrentQuest(quest: Quest) {
|
||||
private suspend fun setCurrentQuest(quest: Quest) {
|
||||
questEditorStore.setCurrentQuest(convertQuestToModel(quest, areaStore::getVariant))
|
||||
}
|
||||
|
||||
|
@ -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<Triple<Episode, Int, Int>, 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<CacheKey, Pair<TransformNode, List<SectionModel>>> { it.first.dispose() }
|
||||
)
|
||||
|
||||
private val collisionObjectCache = addDisposable(
|
||||
LoadingCache<CacheKey, TransformNode> { it.dispose() }
|
||||
)
|
||||
|
||||
suspend fun loadSections(episode: Episode, areaVariant: AreaVariantModel): List<SectionModel> =
|
||||
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<TransformNode, List<SectionModel>> =
|
||||
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<Episode, List<Pair<String, Int>>> = 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<TransformNode, List<SectionModel>> {
|
||||
val sections = mutableListOf<SectionModel>()
|
||||
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
|
||||
}
|
||||
|
@ -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<Type : EntityType, Entity : QuestEntity<Type>>(
|
||||
private val entity: Entity,
|
||||
) {
|
||||
private val _sectionId = mutableVal(entity.sectionId)
|
||||
private val _section = mutableVal<SectionModel?>(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<Int> = _sectionId
|
||||
|
||||
val section: Val<SectionModel?> = _section
|
||||
|
||||
/**
|
||||
* Section-relative position
|
||||
*/
|
||||
val position: Val<Vector3> = _position
|
||||
|
||||
/**
|
||||
* World position
|
||||
*/
|
||||
val worldPosition: Val<Vector3> = _worldPosition
|
||||
|
||||
/**
|
||||
* Section-relative rotation
|
||||
*/
|
||||
val rotation: Val<Vector3> = _rotation
|
||||
|
||||
val worldRotation: Val<Vector3> = _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()
|
||||
}
|
||||
}
|
||||
|
@ -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)
|
||||
}
|
||||
|
@ -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))
|
||||
|
@ -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
|
||||
}
|
@ -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<Episode, List<AreaModel>>
|
||||
|
||||
init {
|
||||
areas = Episode.values()
|
||||
.map { episode ->
|
||||
episode to getAreasForEpisode(episode).map { area ->
|
||||
episode to getAreasForEpisodeLib(episode).map { area ->
|
||||
val variants = mutableListOf<AreaVariantModel>()
|
||||
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<AreaModel> =
|
||||
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<SectionModel> =
|
||||
areaAssetLoader.loadSections(episode, variant)
|
||||
}
|
||||
|
@ -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<QuestModel?>(null)
|
||||
private val _currentArea = mutableVal<AreaModel?>(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<Boolean> = 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<QuestEntityModel<*, *>>,
|
||||
variant: AreaVariantModel,
|
||||
sections: List<SectionModel>,
|
||||
) {
|
||||
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)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user