All viewer animation controls are now ported.

This commit is contained in:
Daan Vanden Bosch 2021-03-26 19:56:48 +01:00
parent 2ab0baa3b5
commit 874cff7ae5
7 changed files with 105 additions and 15 deletions

View File

@ -8,7 +8,8 @@ import world.phantasmal.lib.fileFormats.ninja.NjKeyframeTrack
import world.phantasmal.lib.fileFormats.ninja.NjMotion import world.phantasmal.lib.fileFormats.ninja.NjMotion
import world.phantasmal.web.externals.three.* import world.phantasmal.web.externals.three.*
const val PSO_FRAME_RATE: Double = 30.0 const val PSO_FRAME_RATE: Int = 30
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 =
@ -27,7 +28,7 @@ fun createAnimationClip(njObject: NinjaObject<*>, njMotion: NjMotion): Animation
val times = jsArrayOf<Double>() val times = jsArrayOf<Double>()
for (keyframe in track.keyframes) { for (keyframe in track.keyframes) {
times.push(keyframe.frame / PSO_FRAME_RATE) times.push(keyframe.frame / PSO_FRAME_RATE_DOUBLE)
} }
val values = jsArrayOf<Double>() val values = jsArrayOf<Double>()
@ -114,7 +115,7 @@ fun createAnimationClip(njObject: NinjaObject<*>, njMotion: NjMotion): Animation
return AnimationClip( return AnimationClip(
"Animation", "Animation",
(njMotion.frameCount - 1) / PSO_FRAME_RATE, (njMotion.frameCount - 1) / PSO_FRAME_RATE_DOUBLE,
tracks.asArray(), tracks.asArray(),
).optimize() ).optimize()
} }

View File

