Added manual area render geometry culling code.

This commit is contained in:
Daan Vanden Bosch 2021-04-11 21:43:34 +02:00
parent 126b50cb00
commit 29192e5684
11 changed files with 551 additions and 200 deletions

View File

@ -20,6 +20,15 @@ fun Int.isBitSet(bit: Int): Boolean =
fun UByte.isBitSet(bit: Int): Boolean = fun UByte.isBitSet(bit: Int): Boolean =
toInt().isBitSet(bit) toInt().isBitSet(bit)
fun Int.setBit(bit: Int, value: Boolean): Int =
if (value) {
this or (1 shl bit)
} else {
this and (1 shl bit).inv()
}
expect fun Int.reinterpretAsFloat(): Float expect fun Int.reinterpretAsFloat(): Float
expect fun Float.reinterpretAsInt(): Int expect fun Float.reinterpretAsInt(): Int
expect fun Float.reinterpretAsUInt(): UInt

View File

@ -43,9 +43,9 @@ inline val <T> JsPair<*, T>.second: T get() = asDynamic()[1].unsafeCast<T>()
inline operator fun <T> JsPair<T, *>.component1(): T = first inline operator fun <T> JsPair<T, *>.component1(): T = first
inline operator fun <T> JsPair<*, T>.component2(): T = second inline operator fun <T> JsPair<*, T>.component2(): T = second
@Suppress("UNUSED_PARAMETER") @Suppress("FunctionName", "UNUSED_PARAMETER")
inline fun objectKeys(jsObject: dynamic): Array<String> = inline fun <A, B> JsPair(first: A, second: B): JsPair<A, B> =
js("Object.keys(jsObject)").unsafeCast<Array<String>>() js("[first, second]").unsafeCast<JsPair<A, B>>()
@Suppress("UNUSED_PARAMETER") @Suppress("UNUSED_PARAMETER")
inline fun objectEntries(jsObject: dynamic): Array<JsPair<String, dynamic>> = inline fun objectEntries(jsObject: dynamic): Array<JsPair<String, dynamic>> =
@ -64,6 +64,10 @@ external interface JsSet<T> {
inline fun <T> emptyJsSet(): JsSet<T> = inline fun <T> emptyJsSet(): JsSet<T> =
js("new Set()").unsafeCast<JsSet<T>>() js("new Set()").unsafeCast<JsSet<T>>()
@Suppress("UNUSED_PARAMETER")
inline fun <T> jsSetOf(vararg values: T): JsSet<T> =
js("new Set(values)").unsafeCast<JsSet<T>>()
external interface JsMap<K, V> { external interface JsMap<K, V> {
val size: Int val size: Int
@ -75,5 +79,9 @@ external interface JsMap<K, V> {
fun set(key: K, value: V): JsMap<K, V> fun set(key: K, value: V): JsMap<K, V>
} }
@Suppress("UNUSED_PARAMETER")
inline fun <K, V> jsMapOf(vararg pairs: JsPair<K, V>): JsMap<K, V> =
js("new Map(pairs)").unsafeCast<JsMap<K, V>>()
inline fun <K, V> emptyJsMap(): JsMap<K, V> = inline fun <K, V> emptyJsMap(): JsMap<K, V> =
js("new Map()").unsafeCast<JsMap<K, V>>() js("new Map()").unsafeCast<JsMap<K, V>>()

View File

@ -14,3 +14,8 @@ actual fun Float.reinterpretAsInt(): Int {
dataView.setFloat32(0, this) dataView.setFloat32(0, this)
return dataView.getInt32(0) return dataView.getInt32(0)
} }
actual fun Float.reinterpretAsUInt(): UInt {
dataView.setFloat32(0, this)
return dataView.getUint32(0).toUInt()
}

View File

@ -8,3 +8,5 @@ import java.lang.Float.intBitsToFloat
actual fun Int.reinterpretAsFloat(): Float = intBitsToFloat(this) actual fun Int.reinterpretAsFloat(): Float = intBitsToFloat(this)
actual fun Float.reinterpretAsInt(): Int = floatToIntBits(this) actual fun Float.reinterpretAsInt(): Int = floatToIntBits(this)
actual fun Float.reinterpretAsUInt(): UInt = reinterpretAsInt().toUInt()

View File

@ -1,40 +1,69 @@
package world.phantasmal.lib.fileFormats package world.phantasmal.lib.fileFormats
import mu.KotlinLogging
import world.phantasmal.core.isBitSet import world.phantasmal.core.isBitSet
import world.phantasmal.lib.cursor.Cursor import world.phantasmal.lib.cursor.Cursor
import world.phantasmal.lib.fileFormats.ninja.XjObject import world.phantasmal.lib.fileFormats.ninja.*
import world.phantasmal.lib.fileFormats.ninja.angleToRad
import world.phantasmal.lib.fileFormats.ninja.parseXjObject
class RenderGeometry( private val logger = KotlinLogging.logger {}
val sections: List<RenderSection>,
class AreaGeometry(
val sections: List<AreaSection>,
) )
class RenderSection( class AreaSection(
val id: Int, val id: Int,
val position: Vec3, val position: Vec3,
val rotation: Vec3, val rotation: Vec3,
val objects: List<XjObject>, val radius: Float,
val animatedObjects: List<XjObject>, val objects: List<AreaObject.Simple>,
val animatedObjects: List<AreaObject.Animated>,
) )
fun parseAreaRenderGeometry(cursor: Cursor): RenderGeometry { sealed class AreaObject {
val sections = mutableListOf<RenderSection>() abstract val offset: Int
abstract val xjObject: XjObject
abstract val flags: Int
cursor.seekEnd(16) class Simple(
override val offset: Int,
override val xjObject: XjObject,
override val flags: Int,
) : AreaObject()
class Animated(
override val offset: Int,
override val xjObject: XjObject,
val njMotion: NjMotion,
val speed: Float,
override val flags: Int,
) : AreaObject()
}
fun parseAreaRenderGeometry(cursor: Cursor): AreaGeometry {
val dataOffset = parseRel(cursor, parseIndex = false).dataOffset val dataOffset = parseRel(cursor, parseIndex = false).dataOffset
cursor.seekStart(dataOffset) cursor.seekStart(dataOffset)
cursor.seek(8) // Format "fmt2" in UTF-16. val format = cursor.stringAscii(maxByteLength = 4, nullTerminated = true, dropRemaining = true)
val sectionCount = cursor.int()
if (format != "fmt2") {
logger.warn { """Expected format to be "fmt2" but was "$format".""" }
}
cursor.seek(4) cursor.seek(4)
val sectionTableOffset = cursor.int() val sectionsCount = cursor.int()
// val textureNameOffset = cursor.int() cursor.seek(4)
val sectionsOffset = cursor.int()
val xjObjectCache = mutableMapOf<Int, List<XjObject>>() val sections = mutableListOf<AreaSection>()
for (i in 0 until sectionCount) { // Cache keys are offsets.
cursor.seekStart(sectionTableOffset + 52 * i) val simpleAreaObjectCache = mutableMapOf<Int, List<AreaObject.Simple>>()
val animatedAreaObjectCache = mutableMapOf<Int, List<AreaObject.Animated>>()
val njMotionCache = mutableMapOf<Int, NjMotion>()
for (i in 0 until sectionsCount) {
cursor.seekStart(sectionsOffset + 52 * i)
val sectionId = cursor.int() val sectionId = cursor.int()
val sectionPosition = cursor.vec3Float() val sectionPosition = cursor.vec3Float()
@ -44,67 +73,110 @@ fun parseAreaRenderGeometry(cursor: Cursor): RenderGeometry {
angleToRad(cursor.int()), angleToRad(cursor.int()),
) )
cursor.seek(4) // Radius? val radius = cursor.float()
val simpleGeometryOffsetTableOffset = cursor.int() val simpleAreaObjectsOffset = cursor.int()
val animatedGeometryOffsetTableOffset = cursor.int() val animatedAreaObjectsOffset = cursor.int()
val simpleGeometryOffsetCount = cursor.int() val simpleAreaObjectsCount = cursor.int()
val animatedGeometryOffsetCount = cursor.int() val animatedAreaObjectsCount = cursor.int()
// Ignore the last 4 bytes. // Ignore the last 4 bytes.
val objects = parseGeometryTable( // println("section $sectionId (index $i), simple geom at $simpleGeometryTableOffset, animated geom at $animatedGeometryTableOffset")
@Suppress("UNCHECKED_CAST")
val simpleObjects = simpleAreaObjectCache.getOrPut(simpleAreaObjectsOffset) {
parseAreaObjects(
cursor, cursor,
xjObjectCache, njMotionCache,
simpleGeometryOffsetTableOffset, simpleAreaObjectsOffset,
simpleGeometryOffsetCount, simpleAreaObjectsCount,
animated = false, animated = false,
) ) as List<AreaObject.Simple>
}
val animatedObjects = parseGeometryTable( @Suppress("UNCHECKED_CAST")
val animatedObjects = animatedAreaObjectCache.getOrPut(animatedAreaObjectsOffset) {
parseAreaObjects(
cursor, cursor,
xjObjectCache, njMotionCache,
animatedGeometryOffsetTableOffset, animatedAreaObjectsOffset,
animatedGeometryOffsetCount, animatedAreaObjectsCount,
animated = true, animated = true,
) ) as List<AreaObject.Animated>
}
sections.add(RenderSection( sections.add(AreaSection(
sectionId, sectionId,
sectionPosition, sectionPosition,
sectionRotation, sectionRotation,
objects, radius,
simpleObjects,
animatedObjects, animatedObjects,
)) ))
} }
return RenderGeometry(sections) return AreaGeometry(sections)
} }
private fun parseGeometryTable( private fun parseAreaObjects(
cursor: Cursor, cursor: Cursor,
xjObjectCache: MutableMap<Int, List<XjObject>>, njMotionCache: MutableMap<Int, NjMotion>,
tableOffset: Int, offset: Int,
tableEntryCount: Int, count: Int,
animated: Boolean, animated: Boolean,
): List<XjObject> { ): List<AreaObject> {
val tableEntrySize = if (animated) 32 else 16 val objectSize = if (animated) 32 else 16
val objects = mutableListOf<XjObject>() val objects = mutableListOf<AreaObject>()
for (i in 0 until tableEntryCount) { for (i in 0 until count) {
cursor.seekStart(tableOffset + tableEntrySize * i) val objectOffset = offset + objectSize * i
cursor.seekStart(objectOffset)
var offset = cursor.int() var xjObjectOffset = cursor.int()
val speed: Float?
val njMotionOffset: Int?
if (animated) {
njMotionOffset = cursor.int()
cursor.seek(8) cursor.seek(8)
speed = cursor.float()
} else {
speed = null
njMotionOffset = null
}
val slideTextureIdOffset = cursor.int()
val swapTextureIdOffset = cursor.int()
val flags = cursor.int() val flags = cursor.int()
if (flags.isBitSet(2)) { if (flags.isBitSet(2)) {
offset = cursor.seekStart(offset).int() xjObjectOffset = cursor.seekStart(xjObjectOffset).int()
} }
objects.addAll( cursor.seekStart(xjObjectOffset)
xjObjectCache.getOrPut(offset) { val xjObjects = parseXjObject(cursor)
cursor.seekStart(offset)
parseXjObject(cursor) if (xjObjects.size > 1) {
logger.warn {
"Expected exactly one xjObject at ${xjObjectOffset}, got ${xjObjects.size}."
}
}
val xjObject = xjObjects.first()
val njMotion = njMotionOffset?.let {
njMotionCache.getOrPut(njMotionOffset) {
cursor.seekStart(njMotionOffset)
parseMotion(cursor, v2Format = false)
}
}
objects.add(
if (animated) {
AreaObject.Animated(objectOffset, xjObject, njMotion!!, speed!!, flags)
} else {
AreaObject.Simple(objectOffset, xjObject, flags)
} }
) )
} }

View File

@ -101,13 +101,13 @@ private fun parseAction(cursor: Cursor): NjMotion {
return parseMotion(cursor, v2Format = false) return parseMotion(cursor, v2Format = false)
} }
private fun parseMotion(cursor: Cursor, v2Format: Boolean): NjMotion { fun parseMotion(cursor: Cursor, v2Format: Boolean): NjMotion {
// For v2, try to determine the end of the mData offset table by finding the lowest mDataOffset // For v2, try to determine the end of the mData offset table by finding the lowest mDataOffset
// value. This is usually the value that the first mDataOffset points to. This value is assumed // value. This is usually the value that the first mDataOffset points to. This value is assumed
// to be the end of the mDataOffset table. // to be the end of the mDataOffset table.
var mDataTableEnd = if (v2Format) cursor.size else cursor.position var mDataTableEnd = if (v2Format) cursor.size else cursor.position
// Points to an array the size of boneCount. // Points to an array with an element per bone.
val mDataTableOffset = cursor.int() val mDataTableOffset = cursor.int()
val frameCount = cursor.int() val frameCount = cursor.int()
val type = cursor.uShort().toInt() val type = cursor.uShort().toInt()

View File

@ -3,7 +3,6 @@ package world.phantasmal.lib.fileFormats.ninja
import world.phantasmal.core.Failure import world.phantasmal.core.Failure
import world.phantasmal.core.PwResult import world.phantasmal.core.PwResult
import world.phantasmal.core.Success import world.phantasmal.core.Success
import world.phantasmal.core.isBitSet
import world.phantasmal.lib.cursor.Cursor import world.phantasmal.lib.cursor.Cursor
import world.phantasmal.lib.fileFormats.Vec3 import world.phantasmal.lib.fileFormats.Vec3
import world.phantasmal.lib.fileFormats.parseIff import world.phantasmal.lib.fileFormats.parseIff
@ -21,6 +20,7 @@ fun parseXjObject(cursor: Cursor): List<XjObject> =
parseSiblingObjects(cursor, { c, _ -> parseXjModel(c) }, ::XjObject, Unit) parseSiblingObjects(cursor, { c, _ -> parseXjModel(c) }, ::XjObject, Unit)
private typealias CreateObject<Model, Obj> = ( private typealias CreateObject<Model, Obj> = (
offset: Int,
evaluationFlags: NinjaEvaluationFlags, evaluationFlags: NinjaEvaluationFlags,
model: Model?, model: Model?,
position: Vec3, position: Vec3,
@ -57,17 +57,8 @@ private fun <Model : NinjaModel, Obj : NinjaObject<Model, Obj>, Context> parseSi
createObject: CreateObject<Model, Obj>, createObject: CreateObject<Model, Obj>,
context: Context, context: Context,
): MutableList<Obj> { ): MutableList<Obj> {
val offset = cursor.position
val evalFlags = cursor.int() val evalFlags = cursor.int()
val noTranslate = evalFlags.isBitSet(0)
val noRotate = evalFlags.isBitSet(1)
val noScale = evalFlags.isBitSet(2)
val hidden = evalFlags.isBitSet(3)
val breakChildTrace = evalFlags.isBitSet(4)
val zxyRotationOrder = evalFlags.isBitSet(5)
val skip = evalFlags.isBitSet(6)
val shapeSkip = evalFlags.isBitSet(7)
val clip = evalFlags.isBitSet(8)
val modifier = evalFlags.isBitSet(9)
val modelOffset = cursor.int() val modelOffset = cursor.int()
val pos = cursor.vec3Float() val pos = cursor.vec3Float()
@ -102,18 +93,8 @@ private fun <Model : NinjaModel, Obj : NinjaObject<Model, Obj>, Context> parseSi
} }
val obj = createObject( val obj = createObject(
NinjaEvaluationFlags( offset,
noTranslate, NinjaEvaluationFlags(evalFlags),
noRotate,
noScale,
hidden,
breakChildTrace,
zxyRotationOrder,
skip,
shapeSkip,
clip,
modifier,
),
model, model,
pos, pos,
rotation, rotation,

View File

@ -1,9 +1,12 @@
package world.phantasmal.lib.fileFormats.ninja package world.phantasmal.lib.fileFormats.ninja
import world.phantasmal.core.isBitSet
import world.phantasmal.core.setBit
import world.phantasmal.lib.fileFormats.Vec2 import world.phantasmal.lib.fileFormats.Vec2
import world.phantasmal.lib.fileFormats.Vec3 import world.phantasmal.lib.fileFormats.Vec3
sealed class NinjaObject<Model : NinjaModel, Self : NinjaObject<Model, Self>>( sealed class NinjaObject<Model : NinjaModel, Self : NinjaObject<Model, Self>>(
val offset: Int,
val evaluationFlags: NinjaEvaluationFlags, val evaluationFlags: NinjaEvaluationFlags,
val model: Model?, val model: Model?,
val position: Vec3, val position: Vec3,
@ -57,6 +60,7 @@ sealed class NinjaObject<Model : NinjaModel, Self : NinjaObject<Model, Self>>(
} }
class NjObject( class NjObject(
offset: Int,
evaluationFlags: NinjaEvaluationFlags, evaluationFlags: NinjaEvaluationFlags,
model: NjModel?, model: NjModel?,
position: Vec3, position: Vec3,
@ -64,6 +68,7 @@ class NjObject(
scale: Vec3, scale: Vec3,
children: MutableList<NjObject>, children: MutableList<NjObject>,
) : NinjaObject<NjModel, NjObject>( ) : NinjaObject<NjModel, NjObject>(
offset,
evaluationFlags, evaluationFlags,
model, model,
position, position,
@ -73,6 +78,7 @@ class NjObject(
) )
class XjObject( class XjObject(
offset: Int,
evaluationFlags: NinjaEvaluationFlags, evaluationFlags: NinjaEvaluationFlags,
model: XjModel?, model: XjModel?,
position: Vec3, position: Vec3,
@ -80,6 +86,7 @@ class XjObject(
scale: Vec3, scale: Vec3,
children: MutableList<XjObject>, children: MutableList<XjObject>,
) : NinjaObject<XjModel, XjObject>( ) : NinjaObject<XjModel, XjObject>(
offset,
evaluationFlags, evaluationFlags,
model, model,
position, position,
@ -88,18 +95,60 @@ class XjObject(
children, children,
) )
class NinjaEvaluationFlags( class NinjaEvaluationFlags(bits: Int) {
var noTranslate: Boolean, var bits: Int = bits
var noRotate: Boolean, private set
var noScale: Boolean, var noTranslate: Boolean
var hidden: Boolean, get() = bits.isBitSet(0)
var breakChildTrace: Boolean, set(value) {
var zxyRotationOrder: Boolean, bits = bits.setBit(0, value)
var skip: Boolean, }
var shapeSkip: Boolean, var noRotate: Boolean
val clip: Boolean, get() = bits.isBitSet(1)
val modifier: Boolean, set(value) {
) bits = bits.setBit(1, value)
}
var noScale: Boolean
get() = bits.isBitSet(2)
set(value) {
bits = bits.setBit(2, value)
}
var hidden: Boolean
get() = bits.isBitSet(3)
set(value) {
bits = bits.setBit(3, value)
}
var breakChildTrace: Boolean
get() = bits.isBitSet(4)
set(value) {
bits = bits.setBit(4, value)
}
var zxyRotationOrder: Boolean
get() = bits.isBitSet(5)
set(value) {
bits = bits.setBit(5, value)
}
var skip: Boolean
get() = bits.isBitSet(6)
set(value) {
bits = bits.setBit(6, value)
}
var shapeSkip: Boolean
get() = bits.isBitSet(7)
set(value) {
bits = bits.setBit(7, value)
}
var clip: Boolean
get() = bits.isBitSet(8)
set(value) {
bits = bits.setBit(8, value)
}
var modifier: Boolean
get() = bits.isBitSet(9)
set(value) {
bits = bits.setBit(9, value)
}
}
sealed class NinjaModel sealed class NinjaModel

View File

@ -4,10 +4,7 @@ import mu.KotlinLogging
import org.khronos.webgl.Float32Array import org.khronos.webgl.Float32Array
import org.khronos.webgl.Uint16Array import org.khronos.webgl.Uint16Array
import world.phantasmal.core.* import world.phantasmal.core.*
import world.phantasmal.lib.fileFormats.CollisionGeometry import world.phantasmal.lib.fileFormats.*
import world.phantasmal.lib.fileFormats.CollisionTriangle
import world.phantasmal.lib.fileFormats.RenderGeometry
import world.phantasmal.lib.fileFormats.RenderSection
import world.phantasmal.lib.fileFormats.ninja.* import world.phantasmal.lib.fileFormats.ninja.*
import world.phantasmal.web.core.dot import world.phantasmal.web.core.dot
import world.phantasmal.web.core.toQuaternion import world.phantasmal.web.core.toQuaternion
@ -76,6 +73,11 @@ private val tmpNormal = Vector3()
private val tmpVec = Vector3() private val tmpVec = Vector3()
private val tmpNormalMatrix = Matrix3() private val tmpNormalMatrix = Matrix3()
interface AreaObjectUserData {
var sectionId: Int
var areaObject: AreaObject
}
fun ninjaObjectToMesh( fun ninjaObjectToMesh(
ninjaObject: NinjaObject<*, *>, ninjaObject: NinjaObject<*, *>,
textures: List<XvrTexture?>, textures: List<XvrTexture?>,
@ -121,24 +123,36 @@ fun ninjaObjectToMeshBuilder(
} }
fun renderGeometryToGroup( fun renderGeometryToGroup(
renderGeometry: RenderGeometry, renderGeometry: AreaGeometry,
textures: List<XvrTexture?>, textures: List<XvrTexture?>,
processMesh: (RenderSection, XjObject, Mesh) -> Unit = { _, _, _ -> }, processMesh: (AreaSection, AreaObject, Mesh) -> Unit = { _, _, _ -> },
): Group { ): Group {
val group = Group() val group = Group()
val textureCache = emptyJsMap<Int, Texture?>() val textureCache = emptyJsMap<Int, Texture?>()
val meshCache = emptyJsMap<XjObject, Mesh>() val meshCache = emptyJsMap<XjObject, Mesh>()
for ((i, section) in renderGeometry.sections.withIndex()) { for ((sectionIndex, section) in renderGeometry.sections.withIndex()) {
for (xjObj in section.objects) { for (areaObj in section.objects) {
group.add(xjObjectToMesh( group.add(areaObjectToMesh(
textures, textureCache, meshCache, xjObj, i, section, processMesh, textures,
textureCache,
meshCache,
section,
sectionIndex,
areaObj,
processMesh,
)) ))
} }
for (xjObj in section.animatedObjects) { for (areaObj in section.animatedObjects) {
group.add(xjObjectToMesh( group.add(areaObjectToMesh(
textures, textureCache, meshCache, xjObj, i, section, processMesh, textures,
textureCache,
meshCache,
section,
sectionIndex,
areaObj,
processMesh,
)) ))
} }
} }
@ -146,41 +160,83 @@ fun renderGeometryToGroup(
return group return group
} }
private fun xjObjectToMesh( /**
* Calculates a fingerprint that can be used to match duplicated [AreaObject]s across sections, area
* variants and even areas.
*/
fun AreaObject.fingerPrint(): String =
buildString {
append(if (this@fingerPrint is AreaObject.Animated) 'a' else 's')
append('_')
var evalFlags = 0
var childCount = 0
var vertCount = 0
var meshCount = 0
var radius = 0f
fun recurse(xjObject: XjObject) {
evalFlags = evalFlags or xjObject.evaluationFlags.bits
childCount += xjObject.children.size
vertCount += xjObject.model?.vertices?.size ?: 0
meshCount += xjObject.model?.meshes?.size ?: 0
radius += xjObject.model?.collisionSphereRadius ?: 0f
xjObject.children.forEach(::recurse)
}
recurse(xjObject)
append(evalFlags.toString(36))
append('_')
append(childCount.toString(36))
append('_')
append(vertCount.toString(36))
append('_')
append(meshCount.toString(36))
append('_')
append(radius.reinterpretAsUInt().toString(36))
}
private fun areaObjectToMesh(
textures: List<XvrTexture?>, textures: List<XvrTexture?>,
textureCache: JsMap<Int, Texture?>, textureCache: JsMap<Int, Texture?>,
meshCache: JsMap<XjObject, Mesh>, meshCache: JsMap<XjObject, Mesh>,
xjObj: XjObject, section: AreaSection,
index: Int, sectionIndex: Int,
section: RenderSection, areaObj: AreaObject,
processMesh: (RenderSection, XjObject, Mesh) -> Unit, processMesh: (AreaSection, AreaObject, Mesh) -> Unit,
): Mesh { ): Mesh {
var mesh = meshCache.get(xjObj) var mesh = meshCache.get(areaObj.xjObject)
if (mesh == null) { if (mesh == null) {
val builder = MeshBuilder(textures, textureCache) val builder = MeshBuilder(textures, textureCache)
ninjaObjectToMeshBuilder(xjObj, builder) ninjaObjectToMeshBuilder(areaObj.xjObject, builder)
builder.defaultMaterial(MeshLambertMaterial(obj { builder.defaultMaterial(MeshLambertMaterial(obj {
color = Color().setHSL((index % 7) / 7.0, 1.0, .5) color = Color().setHSL((sectionIndex % 7) / 7.0, 1.0, .5)
transparent = true transparent = true
opacity = .5 opacity = .5
side = DoubleSide side = DoubleSide
})) }))
mesh = builder.buildMesh(boundingVolumes = true) mesh = builder.buildMesh(boundingVolumes = true)
meshCache.set(xjObj, mesh) meshCache.set(areaObj.xjObject, mesh)
} else { } else {
// If we already have a mesh for this XjObject, make a copy and reuse the existing buffer // If we already have a mesh for this XjObject, make a copy and reuse the existing buffer
// geometry and materials. // geometry and materials.
mesh = Mesh(mesh.geometry, mesh.material.unsafeCast<Array<Material>>()) mesh = Mesh(mesh.geometry, mesh.material.unsafeCast<Array<Material>>())
} }
val userData = mesh.userData.unsafeCast<AreaObjectUserData>()
userData.sectionId = section.id
userData.areaObject = areaObj
mesh.position.setFromVec3(section.position) mesh.position.setFromVec3(section.position)
mesh.rotation.setFromVec3(section.rotation) mesh.rotation.setFromVec3(section.rotation)
mesh.updateMatrixWorld() mesh.updateMatrixWorld()
processMesh(section, xjObj, mesh) processMesh(section, areaObj, mesh)
return mesh return mesh
} }

View File

@ -1,18 +1,14 @@
package world.phantasmal.web.questEditor.loading package world.phantasmal.web.questEditor.loading
import org.khronos.webgl.ArrayBuffer import org.khronos.webgl.ArrayBuffer
import world.phantasmal.core.asJsArray import world.phantasmal.core.*
import world.phantasmal.core.isBitSet
import world.phantasmal.lib.Endianness import world.phantasmal.lib.Endianness
import world.phantasmal.lib.Episode import world.phantasmal.lib.Episode
import world.phantasmal.lib.cursor.cursor import world.phantasmal.lib.cursor.cursor
import world.phantasmal.lib.fileFormats.CollisionGeometry import world.phantasmal.lib.fileFormats.*
import world.phantasmal.lib.fileFormats.RenderGeometry
import world.phantasmal.lib.fileFormats.ninja.XjObject import world.phantasmal.lib.fileFormats.ninja.XjObject
import world.phantasmal.lib.fileFormats.ninja.XvrTexture import world.phantasmal.lib.fileFormats.ninja.XvrTexture
import world.phantasmal.lib.fileFormats.ninja.parseXvm import world.phantasmal.lib.fileFormats.ninja.parseXvm
import world.phantasmal.lib.fileFormats.parseAreaCollisionGeometry
import world.phantasmal.lib.fileFormats.parseAreaRenderGeometry
import world.phantasmal.web.core.dot import world.phantasmal.web.core.dot
import world.phantasmal.web.core.loading.AssetLoader import world.phantasmal.web.core.loading.AssetLoader
import world.phantasmal.web.core.rendering.conversion.* import world.phantasmal.web.core.rendering.conversion.*
@ -123,44 +119,44 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine
} }
} }
private fun cullRenderGeometry(collisionGeom: Object3D, renderGeom: Object3D) { // private fun cullRenderGeometry(collisionGeom: Object3D, renderGeom: Object3D) {
val cullingVolumes = mutableMapOf<Int, Box3>() // val cullingVolumes = mutableMapOf<Int, Box3>()
//
for (collisionMesh in collisionGeom.children) { // for (collisionMesh in collisionGeom.children) {
collisionMesh as Mesh // collisionMesh as Mesh
collisionMesh.userData.unsafeCast<AreaUserData>().section?.let { section -> // collisionMesh.userData.unsafeCast<AreaUserData>().section?.let { section ->
cullingVolumes.getOrPut(section.id, ::Box3) // cullingVolumes.getOrPut(section.id, ::Box3)
.union( // .union(
collisionMesh.geometry.boundingBox!!.applyMatrix4(collisionMesh.matrixWorld) // collisionMesh.geometry.boundingBox!!.applyMatrix4(collisionMesh.matrixWorld)
) // )
} // }
} // }
//
for (cullingVolume in cullingVolumes.values) { // for (cullingVolume in cullingVolumes.values) {
cullingVolume.min.x -= 50 // cullingVolume.min.x -= 50
cullingVolume.min.y = cullingVolume.max.y + 20 // cullingVolume.min.y = cullingVolume.max.y + 20
cullingVolume.min.z -= 50 // cullingVolume.min.z -= 50
cullingVolume.max.x += 50 // cullingVolume.max.x += 50
cullingVolume.max.y = Double.POSITIVE_INFINITY // cullingVolume.max.y = Double.POSITIVE_INFINITY
cullingVolume.max.z += 50 // cullingVolume.max.z += 50
} // }
//
var i = 0 // var i = 0
//
outer@ while (i < renderGeom.children.size) { // outer@ while (i < renderGeom.children.size) {
val renderMesh = renderGeom.children[i] as Mesh // val renderMesh = renderGeom.children[i] as Mesh
val bb = renderMesh.geometry.boundingBox!!.applyMatrix4(renderMesh.matrixWorld) // val bb = renderMesh.geometry.boundingBox!!.applyMatrix4(renderMesh.matrixWorld)
//
for (cullingVolume in cullingVolumes.values) { // for (cullingVolume in cullingVolumes.values) {
if (bb.intersectsBox(cullingVolume)) { // if (bb.intersectsBox(cullingVolume)) {
renderGeom.remove(renderMesh) // renderGeom.remove(renderMesh)
continue@outer // continue@outer
} // }
} // }
//
i++ // i++
} // }
} // }
private fun areaAssetUrl( private fun areaAssetUrl(
episode: Episode, episode: Episode,
@ -206,20 +202,27 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine
} }
private fun areaGeometryToObject3DAndSections( private fun areaGeometryToObject3DAndSections(
renderGeometry: RenderGeometry, renderGeometry: AreaGeometry,
textures: List<XvrTexture>, textures: List<XvrTexture>,
episode: Episode, episode: Episode,
areaVariant: AreaVariantModel, areaVariant: AreaVariantModel,
): Pair<Object3D, List<SectionModel>> { ): Pair<Object3D, List<SectionModel>> {
val renderOnTopTextures = RENDER_ON_TOP_TEXTURES[Pair(episode, areaVariant.area.id)] val fix = MANUAL_FIXES[Pair(episode, areaVariant.area.id)]
val sections = mutableMapOf<Int, SectionModel>() val sections = mutableMapOf<Int, SectionModel>()
console.log(renderGeometry)
val group = val group =
renderGeometryToGroup(renderGeometry, textures) { renderSection, xjObject, mesh -> renderGeometryToGroup(renderGeometry, textures) { renderSection, areaObj, mesh ->
if (shouldRenderOnTop(xjObject, renderOnTopTextures)) { if (fix != null) {
if (fix.shouldRenderOnTop(areaObj.xjObject)) {
mesh.renderOrder = 1 mesh.renderOrder = 1
} }
if (fix.shouldHide(areaObj)) {
mesh.visible = false
}
}
if (renderSection.id >= 0) { if (renderSection.id >= 0) {
val sectionModel = sections.getOrPut(renderSection.id) { val sectionModel = sections.getOrPut(renderSection.id) {
SectionModel( SectionModel(
@ -237,24 +240,6 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine
return Pair(group, sections.values.toList()) return Pair(group, sections.values.toList())
} }
private fun shouldRenderOnTop(obj: XjObject, renderOnTopTextures: Set<Int>?): Boolean {
renderOnTopTextures ?: return false
obj.model?.meshes?.let { meshes ->
for (mesh in meshes) {
mesh.material.textureId?.let { textureId ->
if (textureId in renderOnTopTextures) {
return true
}
}
}
}
return obj.children.any {
shouldRenderOnTop(it, renderOnTopTextures)
}
}
private fun areaCollisionGeometryToObject3D( private fun areaCollisionGeometryToObject3D(
obj: CollisionGeometry, obj: CollisionGeometry,
episode: Episode, episode: Episode,
@ -288,6 +273,38 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine
Render, Collision, Texture Render, Collision, Texture
} }
private class Fix(
/**
* Textures that should be rendered on top of other textures. These are usually very
* translucent. E.g. forest 1 has a mesh with baked-in shadow that's overlaid over the
* regular geometry. Might not be necessary anymore once order-independent rendering is
* implemented.
*/
private val renderOnTopTextures: JsSet<Int> = emptyJsSet(),
/**
* Set of [AreaObject] finger prints.
* These objects should be hidden because they cover floors and other useful geometry.
*/
private val hiddenObjects: JsSet<String> = emptyJsSet(),
) {
fun shouldRenderOnTop(obj: XjObject): Boolean {
obj.model?.meshes?.let { meshes ->
for (mesh in meshes) {
mesh.material.textureId?.let { textureId ->
if (renderOnTopTextures.has(textureId)) {
return true
}
}
}
}
return obj.children.any(::shouldRenderOnTop)
}
fun shouldHide(areaObj: AreaObject): Boolean =
hiddenObjects.has(areaObj.fingerPrint())
}
companion object { companion object {
private val COS_75_DEG = cos(PI / 180 * 75) private val COS_75_DEG = cos(PI / 180 * 75)
private val DOWN = Vector3(.0, -1.0, .0) private val DOWN = Vector3(.0, -1.0, .0)
@ -346,26 +363,178 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine
) )
/** /**
* Mapping of episode and area ID to set of texture IDs. * Mapping of episode and area ID to data for manually fixing issues with the render
* Manual fixes for various areas. Might not be necessary anymore once order-independent * geometry.
* rendering is implemented.
*/ */
val RENDER_ON_TOP_TEXTURES: Map<Pair<Episode, Int>, Set<Int>> = mapOf( private val MANUAL_FIXES: Map<Pair<Episode, Int>, Fix> = mutableMapOf(
// Pioneer 2 // Pioneer 2
Pair(Episode.I, 0) to setOf( Pair(Episode.I, 0) to Fix(
renderOnTopTextures = jsSetOf(
70, 71, 72, 126, 127, 155, 156, 198, 230, 231, 232, 233, 234, 70, 71, 72, 126, 127, 155, 156, 198, 230, 231, 232, 233, 234,
), ),
hiddenObjects = jsSetOf(
"s_m_0_6a_d_iu5sg6",
"s_m_0_4b_7_ioh738",
"s_k_0_1s_3_irasis",
"s_k_0_a_1_ir4eod",
"s_n_0_9e_h_imjyqr", // Hunter Guild roof + walls (seems to remove slightly too much).
"s_n_0_40_a_it58n7", // Neon signs throughout the city.
"s_n_0_2m_1_isvawv",
"s_n_0_o_1_iwk2nr",
"a_n_0_2k_5_iyebd3",
"s_n_0_4_1_ikyjfd",
"s_n_0_g_1_iom8uk",
"s_n_0_j5_b_ivdcj1",
"s_n_0_28_1_iopx1k",
"s_m_0_3q_6_iqmvjr",
"s_m_0_26_2_inh1ma",
"s_m_0_4b_4_immz8l",
"s_m_0_22_2_ilwnn5",
"s_m_0_84_e_iv6noc",
"s_m_0_d_1_ili3v2",
"s_m_0_58_2_igd0am",
"s_m_0_25_3_iovf21",
"s_n_0_8_1_ik11uc",
"s_m_0_19_1_ijocvh",
"s_m_0_2h_5_is8o4b",
"s_m_0_1l_4_ilkky7",
"s_m_0_35_1_il8hoa",
"s_m_0_58_3_in4nwl",
"s_m_0_3d_1_iro50a",
"s_m_0_4_1_is53va",
"s_m_0_3l_6_igzvga",
"s_n_0_en_3_iiawrz",
),
),
// Forest 1 // Forest 1
Pair(Episode.I, 1) to setOf(12, 41), Pair(Episode.I, 1) to Fix(
renderOnTopTextures = jsSetOf(12, 41),
),
// Cave 1
Pair(Episode.I, 3) to Fix(
hiddenObjects = jsSetOf(
"s_n_0_8_1_iqrqjj",
"s_i_0_b5_1_is7ajh",
"s_n_0_24_1_in5ce2",
"s_n_0_u_3_im4944",
"s_n_0_1b_2_im4945",
"s_n_0_2b_1_iktmat",
"s_n_0_3c_1_iksavp",
"s_n_0_31_1_ijhyzw",
"s_n_0_2i_3_ik3g7o",
"s_n_0_39_1_ix3ls0",
"s_n_0_37_1_ix3nxi",
"s_n_0_8x_1_iw2lqw",
"s_n_0_8w_1_ivx9ro",
"s_n_0_2c_1_itkfue",
"s_n_0_2u_1_iuilbk",
"s_n_0_30_1_ivmffx",
"s_n_0_2o_1_iu42tg",
"s_n_0_1u_1_ipk1qq",
"s_n_0_3i_1_iuz9mq",
"s_n_0_36_1_itm5fi",
"s_n_0_2o_1_ircjgr",
"s_n_0_3i_1_iurb4o",
"s_n_0_22_1_ii9035",
"s_n_0_2i_3_iiqupy",
),
),
// Cave 2
Pair(Episode.I, 4) to Fix(
hiddenObjects = jsSetOf(
"s_n_0_4j_1_irf90i",
"s_n_0_5i_1_iqqrft",
"s_n_0_g_1_iipv9r",
"s_n_0_c_1_ihboen",
),
),
// Cave 3
Pair(Episode.I, 5) to Fix(
hiddenObjects = jsSetOf(
"s_n_0_2o_5_inun1c",
"s_n_0_5y_2_ipyair",
),
),
// Mine 1
Pair(Episode.I, 6) to Fix(
hiddenObjects = jsSetOf(
"s_n_0_2e_2_iqfpg8",
"s_n_0_d_1_iruof6",
"s_n_0_o_1_im9ta5",
"s_n_0_18_3_im1kwg",
),
),
// Mine 2 // Mine 2
Pair(Episode.I, 7) to setOf(0, 1, 7, 8, 17, 23, 56, 57, 58, 59, 60, 83), Pair(Episode.I, 7) to Fix(
renderOnTopTextures = jsSetOf(0, 1, 7, 8, 17, 23, 56, 57, 58, 59, 60, 83),
),
// Ruins 1 // Ruins 1
Pair(Episode.I, 8) to setOf(1, 21, 22, 27, 28, 43, 51, 59, 70, 72, 75), Pair(Episode.I, 8) to Fix(
renderOnTopTextures = jsSetOf(1, 21, 22, 27, 28, 43, 51, 59, 70, 72, 75),
hiddenObjects = jsSetOf(
"s_n_0_2p_4_iohs6r",
"s_n_0_2q_4_iohs6r",
"s_m_0_l_1_io448k",
),
),
// Ruins 2
Pair(Episode.I, 9) to Fix(
hiddenObjects = jsSetOf(
"s_m_0_l_1_io448k",
),
),
// Lab // Lab
Pair(Episode.II, 0) to setOf(36, 37, 38, 48, 60, 67, 79, 80), Pair(Episode.II, 0) to Fix(
renderOnTopTextures = jsSetOf(36, 37, 38, 48, 60, 67, 79, 80),
),
// VR Spaceship Alpha
Pair(Episode.II, 3) to Fix(
renderOnTopTextures = jsSetOf(7, 59),
hiddenObjects = jsSetOf(
"s_l_0_45_5_ing07n",
"s_n_0_45_5_ing07k",
"s_n_0_g2_b_im2en1",
"s_n_0_3j_1_irr4qe",
"s_n_0_bp_8_irbqmy",
"s_n_0_4h_1_irkudv",
"s_n_0_4g_1_irkudv",
"s_n_0_l_1_ijtl6r",
"s_n_0_l_1_ijtl6u",
"s_n_0_1s_1_imgj8o",
"s_n_0_r_1_ijua1b",
"s_n_0_g0_c_ilpett",
"s_n_0_16_1_igxq22",
"s_n_0_1c_1_imgj8o",
"s_n_0_1c_1_imgj8p",
"s_n_0_1u_1_imgj8o",
"s_n_0_1u_1_imgj8p",
"s_n_0_20_1_im13wb",
"s_n_0_12_1_ilsbgy",
"s_n_0_8_1_ihmjxh",
"s_n_0_1u_1_imv5rn",
"s_i_0_2d_4_ir3kzk",
"s_g_0_2d_4_ir3kzk",
"s_n_0_1t_1_imgj8o",
"s_n_0_l_1_ijoqlv",
"s_m_0_c_1_iayi9w",
"s_k_0_c_1_iayi9w",
"s_n_0_gl_8_imtj35",
"s_n_0_gc_8_imtj35",
"s_n_0_g_1_ildjm9",
),
),
// Central Control Area // Central Control Area
Pair(Episode.II, 5) to (0..59).toSet() + setOf(69, 77), Pair(Episode.II, 5) to Fix(
) renderOnTopTextures = jsSetOf(*((0..59).toSet() + setOf(69, 77)).toTypedArray()),
),
// Jungle Area East
Pair(Episode.II, 6) to Fix(
renderOnTopTextures = jsSetOf(0, 1, 2, 18, 21, 24),
),
).also {
// VR Spaceship Beta = VR Spaceship Alpha
it[Pair(Episode.II, 4)] = it[Pair(Episode.II, 3)]!!
}
private val raycaster = Raycaster() private val raycaster = Raycaster()
private val tmpVec = Vector3() private val tmpVec = Vector3()

View File

@ -3,8 +3,8 @@ package world.phantasmal.web.viewer.stores
import kotlinx.coroutines.launch import kotlinx.coroutines.launch
import mu.KotlinLogging import mu.KotlinLogging
import world.phantasmal.core.enumValueOfOrNull import world.phantasmal.core.enumValueOfOrNull
import world.phantasmal.lib.fileFormats.AreaGeometry
import world.phantasmal.lib.fileFormats.CollisionGeometry import world.phantasmal.lib.fileFormats.CollisionGeometry
import world.phantasmal.lib.fileFormats.RenderGeometry
import world.phantasmal.lib.fileFormats.ninja.NinjaObject import world.phantasmal.lib.fileFormats.ninja.NinjaObject
import world.phantasmal.lib.fileFormats.ninja.NjMotion import world.phantasmal.lib.fileFormats.ninja.NjMotion
import world.phantasmal.lib.fileFormats.ninja.NjObject import world.phantasmal.lib.fileFormats.ninja.NjObject
@ -29,7 +29,7 @@ private val logger = KotlinLogging.logger {}
sealed class NinjaGeometry { sealed class NinjaGeometry {
class Object(val obj: NinjaObject<*, *>) : NinjaGeometry() class Object(val obj: NinjaObject<*, *>) : NinjaGeometry()
class Render(val geometry: RenderGeometry) : NinjaGeometry() class Render(val geometry: AreaGeometry) : NinjaGeometry()
class Collision(val geometry: CollisionGeometry) : NinjaGeometry() class Collision(val geometry: CollisionGeometry) : NinjaGeometry()
} }