mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 15:28:29 +08:00
Fixed XJ loading bug in the viewer.
This commit is contained in:
parent
0ff4752949
commit
60d0bc6116
@ -2,8 +2,7 @@ package world.phantasmal.lib.fileFormats
|
|||||||
|
|
||||||
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.NinjaObject
|
import world.phantasmal.lib.fileFormats.ninja.XjObject
|
||||||
import world.phantasmal.lib.fileFormats.ninja.XjModel
|
|
||||||
import world.phantasmal.lib.fileFormats.ninja.angleToRad
|
import world.phantasmal.lib.fileFormats.ninja.angleToRad
|
||||||
import world.phantasmal.lib.fileFormats.ninja.parseXjObject
|
import world.phantasmal.lib.fileFormats.ninja.parseXjObject
|
||||||
|
|
||||||
@ -15,7 +14,7 @@ class RenderSection(
|
|||||||
val id: Int,
|
val id: Int,
|
||||||
val position: Vec3,
|
val position: Vec3,
|
||||||
val rotation: Vec3,
|
val rotation: Vec3,
|
||||||
val objects: List<NinjaObject<XjModel>>,
|
val objects: List<XjObject>,
|
||||||
)
|
)
|
||||||
|
|
||||||
fun parseAreaGeometry(cursor: Cursor): RenderObject {
|
fun parseAreaGeometry(cursor: Cursor): RenderObject {
|
||||||
@ -73,8 +72,8 @@ private fun parseGeometryTable(
|
|||||||
cursor: Cursor,
|
cursor: Cursor,
|
||||||
tableOffset: Int,
|
tableOffset: Int,
|
||||||
tableEntryCount: Int,
|
tableEntryCount: Int,
|
||||||
): List<NinjaObject<XjModel>> {
|
): List<XjObject> {
|
||||||
val objects = mutableListOf<NinjaObject<XjModel>>()
|
val objects = mutableListOf<XjObject>()
|
||||||
|
|
||||||
for (i in 0 until tableEntryCount) {
|
for (i in 0 until tableEntryCount) {
|
||||||
cursor.seekStart(tableOffset + 16 * i)
|
cursor.seekStart(tableOffset + 16 * i)
|
||||||
|
@ -11,29 +11,39 @@ import world.phantasmal.lib.fileFormats.vec3Float
|
|||||||
|
|
||||||
private const val NJCM: Int = 0x4D434A4E
|
private const val NJCM: Int = 0x4D434A4E
|
||||||
|
|
||||||
fun parseNj(cursor: Cursor): PwResult<List<NinjaObject<NjModel>>> =
|
fun parseNj(cursor: Cursor): PwResult<List<NjObject>> =
|
||||||
parseNinja(cursor, ::parseNjModel, mutableMapOf())
|
parseNinja(cursor, ::parseNjModel, ::NjObject, mutableMapOf())
|
||||||
|
|
||||||
fun parseXj(cursor: Cursor): PwResult<List<NinjaObject<XjModel>>> =
|
fun parseXj(cursor: Cursor): PwResult<List<XjObject>> =
|
||||||
parseNinja(cursor, { c, _ -> parseXjModel(c) }, Unit)
|
parseNinja(cursor, { c, _ -> parseXjModel(c) }, ::XjObject, Unit)
|
||||||
|
|
||||||
fun parseXjObject(cursor: Cursor): List<NinjaObject<XjModel>> =
|
fun parseXjObject(cursor: Cursor): List<XjObject> =
|
||||||
parseSiblingObjects(cursor, { c, _ -> parseXjModel(c) }, Unit)
|
parseSiblingObjects(cursor, { c, _ -> parseXjModel(c) }, ::XjObject, Unit)
|
||||||
|
|
||||||
private fun <Model : NinjaModel, Context> parseNinja(
|
private typealias CreateObject<Model, Obj> = (
|
||||||
|
evaluationFlags: NinjaEvaluationFlags,
|
||||||
|
model: Model?,
|
||||||
|
position: Vec3,
|
||||||
|
rotation: Vec3,
|
||||||
|
scale: Vec3,
|
||||||
|
children: MutableList<Obj>,
|
||||||
|
) -> Obj
|
||||||
|
|
||||||
|
private fun <Model : NinjaModel, Obj : NinjaObject<Model, Obj>, Context> parseNinja(
|
||||||
cursor: Cursor,
|
cursor: Cursor,
|
||||||
parseModel: (cursor: Cursor, context: Context) -> Model,
|
parseModel: (cursor: Cursor, context: Context) -> Model,
|
||||||
|
createObject: CreateObject<Model, Obj>,
|
||||||
context: Context,
|
context: Context,
|
||||||
): PwResult<List<NinjaObject<Model>>> =
|
): PwResult<List<Obj>> =
|
||||||
when (val parseIffResult = parseIff(cursor)) {
|
when (val parseIffResult = parseIff(cursor)) {
|
||||||
is Failure -> parseIffResult
|
is Failure -> parseIffResult
|
||||||
is Success -> {
|
is Success -> {
|
||||||
// POF0 and other chunks types are ignored.
|
// POF0 and other chunks types are ignored.
|
||||||
val njcmChunks = parseIffResult.value.filter { chunk -> chunk.type == NJCM }
|
val njcmChunks = parseIffResult.value.filter { chunk -> chunk.type == NJCM }
|
||||||
val objects: MutableList<NinjaObject<Model>> = mutableListOf()
|
val objects: MutableList<Obj> = mutableListOf()
|
||||||
|
|
||||||
for (chunk in njcmChunks) {
|
for (chunk in njcmChunks) {
|
||||||
objects.addAll(parseSiblingObjects(chunk.data, parseModel, context))
|
objects.addAll(parseSiblingObjects(chunk.data, parseModel, createObject, context))
|
||||||
}
|
}
|
||||||
|
|
||||||
Success(objects, parseIffResult.problems)
|
Success(objects, parseIffResult.problems)
|
||||||
@ -41,11 +51,12 @@ private fun <Model : NinjaModel, Context> parseNinja(
|
|||||||
}
|
}
|
||||||
|
|
||||||
// TODO: cache model and object offsets so we don't reparse the same data.
|
// TODO: cache model and object offsets so we don't reparse the same data.
|
||||||
private fun <Model : NinjaModel, Context> parseSiblingObjects(
|
private fun <Model : NinjaModel, Obj : NinjaObject<Model, Obj>, Context> parseSiblingObjects(
|
||||||
cursor: Cursor,
|
cursor: Cursor,
|
||||||
parseModel: (cursor: Cursor, context: Context) -> Model,
|
parseModel: (cursor: Cursor, context: Context) -> Model,
|
||||||
|
createObject: CreateObject<Model, Obj>,
|
||||||
context: Context,
|
context: Context,
|
||||||
): MutableList<NinjaObject<Model>> {
|
): MutableList<Obj> {
|
||||||
val evalFlags = cursor.int()
|
val evalFlags = cursor.int()
|
||||||
val noTranslate = evalFlags.isBitSet(0)
|
val noTranslate = evalFlags.isBitSet(0)
|
||||||
val noRotate = evalFlags.isBitSet(1)
|
val noRotate = evalFlags.isBitSet(1)
|
||||||
@ -80,17 +91,17 @@ private fun <Model : NinjaModel, Context> parseSiblingObjects(
|
|||||||
mutableListOf()
|
mutableListOf()
|
||||||
} else {
|
} else {
|
||||||
cursor.seekStart(childOffset)
|
cursor.seekStart(childOffset)
|
||||||
parseSiblingObjects(cursor, parseModel, context)
|
parseSiblingObjects(cursor, parseModel, createObject, context)
|
||||||
}
|
}
|
||||||
|
|
||||||
val siblings = if (siblingOffset == 0) {
|
val siblings = if (siblingOffset == 0) {
|
||||||
mutableListOf()
|
mutableListOf()
|
||||||
} else {
|
} else {
|
||||||
cursor.seekStart(siblingOffset)
|
cursor.seekStart(siblingOffset)
|
||||||
parseSiblingObjects(cursor, parseModel, context)
|
parseSiblingObjects(cursor, parseModel, createObject, context)
|
||||||
}
|
}
|
||||||
|
|
||||||
val obj = NinjaObject(
|
val obj = createObject(
|
||||||
NinjaEvaluationFlags(
|
NinjaEvaluationFlags(
|
||||||
noTranslate,
|
noTranslate,
|
||||||
noRotate,
|
noRotate,
|
||||||
|
@ -3,7 +3,7 @@ package world.phantasmal.lib.fileFormats.ninja
|
|||||||
import world.phantasmal.lib.fileFormats.Vec2
|
import world.phantasmal.lib.fileFormats.Vec2
|
||||||
import world.phantasmal.lib.fileFormats.Vec3
|
import world.phantasmal.lib.fileFormats.Vec3
|
||||||
|
|
||||||
class NinjaObject<Model : NinjaModel>(
|
sealed class NinjaObject<Model : NinjaModel, Self : NinjaObject<Model, Self>>(
|
||||||
val evaluationFlags: NinjaEvaluationFlags,
|
val evaluationFlags: NinjaEvaluationFlags,
|
||||||
val model: Model?,
|
val model: Model?,
|
||||||
val position: Vec3,
|
val position: Vec3,
|
||||||
@ -12,29 +12,31 @@ class NinjaObject<Model : NinjaModel>(
|
|||||||
*/
|
*/
|
||||||
val rotation: Vec3,
|
val rotation: Vec3,
|
||||||
val scale: Vec3,
|
val scale: Vec3,
|
||||||
children: MutableList<NinjaObject<Model>>,
|
children: MutableList<Self>,
|
||||||
) {
|
) {
|
||||||
private val _children = children
|
private val _children = children
|
||||||
val children: List<NinjaObject<Model>> = _children
|
val children: List<Self> = _children
|
||||||
|
|
||||||
fun addChild(child: NinjaObject<Model>) {
|
fun addChild(child: Self) {
|
||||||
_children.add(child)
|
_children.add(child)
|
||||||
}
|
}
|
||||||
|
|
||||||
fun boneCount(): Int {
|
fun boneCount(): Int {
|
||||||
val indexRef = intArrayOf(0)
|
val indexRef = intArrayOf(0)
|
||||||
findBone(this, Int.MAX_VALUE, indexRef)
|
@Suppress("UNCHECKED_CAST")
|
||||||
|
findBone(this as Self, Int.MAX_VALUE, indexRef)
|
||||||
return indexRef[0]
|
return indexRef[0]
|
||||||
}
|
}
|
||||||
|
|
||||||
fun getBone(index: Int): NinjaObject<Model>? =
|
fun getBone(index: Int): Self? =
|
||||||
findBone(this, index, intArrayOf(0))
|
@Suppress("UNCHECKED_CAST")
|
||||||
|
findBone(this as Self, index, intArrayOf(0))
|
||||||
|
|
||||||
private fun findBone(
|
private fun findBone(
|
||||||
obj: NinjaObject<Model>,
|
obj: Self,
|
||||||
boneIndex: Int,
|
boneIndex: Int,
|
||||||
indexRef: IntArray,
|
indexRef: IntArray,
|
||||||
): NinjaObject<Model>? {
|
): Self? {
|
||||||
if (!obj.evaluationFlags.skip) {
|
if (!obj.evaluationFlags.skip) {
|
||||||
val index = indexRef[0]++
|
val index = indexRef[0]++
|
||||||
|
|
||||||
@ -54,6 +56,38 @@ class NinjaObject<Model : NinjaModel>(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class NjObject(
|
||||||
|
evaluationFlags: NinjaEvaluationFlags,
|
||||||
|
model: NjModel?,
|
||||||
|
position: Vec3,
|
||||||
|
rotation: Vec3,
|
||||||
|
scale: Vec3,
|
||||||
|
children: MutableList<NjObject>,
|
||||||
|
) : NinjaObject<NjModel, NjObject>(
|
||||||
|
evaluationFlags,
|
||||||
|
model,
|
||||||
|
position,
|
||||||
|
rotation,
|
||||||
|
scale,
|
||||||
|
children,
|
||||||
|
)
|
||||||
|
|
||||||
|
class XjObject(
|
||||||
|
evaluationFlags: NinjaEvaluationFlags,
|
||||||
|
model: XjModel?,
|
||||||
|
position: Vec3,
|
||||||
|
rotation: Vec3,
|
||||||
|
scale: Vec3,
|
||||||
|
children: MutableList<XjObject>,
|
||||||
|
) : NinjaObject<XjModel, XjObject>(
|
||||||
|
evaluationFlags,
|
||||||
|
model,
|
||||||
|
position,
|
||||||
|
rotation,
|
||||||
|
scale,
|
||||||
|
children,
|
||||||
|
)
|
||||||
|
|
||||||
class NinjaEvaluationFlags(
|
class NinjaEvaluationFlags(
|
||||||
var noTranslate: Boolean,
|
var noTranslate: Boolean,
|
||||||
var noRotate: Boolean,
|
var noRotate: Boolean,
|
||||||
|
@ -11,7 +11,7 @@ import world.phantasmal.web.externals.three.*
|
|||||||
const val PSO_FRAME_RATE: Int = 30
|
const val PSO_FRAME_RATE: Int = 30
|
||||||
const val PSO_FRAME_RATE_DOUBLE: Double = PSO_FRAME_RATE.toDouble()
|
const val PSO_FRAME_RATE_DOUBLE: Double = PSO_FRAME_RATE.toDouble()
|
||||||
|
|
||||||
fun createAnimationClip(njObject: NinjaObject<*>, njMotion: NjMotion): AnimationClip {
|
fun createAnimationClip(njObject: NinjaObject<*, *>, njMotion: NjMotion): AnimationClip {
|
||||||
val interpolation =
|
val interpolation =
|
||||||
if (njMotion.interpolation == NjInterpolation.Spline) InterpolateSmooth
|
if (njMotion.interpolation == NjInterpolation.Spline) InterpolateSmooth
|
||||||
else InterpolateLinear
|
else InterpolateLinear
|
||||||
|
@ -19,8 +19,20 @@ private val tmpNormal = Vector3()
|
|||||||
private val tmpVec = Vector3()
|
private val tmpVec = Vector3()
|
||||||
private val tmpNormalMatrix = Matrix3()
|
private val tmpNormalMatrix = Matrix3()
|
||||||
|
|
||||||
|
fun ninjaObjectToMesh(
|
||||||
|
ninjaObject: NinjaObject<*, *>,
|
||||||
|
textures: List<XvrTexture?>,
|
||||||
|
defaultMaterial: Material? = null,
|
||||||
|
boundingVolumes: Boolean = false,
|
||||||
|
): Mesh {
|
||||||
|
val builder = MeshBuilder(textures)
|
||||||
|
defaultMaterial?.let { builder.defaultMaterial(defaultMaterial) }
|
||||||
|
ninjaObjectToMeshBuilder(ninjaObject, builder)
|
||||||
|
return builder.buildMesh(boundingVolumes)
|
||||||
|
}
|
||||||
|
|
||||||
fun ninjaObjectToInstancedMesh(
|
fun ninjaObjectToInstancedMesh(
|
||||||
ninjaObject: NinjaObject<*>,
|
ninjaObject: NinjaObject<*, *>,
|
||||||
textures: List<XvrTexture>,
|
textures: List<XvrTexture>,
|
||||||
maxInstances: Int,
|
maxInstances: Int,
|
||||||
defaultMaterial: Material? = null,
|
defaultMaterial: Material? = null,
|
||||||
@ -33,7 +45,7 @@ fun ninjaObjectToInstancedMesh(
|
|||||||
}
|
}
|
||||||
|
|
||||||
fun ninjaObjectToSkinnedMesh(
|
fun ninjaObjectToSkinnedMesh(
|
||||||
ninjaObject: NinjaObject<*>,
|
ninjaObject: NjObject,
|
||||||
textures: List<XvrTexture?>,
|
textures: List<XvrTexture?>,
|
||||||
defaultMaterial: Material? = null,
|
defaultMaterial: Material? = null,
|
||||||
boundingVolumes: Boolean = false,
|
boundingVolumes: Boolean = false,
|
||||||
@ -45,7 +57,7 @@ fun ninjaObjectToSkinnedMesh(
|
|||||||
}
|
}
|
||||||
|
|
||||||
fun ninjaObjectToMeshBuilder(
|
fun ninjaObjectToMeshBuilder(
|
||||||
ninjaObject: NinjaObject<*>,
|
ninjaObject: NinjaObject<*, *>,
|
||||||
builder: MeshBuilder,
|
builder: MeshBuilder,
|
||||||
) {
|
) {
|
||||||
NinjaToMeshConverter(builder).convert(ninjaObject)
|
NinjaToMeshConverter(builder).convert(ninjaObject)
|
||||||
@ -56,11 +68,11 @@ private class NinjaToMeshConverter(private val builder: MeshBuilder) {
|
|||||||
private val vertexHolder = VertexHolder()
|
private val vertexHolder = VertexHolder()
|
||||||
private var boneIndex = 0
|
private var boneIndex = 0
|
||||||
|
|
||||||
fun convert(ninjaObject: NinjaObject<*>) {
|
fun convert(ninjaObject: NinjaObject<*, *>) {
|
||||||
convertObject(ninjaObject, null, Matrix4())
|
convertObject(ninjaObject, null, Matrix4())
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun convertObject(obj: NinjaObject<*>, parentBone: Bone?, parentMatrix: Matrix4) {
|
private fun convertObject(obj: NinjaObject<*, *>, parentBone: Bone?, parentMatrix: Matrix4) {
|
||||||
val ef = obj.evaluationFlags
|
val ef = obj.evaluationFlags
|
||||||
|
|
||||||
val euler = Euler(
|
val euler = Euler(
|
||||||
|
@ -12,8 +12,7 @@ import world.phantasmal.lib.Episode
|
|||||||
import world.phantasmal.lib.cursor.cursor
|
import world.phantasmal.lib.cursor.cursor
|
||||||
import world.phantasmal.lib.fileFormats.CollisionObject
|
import world.phantasmal.lib.fileFormats.CollisionObject
|
||||||
import world.phantasmal.lib.fileFormats.RenderObject
|
import world.phantasmal.lib.fileFormats.RenderObject
|
||||||
import world.phantasmal.lib.fileFormats.ninja.NinjaObject
|
import world.phantasmal.lib.fileFormats.ninja.XjObject
|
||||||
import world.phantasmal.lib.fileFormats.ninja.XjModel
|
|
||||||
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.parseAreaCollisionGeometry
|
||||||
@ -222,7 +221,7 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun shouldRenderOnTop(
|
private fun shouldRenderOnTop(
|
||||||
obj: NinjaObject<XjModel>,
|
obj: XjObject,
|
||||||
episode: Episode,
|
episode: Episode,
|
||||||
areaVariant: AreaVariantModel,
|
areaVariant: AreaVariantModel,
|
||||||
): Boolean {
|
): Boolean {
|
||||||
@ -251,7 +250,7 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
fun recurse(obj: NinjaObject<XjModel>): Boolean {
|
fun recurse(obj: XjObject): Boolean {
|
||||||
obj.model?.meshes?.let { meshes ->
|
obj.model?.meshes?.let { meshes ->
|
||||||
for (mesh in meshes) {
|
for (mesh in meshes) {
|
||||||
mesh.material.textureId?.let {
|
mesh.material.textureId?.let {
|
||||||
|
@ -94,11 +94,11 @@ class EntityAssetLoader(private val assetLoader: AssetLoader) : DisposableContai
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun <Model : NinjaModel> parseGeometry(
|
private fun <Obj : NinjaObject<*, Obj>> parseGeometry(
|
||||||
type: EntityType,
|
type: EntityType,
|
||||||
parts: List<Pair<String, ArrayBuffer>>,
|
parts: List<Pair<String, ArrayBuffer>>,
|
||||||
parse: (Cursor) -> PwResult<List<NinjaObject<Model>>>,
|
parse: (Cursor) -> PwResult<List<Obj>>,
|
||||||
): NinjaObject<Model>? {
|
): Obj? {
|
||||||
val ninjaObjects = parts.flatMap { (path, data) ->
|
val ninjaObjects = parts.flatMap { (path, data) ->
|
||||||
val njObjects = parse(data.cursor(Endianness.Little))
|
val njObjects = parse(data.cursor(Endianness.Little))
|
||||||
|
|
||||||
|
@ -62,7 +62,7 @@ class ViewerToolbarController(private val store: ViewerStore) : Controller() {
|
|||||||
var success = false
|
var success = false
|
||||||
|
|
||||||
try {
|
try {
|
||||||
var ninjaObject: NinjaObject<*>? = null
|
var ninjaObject: NinjaObject<*, *>? = null
|
||||||
var textures: List<XvrTexture>? = null
|
var textures: List<XvrTexture>? = null
|
||||||
var ninjaMotion: NjMotion? = null
|
var ninjaMotion: NjMotion? = null
|
||||||
|
|
||||||
|
@ -13,13 +13,13 @@ import world.phantasmal.web.viewer.models.CharacterClass.*
|
|||||||
import world.phantasmal.webui.DisposableContainer
|
import world.phantasmal.webui.DisposableContainer
|
||||||
|
|
||||||
class CharacterClassAssetLoader(private val assetLoader: AssetLoader) : DisposableContainer() {
|
class CharacterClassAssetLoader(private val assetLoader: AssetLoader) : DisposableContainer() {
|
||||||
private val ninjaObjectCache: LoadingCache<CharacterClass, NinjaObject<NjModel>> =
|
private val ninjaObjectCache: LoadingCache<CharacterClass, NjObject> =
|
||||||
addDisposable(LoadingCache(::loadBodyParts) { /* Nothing to dispose. */ })
|
addDisposable(LoadingCache(::loadBodyParts) { /* Nothing to dispose. */ })
|
||||||
|
|
||||||
private val xvrTextureCache: LoadingCache<CharacterClass, List<XvrTexture?>> =
|
private val xvrTextureCache: LoadingCache<CharacterClass, List<XvrTexture?>> =
|
||||||
addDisposable(LoadingCache(::loadTextures) { /* Nothing to dispose. */ })
|
addDisposable(LoadingCache(::loadTextures) { /* Nothing to dispose. */ })
|
||||||
|
|
||||||
suspend fun loadNinjaObject(char: CharacterClass): NinjaObject<NjModel> =
|
suspend fun loadNinjaObject(char: CharacterClass): NjObject =
|
||||||
ninjaObjectCache.get(char)
|
ninjaObjectCache.get(char)
|
||||||
|
|
||||||
suspend fun loadXvrTextures(
|
suspend fun loadXvrTextures(
|
||||||
@ -42,7 +42,7 @@ class CharacterClassAssetLoader(private val assetLoader: AssetLoader) : Disposab
|
|||||||
/**
|
/**
|
||||||
* Loads the separate body parts and joins them together at the right bones.
|
* Loads the separate body parts and joins them together at the right bones.
|
||||||
*/
|
*/
|
||||||
private suspend fun loadBodyParts(char: CharacterClass): NinjaObject<NjModel> {
|
private suspend fun loadBodyParts(char: CharacterClass): NjObject {
|
||||||
val texIds = textureIds(char, SectionId.Viridia, 0)
|
val texIds = textureIds(char, SectionId.Viridia, 0)
|
||||||
|
|
||||||
val body = loadBodyPart(char, "Body")
|
val body = loadBodyPart(char, "Body")
|
||||||
@ -77,7 +77,7 @@ class CharacterClassAssetLoader(private val assetLoader: AssetLoader) : Disposab
|
|||||||
char: CharacterClass,
|
char: CharacterClass,
|
||||||
bodyPart: String,
|
bodyPart: String,
|
||||||
no: Int? = null,
|
no: Int? = null,
|
||||||
): NinjaObject<NjModel> {
|
): NjObject {
|
||||||
val buffer = assetLoader.loadArrayBuffer("/player/${char.slug}${bodyPart}${no ?: ""}.nj")
|
val buffer = assetLoader.loadArrayBuffer("/player/${char.slug}${bodyPart}${no ?: ""}.nj")
|
||||||
return parseNj(buffer.cursor(Endianness.Little)).unwrap().first()
|
return parseNj(buffer.cursor(Endianness.Little)).unwrap().first()
|
||||||
}
|
}
|
||||||
@ -85,7 +85,7 @@ class CharacterClassAssetLoader(private val assetLoader: AssetLoader) : Disposab
|
|||||||
/**
|
/**
|
||||||
* Shift texture IDs so that the IDs of different body parts don't overlap.
|
* Shift texture IDs so that the IDs of different body parts don't overlap.
|
||||||
*/
|
*/
|
||||||
private fun shiftTextureIds(njObject: NinjaObject<NjModel>, shift: Int) {
|
private fun shiftTextureIds(njObject: NjObject, shift: Int) {
|
||||||
njObject.model?.let { model ->
|
njObject.model?.let { model ->
|
||||||
for (mesh in model.meshes) {
|
for (mesh in model.meshes) {
|
||||||
mesh.textureId = mesh.textureId?.plus(shift)
|
mesh.textureId = mesh.textureId?.plus(shift)
|
||||||
@ -97,9 +97,9 @@ class CharacterClassAssetLoader(private val assetLoader: AssetLoader) : Disposab
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun <M : NinjaModel> addToBone(
|
private fun addToBone(
|
||||||
obj: NinjaObject<M>,
|
obj: NjObject,
|
||||||
child: NinjaObject<M>,
|
child: NjObject,
|
||||||
parentBoneId: Int,
|
parentBoneId: Int,
|
||||||
) {
|
) {
|
||||||
obj.getBone(parentBoneId)?.let { bone ->
|
obj.getBone(parentBoneId)?.let { bone ->
|
||||||
|
@ -5,10 +5,12 @@ import world.phantasmal.core.disposable.TrackedDisposable
|
|||||||
import world.phantasmal.core.math.degToRad
|
import world.phantasmal.core.math.degToRad
|
||||||
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.web.core.rendering.*
|
import world.phantasmal.web.core.rendering.*
|
||||||
import world.phantasmal.web.core.rendering.Renderer
|
import world.phantasmal.web.core.rendering.Renderer
|
||||||
import world.phantasmal.web.core.rendering.conversion.PSO_FRAME_RATE_DOUBLE
|
import world.phantasmal.web.core.rendering.conversion.PSO_FRAME_RATE_DOUBLE
|
||||||
import world.phantasmal.web.core.rendering.conversion.createAnimationClip
|
import world.phantasmal.web.core.rendering.conversion.createAnimationClip
|
||||||
|
import world.phantasmal.web.core.rendering.conversion.ninjaObjectToMesh
|
||||||
import world.phantasmal.web.core.rendering.conversion.ninjaObjectToSkinnedMesh
|
import world.phantasmal.web.core.rendering.conversion.ninjaObjectToSkinnedMesh
|
||||||
import world.phantasmal.web.core.times
|
import world.phantasmal.web.core.times
|
||||||
import world.phantasmal.web.externals.three.*
|
import world.phantasmal.web.externals.three.*
|
||||||
@ -91,7 +93,7 @@ class MeshRenderer(
|
|||||||
skeletonHelper = null
|
skeletonHelper = null
|
||||||
}
|
}
|
||||||
|
|
||||||
val njObject = viewerStore.currentNinjaObject.value
|
val ninjaObject = viewerStore.currentNinjaObject.value
|
||||||
val textures = viewerStore.currentTextures.value
|
val textures = viewerStore.currentTextures.value
|
||||||
|
|
||||||
// Stop and clean up previous animation and store animation time.
|
// Stop and clean up previous animation and store animation time.
|
||||||
@ -104,8 +106,13 @@ class MeshRenderer(
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Create a new mesh if necessary.
|
// Create a new mesh if necessary.
|
||||||
if (njObject != null) {
|
if (ninjaObject != null) {
|
||||||
val mesh = ninjaObjectToSkinnedMesh(njObject, textures, boundingVolumes = true)
|
val mesh =
|
||||||
|
if (ninjaObject is NjObject) {
|
||||||
|
ninjaObjectToSkinnedMesh(ninjaObject, textures, boundingVolumes = true)
|
||||||
|
} else {
|
||||||
|
ninjaObjectToMesh(ninjaObject, textures, boundingVolumes = true)
|
||||||
|
}
|
||||||
|
|
||||||
// Determine whether camera needs to be reset. Resets should always happen when the
|
// Determine whether camera needs to be reset. Resets should always happen when the
|
||||||
// Ninja object changes except when we're switching between character class models.
|
// Ninja object changes except when we're switching between character class models.
|
||||||
@ -125,20 +132,22 @@ class MeshRenderer(
|
|||||||
context.scene.add(mesh)
|
context.scene.add(mesh)
|
||||||
this.mesh = mesh
|
this.mesh = mesh
|
||||||
|
|
||||||
// Add skeleton.
|
if (mesh is SkinnedMesh) {
|
||||||
val skeletonHelper = SkeletonHelper(mesh)
|
// Add skeleton.
|
||||||
skeletonHelper.visible = viewerStore.showSkeleton.value
|
val skeletonHelper = SkeletonHelper(mesh)
|
||||||
skeletonHelper.asDynamic().material.lineWidth = 3
|
skeletonHelper.visible = viewerStore.showSkeleton.value
|
||||||
|
skeletonHelper.asDynamic().material.lineWidth = 3
|
||||||
|
|
||||||
context.scene.add(skeletonHelper)
|
context.scene.add(skeletonHelper)
|
||||||
this.skeletonHelper = skeletonHelper
|
this.skeletonHelper = skeletonHelper
|
||||||
|
|
||||||
// Create a new animation mixer and clip.
|
// Create a new animation mixer and clip.
|
||||||
viewerStore.currentNinjaMotion.value?.let { njMotion ->
|
viewerStore.currentNinjaMotion.value?.let { njMotion ->
|
||||||
animation = Animation(njObject, njMotion, mesh).also {
|
animation = Animation(ninjaObject, njMotion, mesh).also {
|
||||||
it.mixer.timeScale = viewerStore.frameRate.value / PSO_FRAME_RATE_DOUBLE
|
it.mixer.timeScale = viewerStore.frameRate.value / PSO_FRAME_RATE_DOUBLE
|
||||||
it.action.time = animationTime ?: .0
|
it.action.time = animationTime ?: .0
|
||||||
it.action.play()
|
it.action.play()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -192,7 +201,7 @@ class MeshRenderer(
|
|||||||
}
|
}
|
||||||
|
|
||||||
private class Animation(
|
private class Animation(
|
||||||
njObject: NinjaObject<*>,
|
njObject: NinjaObject<*, *>,
|
||||||
njMotion: NjMotion,
|
njMotion: NjMotion,
|
||||||
root: Object3D,
|
root: Object3D,
|
||||||
) : TrackedDisposable() {
|
) : TrackedDisposable() {
|
||||||
|
@ -29,7 +29,7 @@ class ViewerStore(
|
|||||||
uiStore: UiStore,
|
uiStore: UiStore,
|
||||||
) : Store() {
|
) : Store() {
|
||||||
// Ninja concepts.
|
// Ninja concepts.
|
||||||
private val _currentNinjaObject = mutableVal<NinjaObject<*>?>(null)
|
private val _currentNinjaObject = mutableVal<NinjaObject<*, *>?>(null)
|
||||||
private val _currentTextures = mutableListVal<XvrTexture?>()
|
private val _currentTextures = mutableListVal<XvrTexture?>()
|
||||||
private val _currentNinjaMotion = mutableVal<NjMotion?>(null)
|
private val _currentNinjaMotion = mutableVal<NjMotion?>(null)
|
||||||
|
|
||||||
@ -47,7 +47,7 @@ class ViewerStore(
|
|||||||
private val _frame = mutableVal(0)
|
private val _frame = mutableVal(0)
|
||||||
|
|
||||||
// Ninja concepts.
|
// Ninja concepts.
|
||||||
val currentNinjaObject: Val<NinjaObject<*>?> = _currentNinjaObject
|
val currentNinjaObject: Val<NinjaObject<*, *>?> = _currentNinjaObject
|
||||||
val currentTextures: ListVal<XvrTexture?> = _currentTextures
|
val currentTextures: ListVal<XvrTexture?> = _currentTextures
|
||||||
val currentNinjaMotion: Val<NjMotion?> = _currentNinjaMotion
|
val currentNinjaMotion: Val<NjMotion?> = _currentNinjaMotion
|
||||||
|
|
||||||
@ -143,7 +143,7 @@ class ViewerStore(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fun setCurrentNinjaObject(ninjaObject: NinjaObject<*>?) {
|
fun setCurrentNinjaObject(ninjaObject: NinjaObject<*, *>?) {
|
||||||
if (_currentCharacterClass.value != null) {
|
if (_currentCharacterClass.value != null) {
|
||||||
_currentCharacterClass.value = null
|
_currentCharacterClass.value = null
|
||||||
_currentTextures.clear()
|
_currentTextures.clear()
|
||||||
|
Loading…
Reference in New Issue
Block a user