Character animations can be selected again in the viewer.

This commit is contained in:
Daan Vanden Bosch 2021-03-26 14:49:48 +01:00
parent 23716615bf
commit 7e5dea8e25
8 changed files with 107 additions and 36 deletions

View File

@ -10,6 +10,7 @@ import world.phantasmal.web.core.widgets.RendererWidget
import world.phantasmal.web.viewer.controllers.CharacterClassOptionsController import world.phantasmal.web.viewer.controllers.CharacterClassOptionsController
import world.phantasmal.web.viewer.controllers.ViewerController import world.phantasmal.web.viewer.controllers.ViewerController
import world.phantasmal.web.viewer.controllers.ViewerToolbarController 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.loading.CharacterClassAssetLoader
import world.phantasmal.web.viewer.rendering.MeshRenderer import world.phantasmal.web.viewer.rendering.MeshRenderer
import world.phantasmal.web.viewer.rendering.TextureRenderer import world.phantasmal.web.viewer.rendering.TextureRenderer
@ -30,9 +31,11 @@ class Viewer(
override fun initialize(): Widget { override fun initialize(): Widget {
// Asset Loaders // Asset Loaders
val characterClassAssetLoader = addDisposable(CharacterClassAssetLoader(assetLoader)) val characterClassAssetLoader = addDisposable(CharacterClassAssetLoader(assetLoader))
val animationAssetLoader = addDisposable(AnimationAssetLoader(assetLoader))
// Stores // Stores
val viewerStore = addDisposable(ViewerStore(characterClassAssetLoader, uiStore)) val viewerStore =
addDisposable(ViewerStore(characterClassAssetLoader, animationAssetLoader, uiStore))
// Controllers // Controllers
val viewerController = addDisposable(ViewerController(uiStore, viewerStore)) val viewerController = addDisposable(ViewerController(uiStore, viewerStore))

View File

@ -6,6 +6,7 @@ import world.phantasmal.web.core.controllers.PathAwareTab
import world.phantasmal.web.core.controllers.PathAwareTabContainerController import world.phantasmal.web.core.controllers.PathAwareTabContainerController
import world.phantasmal.web.core.stores.UiStore import world.phantasmal.web.core.stores.UiStore
import world.phantasmal.web.viewer.ViewerUrls 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.models.CharacterClass
import world.phantasmal.web.viewer.stores.ViewerStore import world.phantasmal.web.viewer.stores.ViewerStore
@ -28,7 +29,14 @@ class ViewerController(
val characterClasses: List<CharacterClass> = CharacterClass.VALUES_LIST val characterClasses: List<CharacterClass> = CharacterClass.VALUES_LIST
val currentCharacterClass: Val<CharacterClass?> = store.currentCharacterClass val currentCharacterClass: Val<CharacterClass?> = store.currentCharacterClass
val animations: List<AnimationModel> = store.animations
val currentAnimation: Val<AnimationModel?> = store.currentAnimation
suspend fun setCurrentCharacterClass(char: CharacterClass?) { suspend fun setCurrentCharacterClass(char: CharacterClass?) {
store.setCurrentCharacterClass(char) store.setCurrentCharacterClass(char)
} }
suspend fun setCurrentAnimation(animation: AnimationModel) {
store.setCurrentAnimation(animation)
}
} }

View File

@ -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<String, NjMotion> =
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))
}

View File

@ -6,8 +6,8 @@ import world.phantasmal.lib.cursor.cursor
import world.phantasmal.lib.fileFormats.ninja.* import world.phantasmal.lib.fileFormats.ninja.*
import world.phantasmal.lib.fileFormats.parseAfs import world.phantasmal.lib.fileFormats.parseAfs
import world.phantasmal.web.core.loading.AssetLoader 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.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.web.viewer.models.CharacterClass.* import world.phantasmal.web.viewer.models.CharacterClass.*
import world.phantasmal.webui.DisposableContainer import world.phantasmal.webui.DisposableContainer

View File

@ -0,0 +1,3 @@
package world.phantasmal.web.viewer.models
class AnimationModel(val name: String, val filePath: String)

View File