@ -24,7 +24,10 @@ class ViewerToolbarController(private val store: ViewerStore) : Controller() {
val showSkeleton: Val<Boolean> = store.showSkeleton val showSkeleton: Val<Boolean> = store.showSkeleton
val playAnimation: Val<Boolean> = store.animationPlaying val playAnimation: Val<Boolean> = store.animationPlaying
val clearCurrentAnimationButtonEnabled = store.currentNinjaMotion.isNotNull() val frameRate: Val<Int> = store.frameRate
val frame: Val<Int> = store.frame
val animationControlsEnabled: Val<Boolean> = store.currentNinjaMotion.isNotNull()
val maxFrame: Val<String> = store.currentNinjaMotion.map { "/ ${it?.frameCount ?: 0}" }
val resultDialogVisible: Val<Boolean> = _resultDialogVisible val resultDialogVisible: Val<Boolean> = _resultDialogVisible
val result: Val<PwResult<*>?> = _result val result: Val<PwResult<*>?> = _result
val resultMessage: Val<String> = result.map { val resultMessage: Val<String> = result.map {
@ -42,6 +45,14 @@ class ViewerToolbarController(private val store: ViewerStore) : Controller() {
store.setAnimationPlaying(play) store.setAnimationPlaying(play)
} }
fun setFrameRate(frameRate: Int) {
store.setFrameRate(frameRate)
}
fun setFrame(frame: Int) {
store.setFrame(frame)
}
suspend fun clearCurrentAnimation() { suspend fun clearCurrentAnimation() {
store.setCurrentAnimation(null) store.setCurrentAnimation(null)
} }

View File

@ -5,11 +5,13 @@ import world.phantasmal.core.math.degToRad
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
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.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.*
import world.phantasmal.web.viewer.stores.ViewerStore import world.phantasmal.web.viewer.stores.ViewerStore
import kotlin.math.roundToInt
import kotlin.math.tan import kotlin.math.tan
class MeshRenderer( class MeshRenderer(
@ -21,6 +23,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 updateAnimationTime = true
private var charClassActive = false private var charClassActive = false
override val context = addDisposable(RenderContext( override val context = addDisposable(RenderContext(
@ -48,6 +51,8 @@ class MeshRenderer(
observe(viewerStore.currentNinjaMotion, ::ninjaMotionChanged) observe(viewerStore.currentNinjaMotion, ::ninjaMotionChanged)
observe(viewerStore.showSkeleton) { skeletonHelper?.visible = it } observe(viewerStore.showSkeleton) { skeletonHelper?.visible = it }
observe(viewerStore.animationPlaying, ::animationPlayingChanged) observe(viewerStore.animationPlaying, ::animationPlayingChanged)
observe(viewerStore.frameRate, ::frameRateChanged)
observe(viewerStore.frame, ::frameChanged)
} }
override fun render() { override fun render() {
@ -58,11 +63,13 @@ class MeshRenderer(
super.render() super.render()
animation?.let { animation?.let {
// TODO: Update current animation frame in store. val action = it.mixer.clipAction(it.clip)
// val action = it.mixer.clipAction(it.clip)
// if (!action.paused) {
// if (!action.paused) { updateAnimationTime = false
// } viewerStore.setFrame((action.time * PSO_FRAME_RATE_DOUBLE + 1).roundToInt())
updateAnimationTime = true
}
} }
} }
@ -126,8 +133,7 @@ class MeshRenderer(
// Create a new animation mixer and clip. // Create a new animation mixer and clip.
viewerStore.currentNinjaMotion.value?.let { nj_motion -> viewerStore.currentNinjaMotion.value?.let { nj_motion ->
val mixer = AnimationMixer(mesh) val mixer = AnimationMixer(mesh)
// TODO: Set time scale. mixer.timeScale = viewerStore.frameRate.value / PSO_FRAME_RATE_DOUBLE
// mixer.timeScale = this.store.animation_frame_rate.val / PSO_FRAME_RATE;
val clip = createAnimationClip(ninjaObject, nj_motion) val clip = createAnimationClip(ninjaObject, nj_motion)
@ -170,8 +176,8 @@ class MeshRenderer(
} }
private fun animationPlayingChanged(playing: Boolean) { private fun animationPlayingChanged(playing: Boolean) {
animation?.let { animation -> animation?.let {
animation.mixer.clipAction(animation.clip).paused = !playing it.mixer.clipAction(it.clip).paused = !playing
if (playing) { if (playing) {
clock.start() clock.start()
@ -181,6 +187,20 @@ class MeshRenderer(
} }
} }
private fun frameRateChanged(frameRate: Int) {
animation?.let {
it.mixer.timeScale = frameRate / PSO_FRAME_RATE_DOUBLE
}
}
private fun frameChanged(frame: Int) {
if (updateAnimationTime) {
animation?.let {
it.mixer.clipAction(it.clip).time = (frame - 1) / PSO_FRAME_RATE_DOUBLE
}
}
}
private class Animation(val mixer: AnimationMixer, val clip: AnimationClip) private class Animation(val mixer: AnimationMixer, val clip: AnimationClip)
companion object { companion object {

View File

@ -11,6 +11,7 @@ import world.phantasmal.observable.value.list.ListVal
import world.phantasmal.observable.value.list.mutableListVal import world.phantasmal.observable.value.list.mutableListVal
import world.phantasmal.observable.value.mutableVal import world.phantasmal.observable.value.mutableVal
import world.phantasmal.web.core.PwToolType import world.phantasmal.web.core.PwToolType
import world.phantasmal.web.core.rendering.conversion.PSO_FRAME_RATE
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
@ -42,6 +43,8 @@ class ViewerStore(
// Settings. // Settings.
private val _showSkeleton = mutableVal(false) private val _showSkeleton = mutableVal(false)
private val _animationPlaying = mutableVal(true) private val _animationPlaying = mutableVal(true)
private val _frameRate = mutableVal(PSO_FRAME_RATE)
private val _frame = mutableVal(0)
// Ninja concepts. // Ninja concepts.
val currentNinjaObject: Val<NinjaObject<*>?> = _currentNinjaObject val currentNinjaObject: Val<NinjaObject<*>?> = _currentNinjaObject
@ -63,6 +66,8 @@ class ViewerStore(
// Settings. // Settings.
val showSkeleton: Val<Boolean> = _showSkeleton val showSkeleton: Val<Boolean> = _showSkeleton
val animationPlaying: Val<Boolean> = _animationPlaying val animationPlaying: Val<Boolean> = _animationPlaying
val frameRate: Val<Int> = _frameRate
val frame: Val<Int> = _frame
init { init {
for (path in listOf(ViewerUrls.mesh, ViewerUrls.texture)) { for (path in listOf(ViewerUrls.mesh, ViewerUrls.texture)) {
@ -198,6 +203,20 @@ class ViewerStore(
_animationPlaying.value = playing _animationPlaying.value = playing
} }
fun setFrameRate(frameRate: Int) {
_frameRate.value = frameRate
}
fun setFrame(frame: Int) {
val maxFrame = currentNinjaMotion.value?.frameCount ?: Int.MAX_VALUE
_frame.value = when {
frame > maxFrame -> 1
frame < 1 -> maxFrame
else -> frame
}
}
private suspend fun loadCharacterClassNinjaObject(clearAnimation: Boolean) { private suspend fun loadCharacterClassNinjaObject(clearAnimation: Boolean) {
val char = currentCharacterClass.value val char = currentCharacterClass.value
?: return ?: return

View File

@ -28,12 +28,34 @@ class ViewerToolbar(private val ctrl: ViewerToolbarController) : Widget() {
), ),
Checkbox( Checkbox(
label = "Play animation", label = "Play animation",
enabled = ctrl.animationControlsEnabled,
checked = ctrl.playAnimation, checked = ctrl.playAnimation,
onChange = ctrl::setPlayAnimation, onChange = ctrl::setPlayAnimation,
), ),
IntInput(
label = "Frame rate:",
enabled = ctrl.animationControlsEnabled,
value = ctrl.frameRate,
onChange = ctrl::setFrameRate,
min = 1,
max = 240,
step = 1,
),
IntInput(
label = "Frame:",
enabled = ctrl.animationControlsEnabled,
value = ctrl.frame,
onChange = ctrl::setFrame,
step = 1,
),
Label(
enabled = ctrl.animationControlsEnabled,
textVal = ctrl.maxFrame,
),
Button( Button(
className = "pw-viewer-toolbar-clear-animation",
text = "Clear animation", text = "Clear animation",
enabled = ctrl.clearCurrentAnimationButtonEnabled, enabled = ctrl.animationControlsEnabled,
onClick = { scope.launch { ctrl.clearCurrentAnimation() } }, onClick = { scope.launch { ctrl.clearCurrentAnimation() } },
), ),
) )
@ -45,4 +67,16 @@ class ViewerToolbar(private val ctrl: ViewerToolbarController) : Widget() {
onDismiss = ctrl::dismissResultDialog, onDismiss = ctrl::dismissResultDialog,
)) ))
} }
companion object {
init {
@Suppress("CssUnusedSymbol")
// language=css
style("""
.pw-viewer-toolbar > .pw-toolbar > .pw-viewer-toolbar-clear-animation {
margin-left: 6px;
}
""".trimIndent())
}
}
} }

View File

@ -15,6 +15,7 @@ open class Button(
visible: Val<Boolean> = trueVal(), visible: Val<Boolean> = trueVal(),
enabled: Val<Boolean> = trueVal(), enabled: Val<Boolean> = trueVal(),
tooltip: Val<String?> = nullVal(), tooltip: Val<String?> = nullVal(),
private val className: String? = null,
private val text: String? = null, private val text: String? = null,
private val textVal: Val<String>? = null, private val textVal: Val<String>? = null,
private val iconLeft: Icon? = null, private val iconLeft: Icon? = null,
@ -29,6 +30,9 @@ open class Button(
override fun Node.createElement() = override fun Node.createElement() =
button { button {
className = "pw-button" className = "pw-button"
this@Button.className?.let { classList.add(it) }
onmousedown = onMouseDown onmousedown = onMouseDown
onmouseup = onMouseUp onmouseup = onMouseUp
onclick = onClick onclick = onClick

View File

@ -13,6 +13,7 @@ class FileButton(
visible: Val<Boolean> = trueVal(), visible: Val<Boolean> = trueVal(),
enabled: Val<Boolean> = trueVal(), enabled: Val<Boolean> = trueVal(),
tooltip: Val<String?> = nullVal(), tooltip: Val<String?> = nullVal(),
className: String? = null,
text: String? = null, text: String? = null,
textVal: Val<String>? = null, textVal: Val<String>? = null,
iconLeft: Icon? = null, iconLeft: Icon? = null,
@ -20,7 +21,7 @@ class FileButton(
private val accept: String = "", private val accept: String = "",
private val multiple: Boolean = false, private val multiple: Boolean = false,
private val filesSelected: ((List<File>) -> Unit)? = null, private val filesSelected: ((List<File>) -> Unit)? = null,
) : Button(visible, enabled, tooltip, text, textVal, iconLeft, iconRight) { ) : Button(visible, enabled, tooltip, className, text, textVal, iconLeft, iconRight) {
override fun interceptElement(element: HTMLElement) { override fun interceptElement(element: HTMLElement) {
element.classList.add("pw-file-button") element.classList.add("pw-file-button")