mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-04 06:28:28 +08:00
Added manual area render geometry culling code.
This commit is contained in:
parent
126b50cb00
commit
29192e5684
@ -20,6 +20,15 @@ fun Int.isBitSet(bit: Int): Boolean =
|
||||
fun UByte.isBitSet(bit: Int): Boolean =
|
||||
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 Float.reinterpretAsInt(): Int
|
||||
|
||||
expect fun Float.reinterpretAsUInt(): UInt
|
||||
|
@ -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>.component2(): T = second
|
||||
|
||||
@Suppress("UNUSED_PARAMETER")
|
||||
inline fun objectKeys(jsObject: dynamic): Array<String> =
|
||||
js("Object.keys(jsObject)").unsafeCast<Array<String>>()
|
||||
@Suppress("FunctionName", "UNUSED_PARAMETER")
|
||||
inline fun <A, B> JsPair(first: A, second: B): JsPair<A, B> =
|
||||
js("[first, second]").unsafeCast<JsPair<A, B>>()
|
||||
|
||||
@Suppress("UNUSED_PARAMETER")
|
||||
inline fun objectEntries(jsObject: dynamic): Array<JsPair<String, dynamic>> =
|
||||
@ -64,6 +64,10 @@ external interface JsSet<T> {
|
||||
inline fun <T> emptyJsSet(): 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> {
|
||||
val size: Int
|
||||
|
||||
@ -75,5 +79,9 @@ external interface 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> =
|
||||
js("new Map()").unsafeCast<JsMap<K, V>>()
|
||||
|
@ -14,3 +14,8 @@ actual fun Float.reinterpretAsInt(): Int {
|
||||
dataView.setFloat32(0, this)
|
||||
return dataView.getInt32(0)
|
||||
}
|
||||
|
||||
actual fun Float.reinterpretAsUInt(): UInt {
|
||||
dataView.setFloat32(0, this)
|
||||
return dataView.getUint32(0).toUInt()
|
||||
}
|
||||
|
@ -8,3 +8,5 @@ import java.lang.Float.intBitsToFloat
|
||||
actual fun Int.reinterpretAsFloat(): Float = intBitsToFloat(this)
|
||||
|
||||
actual fun Float.reinterpretAsInt(): Int = floatToIntBits(this)
|
||||
|
||||
actual fun Float.reinterpretAsUInt(): UInt = reinterpretAsInt().toUInt()
|
||||
|
@ -1,40 +1,69 @@
|
||||
package world.phantasmal.lib.fileFormats
|
||||
|
||||
import mu.KotlinLogging
|
||||
import world.phantasmal.core.isBitSet
|
||||
import world.phantasmal.lib.cursor.Cursor
|
||||
import world.phantasmal.lib.fileFormats.ninja.XjObject
|
||||
import world.phantasmal.lib.fileFormats.ninja.angleToRad
|
||||
import world.phantasmal.lib.fileFormats.ninja.parseXjObject
|
||||
import world.phantasmal.lib.fileFormats.ninja.*
|
||||
|
||||
class RenderGeometry(
|
||||
val sections: List<RenderSection>,
|
||||
private val logger = KotlinLogging.logger {}
|
||||
|
||||
class AreaGeometry(
|
||||
val sections: List<AreaSection>,
|
||||
)
|
||||
|
||||
class RenderSection(
|
||||
class AreaSection(
|
||||
val id: Int,
|
||||
val position: Vec3,
|
||||
val rotation: Vec3,
|
||||
val objects: List<XjObject>,
|
||||
val animatedObjects: List<XjObject>,
|
||||
val radius: Float,
|
||||
val objects: List<AreaObject.Simple>,
|
||||
val animatedObjects: List<AreaObject.Animated>,
|
||||
)
|
||||
|
||||
fun parseAreaRenderGeometry(cursor: Cursor): RenderGeometry {
|
||||
val sections = mutableListOf<RenderSection>()
|
||||
sealed class AreaObject {
|
||||
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
|
||||
|
||||
cursor.seekStart(dataOffset)
|
||||
cursor.seek(8) // Format "fmt2" in UTF-16.
|
||||
val sectionCount = cursor.int()
|
||||
val format = cursor.stringAscii(maxByteLength = 4, nullTerminated = true, dropRemaining = true)
|
||||
|
||||
if (format != "fmt2") {
|
||||
logger.warn { """Expected format to be "fmt2" but was "$format".""" }
|
||||
}
|
||||
|
||||
cursor.seek(4)
|
||||
val sectionTableOffset = cursor.int()
|
||||
// val textureNameOffset = cursor.int()
|
||||
val sectionsCount = 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) {
|
||||
cursor.seekStart(sectionTableOffset + 52 * i)
|
||||
// Cache keys are offsets.
|
||||
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 sectionPosition = cursor.vec3Float()
|
||||
@ -44,67 +73,110 @@ fun parseAreaRenderGeometry(cursor: Cursor): RenderGeometry {
|
||||
angleToRad(cursor.int()),
|
||||
)
|
||||
|
||||
cursor.seek(4) // Radius?
|
||||
val radius = cursor.float()
|
||||
|
||||
val simpleGeometryOffsetTableOffset = cursor.int()
|
||||
val animatedGeometryOffsetTableOffset = cursor.int()
|
||||
val simpleGeometryOffsetCount = cursor.int()
|
||||
val animatedGeometryOffsetCount = cursor.int()
|
||||
val simpleAreaObjectsOffset = cursor.int()
|
||||
val animatedAreaObjectsOffset = cursor.int()
|
||||
val simpleAreaObjectsCount = cursor.int()
|
||||
val animatedAreaObjectsCount = cursor.int()
|
||||
// Ignore the last 4 bytes.
|
||||
|
||||
val objects = parseGeometryTable(
|
||||
cursor,
|
||||
xjObjectCache,
|
||||
simpleGeometryOffsetTableOffset,
|
||||
simpleGeometryOffsetCount,
|
||||
animated = false,
|
||||
)
|
||||
// println("section $sectionId (index $i), simple geom at $simpleGeometryTableOffset, animated geom at $animatedGeometryTableOffset")
|
||||
|
||||
val animatedObjects = parseGeometryTable(
|
||||
cursor,
|
||||
xjObjectCache,
|
||||
animatedGeometryOffsetTableOffset,
|
||||
animatedGeometryOffsetCount,
|
||||
animated = true,
|
||||
)
|
||||
@Suppress("UNCHECKED_CAST")
|
||||
val simpleObjects = simpleAreaObjectCache.getOrPut(simpleAreaObjectsOffset) {
|
||||
parseAreaObjects(
|
||||
cursor,
|
||||
njMotionCache,
|
||||
simpleAreaObjectsOffset,
|
||||
simpleAreaObjectsCount,
|
||||
animated = false,
|
||||
) as List<AreaObject.Simple>
|
||||
}
|
||||
|
||||
sections.add(RenderSection(
|
||||
@Suppress("UNCHECKED_CAST")
|
||||
val animatedObjects = animatedAreaObjectCache.getOrPut(animatedAreaObjectsOffset) {
|
||||
parseAreaObjects(
|
||||
cursor,
|
||||
njMotionCache,
|
||||
animatedAreaObjectsOffset,
|
||||
animatedAreaObjectsCount,
|
||||
animated = true,
|
||||
) as List<AreaObject.Animated>
|
||||
}
|
||||
|
||||
sections.add(AreaSection(
|
||||
sectionId,
|
||||
sectionPosition,
|
||||
sectionRotation,
|
||||
objects,
|
||||
radius,
|
||||
simpleObjects,
|
||||
animatedObjects,
|
||||
))
|
||||
}
|
||||
|
||||
return RenderGeometry(sections)
|
||||
return AreaGeometry(sections)
|
||||
}
|
||||
|
||||
private fun parseGeometryTable(
|
||||
private fun parseAreaObjects(
|
||||
cursor: Cursor,
|
||||
xjObjectCache: MutableMap<Int, List<XjObject>>,
|
||||
tableOffset: Int,
|
||||
tableEntryCount: Int,
|
||||
njMotionCache: MutableMap<Int, NjMotion>,
|
||||
offset: Int,
|
||||
count: Int,
|
||||
animated: Boolean,
|
||||
): List<XjObject> {
|
||||
val tableEntrySize = if (animated) 32 else 16
|
||||
val objects = mutableListOf<XjObject>()
|
||||
): List<AreaObject> {
|
||||
val objectSize = if (animated) 32 else 16
|
||||
val objects = mutableListOf<AreaObject>()
|
||||
|
||||
for (i in 0 until tableEntryCount) {
|
||||
cursor.seekStart(tableOffset + tableEntrySize * i)
|
||||
for (i in 0 until count) {
|
||||
val objectOffset = offset + objectSize * i
|
||||
cursor.seekStart(objectOffset)
|
||||
|
||||
var xjObjectOffset = cursor.int()
|
||||
val speed: Float?
|
||||
val njMotionOffset: Int?
|
||||
|
||||
if (animated) {
|
||||
njMotionOffset = cursor.int()
|
||||
cursor.seek(8)
|
||||
speed = cursor.float()
|
||||
} else {
|
||||
speed = null
|
||||
njMotionOffset = null
|
||||
}
|
||||
|
||||
val slideTextureIdOffset = cursor.int()
|
||||
val swapTextureIdOffset = cursor.int()
|
||||
|
||||
var offset = cursor.int()
|
||||
cursor.seek(8)
|
||||
val flags = cursor.int()
|
||||
|
||||
if (flags.isBitSet(2)) {
|
||||
offset = cursor.seekStart(offset).int()
|
||||
xjObjectOffset = cursor.seekStart(xjObjectOffset).int()
|
||||
}
|
||||
|
||||
objects.addAll(
|
||||
xjObjectCache.getOrPut(offset) {
|
||||
cursor.seekStart(offset)
|
||||
parseXjObject(cursor)
|
||||
cursor.seekStart(xjObjectOffset)
|
||||
val xjObjects = 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)
|
||||
}
|
||||
)
|
||||
}
|
||||
|
@ -101,13 +101,13 @@ private fun parseAction(cursor: Cursor): NjMotion {
|
||||
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
|
||||
// value. This is usually the value that the first mDataOffset points to. This value is assumed
|
||||
// to be the end of the mDataOffset table.
|
||||
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 frameCount = cursor.int()
|
||||
val type = cursor.uShort().toInt()
|
||||
|
@ -3,7 +3,6 @@ package world.phantasmal.lib.fileFormats.ninja
|
||||
import world.phantasmal.core.Failure
|
||||
import world.phantasmal.core.PwResult
|
||||
import world.phantasmal.core.Success
|
||||
import world.phantasmal.core.isBitSet
|
||||
import world.phantasmal.lib.cursor.Cursor
|
||||
import world.phantasmal.lib.fileFormats.Vec3
|
||||
import world.phantasmal.lib.fileFormats.parseIff
|
||||
@ -21,6 +20,7 @@ fun parseXjObject(cursor: Cursor): List<XjObject> =
|
||||
parseSiblingObjects(cursor, { c, _ -> parseXjModel(c) }, ::XjObject, Unit)
|
||||
|
||||
private typealias CreateObject<Model, Obj> = (
|
||||
offset: Int,
|
||||
evaluationFlags: NinjaEvaluationFlags,
|
||||
model: Model?,
|
||||
position: Vec3,
|
||||
@ -57,17 +57,8 @@ private fun <Model : NinjaModel, Obj : NinjaObject<Model, Obj>, Context> parseSi
|
||||
createObject: CreateObject<Model, Obj>,
|
||||
context: Context,
|
||||
): MutableList<Obj> {
|
||||
val offset = cursor.position
|
||||
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 pos = cursor.vec3Float()
|
||||
@ -102,18 +93,8 @@ private fun <Model : NinjaModel, Obj : NinjaObject<Model, Obj>, Context> parseSi
|
||||
}
|
||||
|
||||
val obj = createObject(
|
||||
NinjaEvaluationFlags(
|
||||
noTranslate,
|
||||
noRotate,
|
||||
noScale,
|
||||
hidden,
|
||||
breakChildTrace,
|
||||
zxyRotationOrder,
|
||||
skip,
|
||||
shapeSkip,
|
||||
clip,
|
||||
modifier,
|
||||
),
|
||||
offset,
|
||||
NinjaEvaluationFlags(evalFlags),
|
||||
model,
|
||||
pos,
|
||||
rotation,
|
||||
|
@ -1,9 +1,12 @@
|
||||
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.Vec3
|
||||
|
||||
sealed class NinjaObject<Model : NinjaModel, Self : NinjaObject<Model, Self>>(
|
||||
val offset: Int,
|
||||
val evaluationFlags: NinjaEvaluationFlags,
|
||||
val model: Model?,
|
||||
val position: Vec3,
|
||||
@ -57,6 +60,7 @@ sealed class NinjaObject<Model : NinjaModel, Self : NinjaObject<Model, Self>>(
|
||||
}
|
||||
|
||||
class NjObject(
|
||||
offset: Int,
|
||||
evaluationFlags: NinjaEvaluationFlags,
|
||||
model: NjModel?,
|
||||
position: Vec3,
|
||||
@ -64,6 +68,7 @@ class NjObject(
|
||||
scale: Vec3,
|
||||
children: MutableList<NjObject>,
|
||||
) : NinjaObject<NjModel, NjObject>(
|
||||
offset,
|
||||
evaluationFlags,
|
||||
model,
|
||||
position,
|
||||
@ -73,6 +78,7 @@ class NjObject(
|
||||
)
|
||||
|
||||
class XjObject(
|
||||
offset: Int,
|
||||
evaluationFlags: NinjaEvaluationFlags,
|
||||
model: XjModel?,
|
||||
position: Vec3,
|
||||
@ -80,6 +86,7 @@ class XjObject(
|
||||
scale: Vec3,
|
||||
children: MutableList<XjObject>,
|
||||
) : NinjaObject<XjModel, XjObject>(
|
||||
offset,
|
||||
evaluationFlags,
|
||||
model,
|
||||
position,
|
||||
@ -88,18 +95,60 @@ class XjObject(
|
||||
children,
|
||||
)
|
||||
|
||||
class NinjaEvaluationFlags(
|
||||
var noTranslate: Boolean,
|
||||
var noRotate: Boolean,
|
||||
var noScale: Boolean,
|
||||
var hidden: Boolean,
|
||||
var breakChildTrace: Boolean,
|
||||
var zxyRotationOrder: Boolean,
|
||||
var skip: Boolean,
|
||||
var shapeSkip: Boolean,
|
||||
val clip: Boolean,
|
||||
val modifier: Boolean,
|
||||
)
|
||||
class NinjaEvaluationFlags(bits: Int) {
|
||||
var bits: Int = bits
|
||||
private set
|
||||
var noTranslate: Boolean
|
||||
get() = bits.isBitSet(0)
|
||||
set(value) {
|
||||
bits = bits.setBit(0, value)
|
||||
}
|
||||
var noRotate: Boolean
|
||||
get() = bits.isBitSet(1)
|
||||
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
|
||||
|
||||
|
@ -4,10 +4,7 @@ import mu.KotlinLogging
|
||||
import org.khronos.webgl.Float32Array
|
||||
import org.khronos.webgl.Uint16Array
|
||||
import world.phantasmal.core.*
|
||||
import world.phantasmal.lib.fileFormats.CollisionGeometry
|
||||
import world.phantasmal.lib.fileFormats.CollisionTriangle
|
||||
import world.phantasmal.lib.fileFormats.RenderGeometry
|
||||
import world.phantasmal.lib.fileFormats.RenderSection
|
||||
import world.phantasmal.lib.fileFormats.*
|
||||
import world.phantasmal.lib.fileFormats.ninja.*
|
||||
import world.phantasmal.web.core.dot
|
||||
import world.phantasmal.web.core.toQuaternion
|
||||
@ -76,6 +73,11 @@ private val tmpNormal = Vector3()
|
||||
private val tmpVec = Vector3()
|
||||
private val tmpNormalMatrix = Matrix3()
|
||||
|
||||
interface AreaObjectUserData {
|
||||
var sectionId: Int
|
||||
var areaObject: AreaObject
|
||||
}
|
||||
|
||||
fun ninjaObjectToMesh(
|
||||
ninjaObject: NinjaObject<*, *>,
|
||||
textures: List<XvrTexture?>,
|
||||
@ -121,24 +123,36 @@ fun ninjaObjectToMeshBuilder(
|
||||
}
|
||||
|
||||
fun renderGeometryToGroup(
|
||||
renderGeometry: RenderGeometry,
|
||||
renderGeometry: AreaGeometry,
|
||||
textures: List<XvrTexture?>,
|
||||
processMesh: (RenderSection, XjObject, Mesh) -> Unit = { _, _, _ -> },
|
||||
processMesh: (AreaSection, AreaObject, Mesh) -> Unit = { _, _, _ -> },
|
||||
): Group {
|
||||
val group = Group()
|
||||
val textureCache = emptyJsMap<Int, Texture?>()
|
||||
val meshCache = emptyJsMap<XjObject, Mesh>()
|
||||
|
||||
for ((i, section) in renderGeometry.sections.withIndex()) {
|
||||
for (xjObj in section.objects) {
|
||||
group.add(xjObjectToMesh(
|
||||
textures, textureCache, meshCache, xjObj, i, section, processMesh,
|
||||
for ((sectionIndex, section) in renderGeometry.sections.withIndex()) {
|
||||
for (areaObj in section.objects) {
|
||||
group.add(areaObjectToMesh(
|
||||
textures,
|
||||
textureCache,
|
||||
meshCache,
|
||||
section,
|
||||
sectionIndex,
|
||||
areaObj,
|
||||
processMesh,
|
||||
))
|
||||
}
|
||||
|
||||
for (xjObj in section.animatedObjects) {
|
||||
group.add(xjObjectToMesh(
|
||||
textures, textureCache, meshCache, xjObj, i, section, processMesh,
|
||||
for (areaObj in section.animatedObjects) {
|
||||
group.add(areaObjectToMesh(
|
||||
textures,
|
||||
textureCache,
|
||||
meshCache,
|
||||
section,
|
||||
sectionIndex,
|
||||
areaObj,
|
||||
processMesh,
|
||||
))
|
||||
}
|
||||
}
|
||||
@ -146,41 +160,83 @@ fun renderGeometryToGroup(
|
||||
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?>,
|
||||
textureCache: JsMap<Int, Texture?>,
|
||||
meshCache: JsMap<XjObject, Mesh>,
|
||||
xjObj: XjObject,
|
||||
index: Int,
|
||||
section: RenderSection,
|
||||
processMesh: (RenderSection, XjObject, Mesh) -> Unit,
|
||||
section: AreaSection,
|
||||
sectionIndex: Int,
|
||||
areaObj: AreaObject,
|
||||
processMesh: (AreaSection, AreaObject, Mesh) -> Unit,
|
||||
): Mesh {
|
||||
var mesh = meshCache.get(xjObj)
|
||||
var mesh = meshCache.get(areaObj.xjObject)
|
||||
|
||||
if (mesh == null) {
|
||||
val builder = MeshBuilder(textures, textureCache)
|
||||
ninjaObjectToMeshBuilder(xjObj, builder)
|
||||
ninjaObjectToMeshBuilder(areaObj.xjObject, builder)
|
||||
|
||||
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
|
||||
opacity = .5
|
||||
side = DoubleSide
|
||||
}))
|
||||
|
||||
mesh = builder.buildMesh(boundingVolumes = true)
|
||||
meshCache.set(xjObj, mesh)
|
||||
meshCache.set(areaObj.xjObject, mesh)
|
||||
} 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>>())
|
||||
}
|
||||
|
||||
val userData = mesh.userData.unsafeCast<AreaObjectUserData>()
|
||||
userData.sectionId = section.id
|
||||
userData.areaObject = areaObj
|
||||
|
||||
mesh.position.setFromVec3(section.position)
|
||||
mesh.rotation.setFromVec3(section.rotation)
|
||||
mesh.updateMatrixWorld()
|
||||
|
||||
processMesh(section, xjObj, mesh)
|
||||
processMesh(section, areaObj, mesh)
|
||||
|
||||
return mesh
|
||||
}
|
||||
|
@ -1,18 +1,14 @@
|
||||
package world.phantasmal.web.questEditor.loading
|
||||
|
||||
import org.khronos.webgl.ArrayBuffer
|
||||
import world.phantasmal.core.asJsArray
|
||||
import world.phantasmal.core.isBitSet
|
||||
import world.phantasmal.core.*
|
||||
import world.phantasmal.lib.Endianness
|
||||
import world.phantasmal.lib.Episode
|
||||
import world.phantasmal.lib.cursor.cursor
|
||||
import world.phantasmal.lib.fileFormats.CollisionGeometry
|
||||
import world.phantasmal.lib.fileFormats.RenderGeometry
|
||||
import world.phantasmal.lib.fileFormats.*
|
||||
import world.phantasmal.lib.fileFormats.ninja.XjObject
|
||||
import world.phantasmal.lib.fileFormats.ninja.XvrTexture
|
||||
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.loading.AssetLoader
|
||||
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) {
|
||||
val cullingVolumes = mutableMapOf<Int, Box3>()
|
||||
|
||||
for (collisionMesh in collisionGeom.children) {
|
||||
collisionMesh as Mesh
|
||||
collisionMesh.userData.unsafeCast<AreaUserData>().section?.let { section ->
|
||||
cullingVolumes.getOrPut(section.id, ::Box3)
|
||||
.union(
|
||||
collisionMesh.geometry.boundingBox!!.applyMatrix4(collisionMesh.matrixWorld)
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
for (cullingVolume in cullingVolumes.values) {
|
||||
cullingVolume.min.x -= 50
|
||||
cullingVolume.min.y = cullingVolume.max.y + 20
|
||||
cullingVolume.min.z -= 50
|
||||
cullingVolume.max.x += 50
|
||||
cullingVolume.max.y = Double.POSITIVE_INFINITY
|
||||
cullingVolume.max.z += 50
|
||||
}
|
||||
|
||||
var i = 0
|
||||
|
||||
outer@ while (i < renderGeom.children.size) {
|
||||
val renderMesh = renderGeom.children[i] as Mesh
|
||||
val bb = renderMesh.geometry.boundingBox!!.applyMatrix4(renderMesh.matrixWorld)
|
||||
|
||||
for (cullingVolume in cullingVolumes.values) {
|
||||
if (bb.intersectsBox(cullingVolume)) {
|
||||
renderGeom.remove(renderMesh)
|
||||
continue@outer
|
||||
}
|
||||
}
|
||||
|
||||
i++
|
||||
}
|
||||
}
|
||||
// private fun cullRenderGeometry(collisionGeom: Object3D, renderGeom: Object3D) {
|
||||
// val cullingVolumes = mutableMapOf<Int, Box3>()
|
||||
//
|
||||
// for (collisionMesh in collisionGeom.children) {
|
||||
// collisionMesh as Mesh
|
||||
// collisionMesh.userData.unsafeCast<AreaUserData>().section?.let { section ->
|
||||
// cullingVolumes.getOrPut(section.id, ::Box3)
|
||||
// .union(
|
||||
// collisionMesh.geometry.boundingBox!!.applyMatrix4(collisionMesh.matrixWorld)
|
||||
// )
|
||||
// }
|
||||
// }
|
||||
//
|
||||
// for (cullingVolume in cullingVolumes.values) {
|
||||
// cullingVolume.min.x -= 50
|
||||
// cullingVolume.min.y = cullingVolume.max.y + 20
|
||||
// cullingVolume.min.z -= 50
|
||||
// cullingVolume.max.x += 50
|
||||
// cullingVolume.max.y = Double.POSITIVE_INFINITY
|
||||
// cullingVolume.max.z += 50
|
||||
// }
|
||||
//
|
||||
// var i = 0
|
||||
//
|
||||
// outer@ while (i < renderGeom.children.size) {
|
||||
// val renderMesh = renderGeom.children[i] as Mesh
|
||||
// val bb = renderMesh.geometry.boundingBox!!.applyMatrix4(renderMesh.matrixWorld)
|
||||
//
|
||||
// for (cullingVolume in cullingVolumes.values) {
|
||||
// if (bb.intersectsBox(cullingVolume)) {
|
||||
// renderGeom.remove(renderMesh)
|
||||
// continue@outer
|
||||
// }
|
||||
// }
|
||||
//
|
||||
// i++
|
||||
// }
|
||||
// }
|
||||
|
||||
private fun areaAssetUrl(
|
||||
episode: Episode,
|
||||
@ -206,18 +202,25 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine
|
||||
}
|
||||
|
||||
private fun areaGeometryToObject3DAndSections(
|
||||
renderGeometry: RenderGeometry,
|
||||
renderGeometry: AreaGeometry,
|
||||
textures: List<XvrTexture>,
|
||||
episode: Episode,
|
||||
areaVariant: AreaVariantModel,
|
||||
): 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>()
|
||||
console.log(renderGeometry)
|
||||
|
||||
val group =
|
||||
renderGeometryToGroup(renderGeometry, textures) { renderSection, xjObject, mesh ->
|
||||
if (shouldRenderOnTop(xjObject, renderOnTopTextures)) {
|
||||
mesh.renderOrder = 1
|
||||
renderGeometryToGroup(renderGeometry, textures) { renderSection, areaObj, mesh ->
|
||||
if (fix != null) {
|
||||
if (fix.shouldRenderOnTop(areaObj.xjObject)) {
|
||||
mesh.renderOrder = 1
|
||||
}
|
||||
|
||||
if (fix.shouldHide(areaObj)) {
|
||||
mesh.visible = false
|
||||
}
|
||||
}
|
||||
|
||||
if (renderSection.id >= 0) {
|
||||
@ -237,24 +240,6 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine
|
||||
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(
|
||||
obj: CollisionGeometry,
|
||||
episode: Episode,
|
||||
@ -288,6 +273,38 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine
|
||||
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 {
|
||||
private val COS_75_DEG = cos(PI / 180 * 75)
|
||||
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.
|
||||
* Manual fixes for various areas. Might not be necessary anymore once order-independent
|
||||
* rendering is implemented.
|
||||
* Mapping of episode and area ID to data for manually fixing issues with the render
|
||||
* geometry.
|
||||
*/
|
||||
val RENDER_ON_TOP_TEXTURES: Map<Pair<Episode, Int>, Set<Int>> = mapOf(
|
||||
private val MANUAL_FIXES: Map<Pair<Episode, Int>, Fix> = mutableMapOf(
|
||||
// Pioneer 2
|
||||
Pair(Episode.I, 0) to setOf(
|
||||
70, 71, 72, 126, 127, 155, 156, 198, 230, 231, 232, 233, 234,
|
||||
Pair(Episode.I, 0) to Fix(
|
||||
renderOnTopTextures = jsSetOf(
|
||||
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
|
||||
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
|
||||
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
|
||||
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
|
||||
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
|
||||
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 tmpVec = Vector3()
|
||||
|
@ -3,8 +3,8 @@ package world.phantasmal.web.viewer.stores
|
||||
import kotlinx.coroutines.launch
|
||||
import mu.KotlinLogging
|
||||
import world.phantasmal.core.enumValueOfOrNull
|
||||
import world.phantasmal.lib.fileFormats.AreaGeometry
|
||||
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.NjMotion
|
||||
import world.phantasmal.lib.fileFormats.ninja.NjObject
|
||||
@ -29,7 +29,7 @@ private val logger = KotlinLogging.logger {}
|
||||
|
||||
sealed class 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()
|
||||
}
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user