From 7e5dea8e258058ebe5108e4c33bb7a2c6e25ee85 Mon Sep 17 00:00:00 2001 From: Daan Vanden Bosch Date: Fri, 26 Mar 2021 14:49:48 +0100 Subject: [PATCH] Character animations can be selected again in the viewer. --- .../world/phantasmal/web/viewer/Viewer.kt | 5 +- .../viewer/controllers/ViewerController.kt | 8 ++ .../viewer/loading/AnimationAssetLoader.kt | 20 +++++ .../loading/CharacterClassAssetLoader.kt | 2 +- .../web/viewer/models/AnimationModel.kt | 3 + .../web/viewer/rendering/MeshRenderer.kt | 25 +++---- .../web/viewer/stores/ViewerStore.kt | 73 +++++++++++++------ .../web/viewer/widgets/ViewerWidget.kt | 7 ++ 8 files changed, 107 insertions(+), 36 deletions(-) create mode 100644 web/src/main/kotlin/world/phantasmal/web/viewer/loading/AnimationAssetLoader.kt create mode 100644 web/src/main/kotlin/world/phantasmal/web/viewer/models/AnimationModel.kt diff --git a/web/src/main/kotlin/world/phantasmal/web/viewer/Viewer.kt b/web/src/main/kotlin/world/phantasmal/web/viewer/Viewer.kt index 91ae3407..9bf822d8 100644 --- a/web/src/main/kotlin/world/phantasmal/web/viewer/Viewer.kt +++ b/web/src/main/kotlin/world/phantasmal/web/viewer/Viewer.kt @@ -10,6 +10,7 @@ import world.phantasmal.web.core.widgets.RendererWidget import world.phantasmal.web.viewer.controllers.CharacterClassOptionsController import world.phantasmal.web.viewer.controllers.ViewerController import world.phantasmal.web.viewer.controllers.ViewerToolbarController +import world.phantasmal.web.viewer.loading.AnimationAssetLoader import world.phantasmal.web.viewer.loading.CharacterClassAssetLoader import world.phantasmal.web.viewer.rendering.MeshRenderer import world.phantasmal.web.viewer.rendering.TextureRenderer @@ -30,9 +31,11 @@ class Viewer( override fun initialize(): Widget { // Asset Loaders val characterClassAssetLoader = addDisposable(CharacterClassAssetLoader(assetLoader)) + val animationAssetLoader = addDisposable(AnimationAssetLoader(assetLoader)) // Stores - val viewerStore = addDisposable(ViewerStore(characterClassAssetLoader, uiStore)) + val viewerStore = + addDisposable(ViewerStore(characterClassAssetLoader, animationAssetLoader, uiStore)) // Controllers val viewerController = addDisposable(ViewerController(uiStore, viewerStore)) diff --git a/web/src/main/kotlin/world/phantasmal/web/viewer/controllers/ViewerController.kt b/web/src/main/kotlin/world/phantasmal/web/viewer/controllers/ViewerController.kt index 6d9150ad..2a7a3986 100644 --- a/web/src/main/kotlin/world/phantasmal/web/viewer/controllers/ViewerController.kt +++ b/web/src/main/kotlin/world/phantasmal/web/viewer/controllers/ViewerController.kt @@ -6,6 +6,7 @@ import world.phantasmal.web.core.controllers.PathAwareTab import world.phantasmal.web.core.controllers.PathAwareTabContainerController import world.phantasmal.web.core.stores.UiStore import world.phantasmal.web.viewer.ViewerUrls +import world.phantasmal.web.viewer.models.AnimationModel import world.phantasmal.web.viewer.models.CharacterClass import world.phantasmal.web.viewer.stores.ViewerStore @@ -28,7 +29,14 @@ class ViewerController( val characterClasses: List = CharacterClass.VALUES_LIST val currentCharacterClass: Val = store.currentCharacterClass + val animations: List = store.animations + val currentAnimation: Val = store.currentAnimation + suspend fun setCurrentCharacterClass(char: CharacterClass?) { store.setCurrentCharacterClass(char) } + + suspend fun setCurrentAnimation(animation: AnimationModel) { + store.setCurrentAnimation(animation) + } } diff --git a/web/src/main/kotlin/world/phantasmal/web/viewer/loading/AnimationAssetLoader.kt b/web/src/main/kotlin/world/phantasmal/web/viewer/loading/AnimationAssetLoader.kt new file mode 100644 index 00000000..dba7d3f9 --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/viewer/loading/AnimationAssetLoader.kt @@ -0,0 +1,20 @@ +package world.phantasmal.web.viewer.loading + +import world.phantasmal.lib.Endianness +import world.phantasmal.lib.cursor.cursor +import world.phantasmal.lib.fileFormats.ninja.NjMotion +import world.phantasmal.lib.fileFormats.ninja.parseNjm +import world.phantasmal.web.core.loading.AssetLoader +import world.phantasmal.web.questEditor.loading.LoadingCache +import world.phantasmal.webui.DisposableContainer + +class AnimationAssetLoader(private val assetLoader: AssetLoader) : DisposableContainer() { + private val ninjaMotionCache: LoadingCache = + addDisposable(LoadingCache(::loadNinjaMotion) { /* Nothing to dispose. */ }) + + suspend fun loadAnimation(filePath: String): NjMotion = + ninjaMotionCache.get(filePath) + + private suspend fun loadNinjaMotion(filePath: String): NjMotion = + parseNjm(assetLoader.loadArrayBuffer(filePath).cursor(Endianness.Little)) +} diff --git a/web/src/main/kotlin/world/phantasmal/web/viewer/loading/CharacterClassAssetLoader.kt b/web/src/main/kotlin/world/phantasmal/web/viewer/loading/CharacterClassAssetLoader.kt index 8953af20..2d37a942 100644 --- a/web/src/main/kotlin/world/phantasmal/web/viewer/loading/CharacterClassAssetLoader.kt +++ b/web/src/main/kotlin/world/phantasmal/web/viewer/loading/CharacterClassAssetLoader.kt @@ -6,8 +6,8 @@ import world.phantasmal.lib.cursor.cursor import world.phantasmal.lib.fileFormats.ninja.* import world.phantasmal.lib.fileFormats.parseAfs import world.phantasmal.web.core.loading.AssetLoader -import world.phantasmal.web.shared.dto.SectionId import world.phantasmal.web.questEditor.loading.LoadingCache +import world.phantasmal.web.shared.dto.SectionId import world.phantasmal.web.viewer.models.CharacterClass import world.phantasmal.web.viewer.models.CharacterClass.* import world.phantasmal.webui.DisposableContainer diff --git a/web/src/main/kotlin/world/phantasmal/web/viewer/models/AnimationModel.kt b/web/src/main/kotlin/world/phantasmal/web/viewer/models/AnimationModel.kt new file mode 100644 index 00000000..9f2d1c16 --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/viewer/models/AnimationModel.kt @@ -0,0 +1,3 @@ +package world.phantasmal.web.viewer.models + +class AnimationModel(val name: String, val filePath: String) diff --git a/web/src/main/kotlin/world/phantasmal/web/viewer/rendering/MeshRenderer.kt b/web/src/main/kotlin/world/phantasmal/web/viewer/rendering/MeshRenderer.kt index ce59fc90..2627ab01 100644 --- a/web/src/main/kotlin/world/phantasmal/web/viewer/rendering/MeshRenderer.kt +++ b/web/src/main/kotlin/world/phantasmal/web/viewer/rendering/MeshRenderer.kt @@ -21,6 +21,7 @@ class MeshRenderer( private var mesh: Mesh? = null private var skeletonHelper: SkeletonHelper? = null private var animation: Animation? = null + private var charClassActive = false override val context = addDisposable(RenderContext( createCanvas(), @@ -28,7 +29,7 @@ class MeshRenderer( fov = 45.0, aspect = 1.0, near = 10.0, - far = 5_000.0 + far = 5_000.0, ) )) @@ -38,16 +39,12 @@ class MeshRenderer( context.canvas, context.camera, Vector3(), - screenSpacePanning = true + screenSpacePanning = true, )) init { - observe(viewerStore.currentNinjaObject) { - ninjaObjectOrXvmChanged(resetCamera = true) - } - observe(viewerStore.currentTextures) { - ninjaObjectOrXvmChanged(resetCamera = false) - } + observe(viewerStore.currentNinjaObject) { ninjaObjectOrXvmChanged() } + observe(viewerStore.currentTextures) { ninjaObjectOrXvmChanged() } observe(viewerStore.currentNinjaMotion, ::ninjaMotionChanged) observe(viewerStore.showSkeleton) { skeletonHelper?.visible = it } } @@ -66,7 +63,7 @@ class MeshRenderer( } } - private fun ninjaObjectOrXvmChanged(resetCamera: Boolean) { + private fun ninjaObjectOrXvmChanged() { // Remove the previous mesh. mesh?.let { mesh -> disposeObject3DResources(mesh) @@ -79,10 +76,6 @@ class MeshRenderer( skeletonHelper = null } - if (resetCamera) { - inputManager.resetCamera() - } - val ninjaObject = viewerStore.currentNinjaObject.value val textures = viewerStore.currentTextures.value @@ -101,6 +94,12 @@ class MeshRenderer( if (ninjaObject != null) { val mesh = ninjaObjectToSkinnedMesh(ninjaObject, textures, boundingVolumes = true) + // 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. + val charClassActive = viewerStore.currentCharacterClass.value != null + val resetCamera = !charClassActive || !this.charClassActive + this.charClassActive = charClassActive + if (resetCamera) { // Compute camera position. val bSphere = mesh.geometry.boundingSphere!! diff --git a/web/src/main/kotlin/world/phantasmal/web/viewer/stores/ViewerStore.kt b/web/src/main/kotlin/world/phantasmal/web/viewer/stores/ViewerStore.kt index 78d53b14..f4c8011e 100644 --- a/web/src/main/kotlin/world/phantasmal/web/viewer/stores/ViewerStore.kt +++ b/web/src/main/kotlin/world/phantasmal/web/viewer/stores/ViewerStore.kt @@ -1,8 +1,6 @@ package world.phantasmal.web.viewer.stores -import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch -import kotlinx.coroutines.withContext import mu.KotlinLogging import world.phantasmal.core.enumValueOfOrNull import world.phantasmal.lib.fileFormats.ninja.NinjaObject @@ -16,35 +14,52 @@ import world.phantasmal.web.core.PwToolType import world.phantasmal.web.core.stores.UiStore import world.phantasmal.web.shared.dto.SectionId import world.phantasmal.web.viewer.ViewerUrls +import world.phantasmal.web.viewer.loading.AnimationAssetLoader import world.phantasmal.web.viewer.loading.CharacterClassAssetLoader +import world.phantasmal.web.viewer.models.AnimationModel import world.phantasmal.web.viewer.models.CharacterClass import world.phantasmal.webui.stores.Store private val logger = KotlinLogging.logger {} class ViewerStore( - private val assetLoader: CharacterClassAssetLoader, + private val characterClassAssetLoader: CharacterClassAssetLoader, + private val animationAssetLoader: AnimationAssetLoader, uiStore: UiStore, ) : Store() { + // Ninja concepts. private val _currentNinjaObject = mutableVal?>(null) private val _currentTextures = mutableListVal() + private val _currentNinjaMotion = mutableVal(null) + + // High-level concepts. private val _currentCharacterClass = mutableVal(CharacterClass.VALUES.random()) private val _currentSectionId = mutableVal(SectionId.VALUES.random()) private val _currentBody = mutableVal((1.._currentCharacterClass.value!!.bodyStyleCount).random()) - private val _currentNinjaMotion = mutableVal(null) + private val _currentAnimation = mutableVal(null) - // Settings + // Settings. private val _showSkeleton = mutableVal(false) + // Ninja concepts. val currentNinjaObject: Val?> = _currentNinjaObject val currentTextures: ListVal = _currentTextures + val currentNinjaMotion: Val = _currentNinjaMotion + + // High-level concepts. val currentCharacterClass: Val = _currentCharacterClass val currentSectionId: Val = _currentSectionId val currentBody: Val = _currentBody - val currentNinjaMotion: Val = _currentNinjaMotion + val animations: List = (0 until 572).map { + AnimationModel( + "Animation ${it + 1}", + "/player/animation/animation_${it.toString().padStart(3, '0')}.njm" + ) + } + val currentAnimation: Val = _currentAnimation - // Settings + // Settings. val showSkeleton: Val = _showSkeleton init { @@ -116,7 +131,7 @@ class ViewerStore( ) } - scope.launch(Dispatchers.Default) { + scope.launch { loadCharacterClassNinjaObject(clearAnimation = true) } } @@ -127,6 +142,7 @@ class ViewerStore( _currentTextures.clear() } + _currentAnimation.value = null _currentNinjaMotion.value = null _currentNinjaObject.value = ninjaObject } @@ -161,6 +177,11 @@ class ViewerStore( _currentNinjaMotion.value = njm } + suspend fun setCurrentAnimation(animation: AnimationModel) { + _currentAnimation.value = animation + loadAnimation(animation) + } + fun setShowSkeleton(show: Boolean) { _showSkeleton.value = show } @@ -173,25 +194,35 @@ class ViewerStore( val body = currentBody.value try { - val ninjaObject = assetLoader.loadNinjaObject(char) - val textures = assetLoader.loadXvrTextures(char, sectionId, body) + val ninjaObject = characterClassAssetLoader.loadNinjaObject(char) + val textures = characterClassAssetLoader.loadXvrTextures(char, sectionId, body) - withContext(Dispatchers.Main) { - if (clearAnimation) { - _currentNinjaMotion.value = null - } - - _currentNinjaObject.value = ninjaObject - _currentTextures.replaceAll(textures) + if (clearAnimation) { + _currentAnimation.value = null + _currentNinjaMotion.value = null } + + _currentNinjaObject.value = ninjaObject + _currentTextures.replaceAll(textures) } catch (e: Exception) { logger.error(e) { "Couldn't load Ninja model for $char." } - withContext(Dispatchers.Main) { - _currentNinjaMotion.value = null - _currentNinjaObject.value = null - _currentTextures.clear() + _currentAnimation.value = null + _currentNinjaMotion.value = null + _currentNinjaObject.value = null + _currentTextures.clear() + } + } + + private suspend fun loadAnimation(animation: AnimationModel) { + try { + _currentNinjaMotion.value = animationAssetLoader.loadAnimation(animation.filePath) + } catch (e: Exception) { + logger.error(e) { + "Couldn't load Ninja motion for ${animation.name} (path: ${animation.filePath})." } + + _currentNinjaMotion.value = null } } diff --git a/web/src/main/kotlin/world/phantasmal/web/viewer/widgets/ViewerWidget.kt b/web/src/main/kotlin/world/phantasmal/web/viewer/widgets/ViewerWidget.kt index 146f7521..11d89cf2 100644 --- a/web/src/main/kotlin/world/phantasmal/web/viewer/widgets/ViewerWidget.kt +++ b/web/src/main/kotlin/world/phantasmal/web/viewer/widgets/ViewerWidget.kt @@ -37,6 +37,13 @@ class ViewerWidget( ViewerTab.Texture -> createTextureWidget() } })) + addChild(SelectionWidget( + ctrl.animations, + ctrl.currentAnimation, + { animation -> scope.launch { ctrl.setCurrentAnimation(animation) } }, + { it.name }, + borderLeft = true, + )) } }