@ -21,6 +21,7 @@ class MeshRenderer(
private var mesh: Mesh? = null private var mesh: Mesh? = null
private var skeletonHelper: SkeletonHelper? = null private var skeletonHelper: SkeletonHelper? = null
private var animation: Animation? = null private var animation: Animation? = null
private var charClassActive = false
override val context = addDisposable(RenderContext( override val context = addDisposable(RenderContext(
createCanvas(), createCanvas(),
@ -28,7 +29,7 @@ class MeshRenderer(
fov = 45.0, fov = 45.0,
aspect = 1.0, aspect = 1.0,
near = 10.0, near = 10.0,
far = 5_000.0 far = 5_000.0,
) )
)) ))
@ -38,16 +39,12 @@ class MeshRenderer(
context.canvas, context.canvas,
context.camera, context.camera,
Vector3(), Vector3(),
screenSpacePanning = true screenSpacePanning = true,
)) ))
init { init {
observe(viewerStore.currentNinjaObject) { observe(viewerStore.currentNinjaObject) { ninjaObjectOrXvmChanged() }
ninjaObjectOrXvmChanged(resetCamera = true) observe(viewerStore.currentTextures) { ninjaObjectOrXvmChanged() }
}
observe(viewerStore.currentTextures) {
ninjaObjectOrXvmChanged(resetCamera = false)
}
observe(viewerStore.currentNinjaMotion, ::ninjaMotionChanged) observe(viewerStore.currentNinjaMotion, ::ninjaMotionChanged)
observe(viewerStore.showSkeleton) { skeletonHelper?.visible = it } observe(viewerStore.showSkeleton) { skeletonHelper?.visible = it }
} }
@ -66,7 +63,7 @@ class MeshRenderer(
} }
} }
private fun ninjaObjectOrXvmChanged(resetCamera: Boolean) { private fun ninjaObjectOrXvmChanged() {
// Remove the previous mesh. // Remove the previous mesh.
mesh?.let { mesh -> mesh?.let { mesh ->
disposeObject3DResources(mesh) disposeObject3DResources(mesh)
@ -79,10 +76,6 @@ class MeshRenderer(
skeletonHelper = null skeletonHelper = null
} }
if (resetCamera) {
inputManager.resetCamera()
}
val ninjaObject = viewerStore.currentNinjaObject.value val ninjaObject = viewerStore.currentNinjaObject.value
val textures = viewerStore.currentTextures.value val textures = viewerStore.currentTextures.value
@ -101,6 +94,12 @@ class MeshRenderer(
if (ninjaObject != null) { if (ninjaObject != null) {
val mesh = ninjaObjectToSkinnedMesh(ninjaObject, textures, boundingVolumes = true) 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) { if (resetCamera) {
// Compute camera position. // Compute camera position.
val bSphere = mesh.geometry.boundingSphere!! val bSphere = mesh.geometry.boundingSphere!!

View File

@ -1,8 +1,6 @@
package world.phantasmal.web.viewer.stores package world.phantasmal.web.viewer.stores
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.launch import kotlinx.coroutines.launch
import kotlinx.coroutines.withContext
import mu.KotlinLogging import mu.KotlinLogging
import world.phantasmal.core.enumValueOfOrNull import world.phantasmal.core.enumValueOfOrNull
import world.phantasmal.lib.fileFormats.ninja.NinjaObject 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.core.stores.UiStore
import world.phantasmal.web.shared.dto.SectionId import world.phantasmal.web.shared.dto.SectionId
import world.phantasmal.web.viewer.ViewerUrls 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.loading.CharacterClassAssetLoader
import world.phantasmal.web.viewer.models.AnimationModel
import world.phantasmal.web.viewer.models.CharacterClass import world.phantasmal.web.viewer.models.CharacterClass
import world.phantasmal.webui.stores.Store import world.phantasmal.webui.stores.Store
private val logger = KotlinLogging.logger {} private val logger = KotlinLogging.logger {}
class ViewerStore( class ViewerStore(
private val assetLoader: CharacterClassAssetLoader, private val characterClassAssetLoader: CharacterClassAssetLoader,
private val animationAssetLoader: AnimationAssetLoader,
uiStore: UiStore, uiStore: UiStore,
) : Store() { ) : Store() {
// 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)
// High-level concepts.
private val _currentCharacterClass = mutableVal<CharacterClass?>(CharacterClass.VALUES.random()) private val _currentCharacterClass = mutableVal<CharacterClass?>(CharacterClass.VALUES.random())
private val _currentSectionId = mutableVal(SectionId.VALUES.random()) private val _currentSectionId = mutableVal(SectionId.VALUES.random())
private val _currentBody = private val _currentBody =
mutableVal((1.._currentCharacterClass.value!!.bodyStyleCount).random()) mutableVal((1.._currentCharacterClass.value!!.bodyStyleCount).random())
private val _currentNinjaMotion = mutableVal<NjMotion?>(null) private val _currentAnimation = mutableVal<AnimationModel?>(null)
// Settings // Settings.
private val _showSkeleton = mutableVal(false) private val _showSkeleton = mutableVal(false)
// 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
// High-level concepts.
val currentCharacterClass: Val<CharacterClass?> = _currentCharacterClass val currentCharacterClass: Val<CharacterClass?> = _currentCharacterClass
val currentSectionId: Val<SectionId> = _currentSectionId val currentSectionId: Val<SectionId> = _currentSectionId
val currentBody: Val<Int> = _currentBody val currentBody: Val<Int> = _currentBody
val currentNinjaMotion: Val<NjMotion?> = _currentNinjaMotion val animations: List<AnimationModel> = (0 until 572).map {
AnimationModel(
"Animation ${it + 1}",
"/player/animation/animation_${it.toString().padStart(3, '0')}.njm"
)
}
val currentAnimation: Val<AnimationModel?> = _currentAnimation
// Settings // Settings.
val showSkeleton: Val<Boolean> = _showSkeleton val showSkeleton: Val<Boolean> = _showSkeleton
init { init {
@ -116,7 +131,7 @@ class ViewerStore(
) )
} }
scope.launch(Dispatchers.Default) { scope.launch {
loadCharacterClassNinjaObject(clearAnimation = true) loadCharacterClassNinjaObject(clearAnimation = true)
} }
} }
@ -127,6 +142,7 @@ class ViewerStore(
_currentTextures.clear() _currentTextures.clear()
} }
_currentAnimation.value = null
_currentNinjaMotion.value = null _currentNinjaMotion.value = null
_currentNinjaObject.value = ninjaObject _currentNinjaObject.value = ninjaObject
} }
@ -161,6 +177,11 @@ class ViewerStore(
_currentNinjaMotion.value = njm _currentNinjaMotion.value = njm
} }
suspend fun setCurrentAnimation(animation: AnimationModel) {
_currentAnimation.value = animation
loadAnimation(animation)
}
fun setShowSkeleton(show: Boolean) { fun setShowSkeleton(show: Boolean) {
_showSkeleton.value = show _showSkeleton.value = show
} }
@ -173,26 +194,36 @@ class ViewerStore(
val body = currentBody.value val body = currentBody.value
try { try {
val ninjaObject = assetLoader.loadNinjaObject(char) val ninjaObject = characterClassAssetLoader.loadNinjaObject(char)
val textures = assetLoader.loadXvrTextures(char, sectionId, body) val textures = characterClassAssetLoader.loadXvrTextures(char, sectionId, body)
withContext(Dispatchers.Main) {
if (clearAnimation) { if (clearAnimation) {
_currentAnimation.value = null
_currentNinjaMotion.value = null _currentNinjaMotion.value = null
} }
_currentNinjaObject.value = ninjaObject _currentNinjaObject.value = ninjaObject
_currentTextures.replaceAll(textures) _currentTextures.replaceAll(textures)
}
} catch (e: Exception) { } catch (e: Exception) {
logger.error(e) { "Couldn't load Ninja model for $char." } logger.error(e) { "Couldn't load Ninja model for $char." }
withContext(Dispatchers.Main) { _currentAnimation.value = null
_currentNinjaMotion.value = null _currentNinjaMotion.value = null
_currentNinjaObject.value = null _currentNinjaObject.value = null
_currentTextures.clear() _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
}
} }
companion object { companion object {

View File

@ -37,6 +37,13 @@ class ViewerWidget(
ViewerTab.Texture -> createTextureWidget() ViewerTab.Texture -> createTextureWidget()
} }
})) }))
addChild(SelectionWidget(
ctrl.animations,
ctrl.currentAnimation,
{ animation -> scope.launch { ctrl.setCurrentAnimation(animation) } },
{ it.name },
borderLeft = true,
))
} }
} }