Animated n.rel geometry is now parsed and rendered (without animations).

This commit is contained in:
Daan Vanden Bosch 2021-04-09 16:08:38 +02:00
parent f20711296b
commit 92017ca8ec
2 changed files with 75 additions and 27 deletions

View File

@ -15,6 +15,7 @@ class RenderSection(
val position: Vec3, val position: Vec3,
val rotation: Vec3, val rotation: Vec3,
val objects: List<XjObject>, val objects: List<XjObject>,
val animatedObjects: List<XjObject>,
) )
fun parseAreaRenderGeometry(cursor: Cursor): RenderGeometry { fun parseAreaRenderGeometry(cursor: Cursor): RenderGeometry {
@ -30,6 +31,8 @@ fun parseAreaRenderGeometry(cursor: Cursor): RenderGeometry {
val sectionTableOffset = cursor.int() val sectionTableOffset = cursor.int()
// val textureNameOffset = cursor.int() // val textureNameOffset = cursor.int()
val xjObjectCache = mutableMapOf<Int, List<XjObject>>()
for (i in 0 until sectionCount) { for (i in 0 until sectionCount) {
cursor.seekStart(sectionTableOffset + 52 * i) cursor.seekStart(sectionTableOffset + 52 * i)
@ -41,19 +44,28 @@ fun parseAreaRenderGeometry(cursor: Cursor): RenderGeometry {
angleToRad(cursor.int()), angleToRad(cursor.int()),
) )
cursor.seek(4) cursor.seek(4) // Radius?
val simpleGeometryOffsetTableOffset = cursor.int() val simpleGeometryOffsetTableOffset = cursor.int()
// val animatedGeometryOffsetTableOffset = cursor.int() val animatedGeometryOffsetTableOffset = cursor.int()
cursor.seek(4)
val simpleGeometryOffsetCount = cursor.int() val simpleGeometryOffsetCount = cursor.int()
// val animatedGeometryOffsetCount = cursor.int() val animatedGeometryOffsetCount = cursor.int()
// Ignore animatedGeometryOffsetCount and the last 4 bytes. // Ignore the last 4 bytes.
val objects = parseGeometryTable( val objects = parseGeometryTable(
cursor, cursor,
xjObjectCache,
simpleGeometryOffsetTableOffset, simpleGeometryOffsetTableOffset,
simpleGeometryOffsetCount, simpleGeometryOffsetCount,
animated = false,
)
val animatedObjects = parseGeometryTable(
cursor,
xjObjectCache,
animatedGeometryOffsetTableOffset,
animatedGeometryOffsetCount,
animated = true,
) )
sections.add(RenderSection( sections.add(RenderSection(
@ -61,22 +73,25 @@ fun parseAreaRenderGeometry(cursor: Cursor): RenderGeometry {
sectionPosition, sectionPosition,
sectionRotation, sectionRotation,
objects, objects,
animatedObjects,
)) ))
} }
return RenderGeometry(sections) return RenderGeometry(sections)
} }
// TODO: don't reparse the same objects multiple times. Create DAG instead of tree.
private fun parseGeometryTable( private fun parseGeometryTable(
cursor: Cursor, cursor: Cursor,
xjObjectCache: MutableMap<Int, List<XjObject>>,
tableOffset: Int, tableOffset: Int,
tableEntryCount: Int, tableEntryCount: Int,
animated: Boolean,
): List<XjObject> { ): List<XjObject> {
val tableEntrySize = if (animated) 32 else 16
val objects = mutableListOf<XjObject>() val objects = mutableListOf<XjObject>()
for (i in 0 until tableEntryCount) { for (i in 0 until tableEntryCount) {
cursor.seekStart(tableOffset + 16 * i) cursor.seekStart(tableOffset + tableEntrySize * i)
var offset = cursor.int() var offset = cursor.int()
cursor.seek(8) cursor.seek(8)
@ -86,8 +101,12 @@ private fun parseGeometryTable(
offset = cursor.seekStart(offset).int() offset = cursor.seekStart(offset).int()
} }
cursor.seekStart(offset) objects.addAll(
objects.addAll(parseXjObject(cursor)) xjObjectCache.getOrPut(offset) {
cursor.seekStart(offset)
parseXjObject(cursor)
}
)
} }
return objects return objects

View File

@ -130,34 +130,63 @@ fun renderGeometryToGroup(
): Group { ): Group {
val group = Group() val group = Group()
val textureCache = mutableMapOf<Int, Texture?>() val textureCache = mutableMapOf<Int, Texture?>()
val meshCache = mutableMapOf<XjObject, Mesh>()
for ((i, section) in renderGeometry.sections.withIndex()) { for ((i, section) in renderGeometry.sections.withIndex()) {
for (xjObj in section.objects) { for (xjObj in section.objects) {
val builder = MeshBuilder(textures, textureCache) group.add(xjObjectToMesh(
ninjaObjectToMeshBuilder(xjObj, builder) textures, textureCache, meshCache, xjObj, i, section, processMesh,
))
}
builder.defaultMaterial(MeshBasicMaterial(obj { for (xjObj in section.animatedObjects) {
color = Color().setHSL((i % 7) / 7.0, 1.0, .5) group.add(xjObjectToMesh(
transparent = true textures, textureCache, meshCache, xjObj, i, section, processMesh,
opacity = .25 ))
side = DoubleSide
}))
val mesh = builder.buildMesh(boundingVolumes = true)
mesh.position.setFromVec3(section.position)
mesh.rotation.setFromVec3(section.rotation)
mesh.updateMatrixWorld()
processMesh(section, xjObj, mesh)
group.add(mesh)
} }
} }
return group return group
} }
private fun xjObjectToMesh(
textures: List<XvrTexture?>,
textureCache: MutableMap<Int, Texture?>,
meshCache: MutableMap<XjObject, Mesh>,
xjObj: XjObject,
index: Int,
section: RenderSection,
processMesh: (RenderSection, XjObject, Mesh) -> Unit,
): Mesh {
var mesh = meshCache[xjObj]
if (mesh == null) {
val builder = MeshBuilder(textures, textureCache)
ninjaObjectToMeshBuilder(xjObj, builder)
builder.defaultMaterial(MeshBasicMaterial(obj {
color = Color().setHSL((index % 7) / 7.0, 1.0, .5)
transparent = true
opacity = .25
side = DoubleSide
}))
mesh = builder.buildMesh(boundingVolumes = true)
} else {
// If we already have a mesh for this XjObject, make a copy and reuse the existing buffer
// geometry and materials.
mesh = Mesh(mesh.geometry, mesh.material.unsafeCast<Array<Material>>())
}
mesh.position.setFromVec3(section.position)
mesh.rotation.setFromVec3(section.rotation)
mesh.updateMatrixWorld()
processMesh(section, xjObj, mesh)
return mesh
}
fun collisionGeometryToGroup( fun collisionGeometryToGroup(
collisionGeometry: CollisionGeometry, collisionGeometry: CollisionGeometry,
trianglePredicate: (CollisionTriangle) -> Boolean = { true }, trianglePredicate: (CollisionTriangle) -> Boolean = { true },