mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 15:28:29 +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 =
|
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
|
||||||
|
@ -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>>()
|
||||||
|
@ -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()
|
||||||
|
}
|
||||||
|
@ -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()
|
||||||
|
@ -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)
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -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()
|
||||||
|
@ -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,
|
||||||
|
@ -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
|
||||||
|
|
||||||
|
@ -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
|
||||||
}
|
}
|
||||||
|
@ -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()
|
||||||
|
@ -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()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user