mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
Fixed viewer animation bug: when loading a new animation, the frame rate would change back to 30 without any indication.
This commit is contained in:
parent
874cff7ae5
commit
a0bf918e42
@ -1,7 +1,9 @@
|
|||||||
package world.phantasmal.web.viewer.rendering
|
package world.phantasmal.web.viewer.rendering
|
||||||
|
|
||||||
import org.w3c.dom.HTMLCanvasElement
|
import org.w3c.dom.HTMLCanvasElement
|
||||||
|
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.NjMotion
|
import world.phantasmal.lib.fileFormats.ninja.NjMotion
|
||||||
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
|
||||||
@ -55,6 +57,11 @@ class MeshRenderer(
|
|||||||
observe(viewerStore.frame, ::frameChanged)
|
observe(viewerStore.frame, ::frameChanged)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override fun dispose() {
|
||||||
|
animation?.dispose()
|
||||||
|
super.dispose()
|
||||||
|
}
|
||||||
|
|
||||||
override fun render() {
|
override fun render() {
|
||||||
animation?.mixer?.update(clock.getDelta())
|
animation?.mixer?.update(clock.getDelta())
|
||||||
|
|
||||||
@ -63,11 +70,9 @@ class MeshRenderer(
|
|||||||
super.render()
|
super.render()
|
||||||
|
|
||||||
animation?.let {
|
animation?.let {
|
||||||
val action = it.mixer.clipAction(it.clip)
|
if (!it.action.paused) {
|
||||||
|
|
||||||
if (!action.paused) {
|
|
||||||
updateAnimationTime = false
|
updateAnimationTime = false
|
||||||
viewerStore.setFrame((action.time * PSO_FRAME_RATE_DOUBLE + 1).roundToInt())
|
viewerStore.setFrame((it.action.time * PSO_FRAME_RATE_DOUBLE + 1).roundToInt())
|
||||||
updateAnimationTime = true
|
updateAnimationTime = true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -86,23 +91,21 @@ class MeshRenderer(
|
|||||||
skeletonHelper = null
|
skeletonHelper = null
|
||||||
}
|
}
|
||||||
|
|
||||||
val ninjaObject = viewerStore.currentNinjaObject.value
|
val njObject = 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.
|
||||||
var animationTime: Double? = null
|
var animationTime: Double? = null
|
||||||
|
|
||||||
animation?.let { animation ->
|
animation?.let {
|
||||||
val mixer = animation.mixer
|
animationTime = it.action.time
|
||||||
animationTime = mixer.existingAction(animation.clip).time
|
it.dispose()
|
||||||
mixer.stopAllAction()
|
|
||||||
mixer.uncacheAction(animation.clip)
|
|
||||||
this.animation = null
|
this.animation = null
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create a new mesh if necessary.
|
// Create a new mesh if necessary.
|
||||||
if (ninjaObject != null) {
|
if (njObject != null) {
|
||||||
val mesh = ninjaObjectToSkinnedMesh(ninjaObject, textures, boundingVolumes = true)
|
val mesh = ninjaObjectToSkinnedMesh(njObject, 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.
|
||||||
@ -131,28 +134,19 @@ class MeshRenderer(
|
|||||||
this.skeletonHelper = skeletonHelper
|
this.skeletonHelper = skeletonHelper
|
||||||
|
|
||||||
// Create a new animation mixer and clip.
|
// Create a new animation mixer and clip.
|
||||||
viewerStore.currentNinjaMotion.value?.let { nj_motion ->
|
viewerStore.currentNinjaMotion.value?.let { njMotion ->
|
||||||
val mixer = AnimationMixer(mesh)
|
animation = Animation(njObject, njMotion, mesh).also {
|
||||||
mixer.timeScale = viewerStore.frameRate.value / PSO_FRAME_RATE_DOUBLE
|
it.mixer.timeScale = viewerStore.frameRate.value / PSO_FRAME_RATE_DOUBLE
|
||||||
|
it.action.time = animationTime ?: .0
|
||||||
val clip = createAnimationClip(ninjaObject, nj_motion)
|
it.action.play()
|
||||||
|
}
|
||||||
animation = Animation(mixer, clip)
|
|
||||||
|
|
||||||
val action = mixer.clipAction(clip, mesh)
|
|
||||||
action.time = animationTime ?: .0
|
|
||||||
action.play()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun ninjaMotionChanged(njMotion: NjMotion?) {
|
private fun ninjaMotionChanged(njMotion: NjMotion?) {
|
||||||
var mixer: AnimationMixer? = null
|
|
||||||
|
|
||||||
animation?.let {
|
animation?.let {
|
||||||
it.mixer.stopAllAction()
|
it.dispose()
|
||||||
it.mixer.uncacheAction(it.clip)
|
|
||||||
mixer = it.mixer
|
|
||||||
animation = null
|
animation = null
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -163,21 +157,17 @@ class MeshRenderer(
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mixer == null) {
|
animation = Animation(njObject, njMotion, mesh).also {
|
||||||
mixer = AnimationMixer(mesh)
|
it.mixer.timeScale = viewerStore.frameRate.value / PSO_FRAME_RATE_DOUBLE
|
||||||
|
it.action.play()
|
||||||
}
|
}
|
||||||
|
|
||||||
val clip = createAnimationClip(njObject, njMotion)
|
|
||||||
|
|
||||||
animation = Animation(mixer!!, clip)
|
|
||||||
|
|
||||||
clock.start()
|
clock.start()
|
||||||
mixer!!.clipAction(clip).play()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun animationPlayingChanged(playing: Boolean) {
|
private fun animationPlayingChanged(playing: Boolean) {
|
||||||
animation?.let {
|
animation?.let {
|
||||||
it.mixer.clipAction(it.clip).paused = !playing
|
it.action.paused = !playing
|
||||||
|
|
||||||
if (playing) {
|
if (playing) {
|
||||||
clock.start()
|
clock.start()
|
||||||
@ -196,12 +186,27 @@ class MeshRenderer(
|
|||||||
private fun frameChanged(frame: Int) {
|
private fun frameChanged(frame: Int) {
|
||||||
if (updateAnimationTime) {
|
if (updateAnimationTime) {
|
||||||
animation?.let {
|
animation?.let {
|
||||||
it.mixer.clipAction(it.clip).time = (frame - 1) / PSO_FRAME_RATE_DOUBLE
|
it.action.time = (frame - 1) / PSO_FRAME_RATE_DOUBLE
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private class Animation(val mixer: AnimationMixer, val clip: AnimationClip)
|
private class Animation(
|
||||||
|
njObject: NinjaObject<*>,
|
||||||
|
njMotion: NjMotion,
|
||||||
|
root: Object3D,
|
||||||
|
) : TrackedDisposable() {
|
||||||
|
private val clip: AnimationClip = createAnimationClip(njObject, njMotion)
|
||||||
|
|
||||||
|
val mixer = AnimationMixer(root)
|
||||||
|
val action: AnimationAction = mixer.clipAction(clip)
|
||||||
|
|
||||||
|
override fun dispose() {
|
||||||
|
mixer.stopAllAction()
|
||||||
|
mixer.uncacheAction(clip)
|
||||||
|
super.dispose()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
private val CAMERA_POS = Vector3(1.0, 1.0, 2.0).normalize()
|
private val CAMERA_POS = Vector3(1.0, 1.0, 2.0).normalize()
|
||||||
|
Loading…
Reference in New Issue
Block a user