• Runtimes
  • アニメに影響を与えずにループ終端を変更する、TrackTimeで現在のフレームを正しく変更する方法

連日お世話になっております。

上記のような頭を掻く男性のアニメがあります。
 ・30フレームに到達すると、30-95フレームをループ(頭をかくポーズを維持する)
 ・ループ解除メソッドを起動するとループ=falseにして本来の最終フレームまで動く
そのような挙動を作ろうとしています。

ひとまずこのように TrackEntry の AnimationStart と AnimationEnd をSpineのFPS30で割った箇所に変更してみました。
30-95フレームでループするようになったのですが、95フレームのループ終端を本来のアニメの最終フレームに戻す処理が上手くいきません

ご覧のように最終フレームまで瞬間移動してしまいます
(コンテキストメニューでメソッドを起動しているので、アニメが一時停止するのは無視してください)
trackTimeを95に指定すればそこのフレームに移動してアニメがそこから再生されると考えたのですが、それも上手くいきませんでした。

上記のようなTrackEntryプロパティの使い方は間違っていますか?
アニメに影響を与えずにループ終端を変更する方法と、描画するアニメ現在のフレームを正しく変更する方法が知りたいです。

何卒お願いいたします。

  • Misaki đã trả lời bài viết này.
    Related Discussions
    ...

    アニメのループ終端を変更し、現在のフレームを正しく設定するための方法について説明します。

    まず、ループ終端を変更するためには、TrackEntryAnimationEndプロパティを使用します。以下のコード例では、30フレームから95フレームまでのループを設定し、ループ解除時にアニメーションの最終フレームまで再生する方法を示します。

    float fps = 30;
    TrackEntry loopEntry = skeletonAnimation.state.SetAnimation(0, "animationName", true);
    loopEntry.AnimationStart = 30 / fps;
    loopEntry.AnimationEnd = 95 / fps;
    
    // ループ解除メソッド
    void StopLooping() {
        loopEntry.Loop = false;
        loopEntry.AnimationEnd = loopEntry.animation.duration;
    }

    次に、現在のフレームを正しく設定するためには、TrackEntryTrackTimeプロパティを使用します。以下のコード例では、アニメーションの現在のフレームを95フレームに設定する方法を示します。

    void SetCurrentFrame(float frame) {
        loopEntry.TrackTime = frame / fps;
    }

    この方法で、アニメーションのループ終端を変更し、現在のフレームを正しく設定することができます。

    • Misaki đã thích điều này.

    koyu2 Spinebotの回答は正しいように見えますのでもしこれでうまくいかなければ再度お知らせください。また、前に投稿されたスレッドでお伝えすれば良かったのですが、コードを投稿する際はできるだけ画像ではなくてテキストで投稿してください。そのほうがこちらから回答をする際に投稿されたコードを元にテストしたり修正したコードを返答したりしやすくなるため、次回からはそうしていただけますと幸いです。

    お返事ありがとうございます。

    Spinebotは、質問内容の記述の焼き直しでしかなく、コードの主旨に変わりはありません。
    AnimationStartとEndでループ区間を指定し、AnimationEndをanimation.durationで元アニメの最終フレームに指定し直す、TrackTime で指定フレームへ飛ぶ。
    同様のことを行って依然問題が発生しております。

    問題が再現するプロジェクトをメールアドレスにお送りしました。
    下記はお送りした最小構成用のコードと問題の再現手順です。
    お送りする最小構成を構築のために、コードを変更しましたが、上記の内容自体に変更はありません。

    using Spine;
    using Spine.Unity;
    using UnityEngine;
    
    [DefaultExecutionOrder(-1)]
    public class TestSpineAnim : MonoBehaviour
    {
        [SerializeField] string animName  = "";
        [SerializeField] string animName_AddTrack = "";
    
        public float loopStartFlame = 30;
        public float loopEndFlame = 95;
        public float restartFlame = 95;
    
        SkeletonAnimation skeletonAnimation;
        TrackEntry track0_Main;
    
        private void Start()
        {
            skeletonAnimation = GetComponent<SkeletonAnimation>();
            FlameLoop();
        }
    
        /// <summary> ループ終端を元々のアニメの最終フレームに戻します。直後のアニメが確認しやすいようTimeScaleは0になります。
        /// コンポーネント右クリックからコンテキストメニューでメソッド実行してください </summary>
        [ContextMenu("FlameLoop_Remove")]
        public void _FlameLoop_Remove()
        {
            Debug.Log("<color=yellow>★FlameLoop_Remove_ AnimationTime:</color>" + track0_Main.AnimationTime);
    
            track0_Main.AnimationEnd = track0_Main.Animation.Duration;
            //track0_Main.TrackTime = (restartFlame / 30);
            track0_Main.Loop = false;
            Time.timeScale = 0f;
        }
    
        private void Update()
        {
            skeletonAnimation.skeleton.SetBonesToSetupPose();
        }
    
        /// <summary> 30~95F内をループします。TimeScaleは通常に戻ります </summary>
        [ContextMenu("FlameLoop")]
        void FlameLoop()
        {
            if (animName != "") SetAnim(animName, true, 0);
            track0_Main.AnimationStart = (loopStartFlame / 30);
            track0_Main.AnimationEnd = (loopEndFlame / 30);
            Time.timeScale = 1.0f;
        }
    
        /// <summary> ループ箇所を変更しない元のアニメです。確認用にお使いください。 </summary>
        [ContextMenu("OriginalAnim")]
        void OriginalAnim()
        {
            if (animName != "") SetAnim(animName, true, 0);
            Time.timeScale = 1.0f;
        }
    
    
        public void SetAnim(string animName, bool loop, int track)
        {
            skeletonAnimation.skeleton.SetToSetupPose();
            track0_Main = skeletonAnimation.AnimationState.SetAnimation(track, animName, loop);
            skeletonAnimation.AnimationState.Apply(skeletonAnimation.Skeleton);
        }
    }

    1.TestLoopChangeシーンを起動
    2.頭を掻く男性のアニメが表示される。起動した状態ではAnimationStart:30、AnimationEnd:95Fをループする「FlameLoop」状態になっている
    3.ヒエラルキーのTestManオブジェクトの「TestSpineAnim」コンポーネントを右クリックし、下部の「FlameLoop_Remove」メソッドを起動する。
    4.AnimationEnd = Animation.Duration に設定され、ループ終端がアニメの最終フレーム(154F)に設定される。同時にTimeScaleが0にされるのでループ終端変更直後のアニメが確認できる

     ・単にループ終端が変更されるだけなら描画中のアニメに変化はないはずが、最終フレームまでフレームが飛んだり、手を戻す動作が一部飛んだりする
     ・この現象は「FlameLoop_Remove」を呼び出したタイミングで変化する。(最終フレームまで飛んだり飛ばなかったりします。傾向としては起動してから少し間を置いてFlamLoopRemoveを起動するとフレームが飛ぶように見えます)

    5.TestSpineAnim 32行目のコメントアウト、「track0_Main.TrackTime = (restartFlame / 30);」をコメント解除する
     ・restartFlameは「FlameLoop」で設定されたループ終端と同じ95Fに設定されています。
     → しかし、95Fから再開せず、手を下までおろした状態までフレーム飛びが発生します。

    以上の検証からランタイムに下記の問題が起こっているように見えます
    ・Animation.AnimationEndなどでループを操作すると描画中のアニメにフレーム飛びの問題が発生する
    ・TrackTimeで指定したフレームに正しく移動ができない

    お送りしたプロジェクトをご確認頂ければ幸いです。
    お手数おかけしますがよろしくお願いいたします。

    • Misaki đã trả lời bài viết này.

      koyu2 再現プロジェクトをご送付いただきありがとうございます。問題を再現できることを確認しましたが、spine-unityの担当であるHaraldが緊急の用件で不在のため詳しい回答についてはお時間をいただく可能性があります。恐れ入りますが今しばらくお待ちいただけますと幸いです。

      @koyu2 Thanks for sending the reproduction project.

      In general it's not a good idea to trigger anything timing-relevant from a Unity Editor GUI method like [ContextMenu]. This will never be called at Runtime and does not lie within the ordered Update cycle of Unity.

      Apart from that, the problem seems to be that you are setting trackEntry.Loop = false which may be long after your trackEntry.AnimationEnd has been exceeded. Then at the next Update() call, the last frame of your animation will be shown (as TrackTime is larger than Animation.Duration and loop is false), as you can see in your reproduction project. Changing trackEntry.Loop = false mid-animation does not adjust other time values.

      Regarding your second question, item (5), you are incorrectly setting trackEntry.TrackTime = restartFlame / 30 but you should be instead setting it to trackEntry.TrackTime = 0 when you already have set the loop offset via trackEntry.AnimationStart = restartFlame / 30. Otherwise you receive the start-offset twice.

      プロジェクトの確認とお返事ありがとうございます!
      TrackEntryとAnimationEndについてもう少し質問させてください。
      まず下記の画像を見てください

      ◆Question1
      TrackTimeはLoop = trueでも増え続けていませんか?

      the problem seems to be that you are setting trackEntry.Loop = false which may be long after your trackEntry.AnimationEnd has been exceeded.

      はい。設定しています。しかしfalseになるのは30-95Fのループを解除する命令を呼び出した時です。
      上記の画像のように最初のSetAnimationはLoop=trueで呼び出されており、30-95Fをループしています

          public void _FlameLoop_Remove()
          {
              Debug.Log("--------- _FlameLoop_Remove-------------------------------------------------");
              Debug.Log("<color=yellow>★TrackTime:</color>" + track0_Main.TrackTime);
              Debug.Log("<color=#00ffff>★AnimationTime:</color>" + track0_Main.AnimationTime);
              Debug.Log("<color=green>★Duration:</color>" + track0_Main.Animation.Duration);
              Debug.Log("<color=red>★AnimationEnd:</color>" + track0_Main.AnimationEnd);
            
              track0_Main.AnimationEnd = track0_Main.Animation.Duration;
      
              Debug.Log("------- track0_Main.AnimationEnd = track0_Main.Animation.Duration; ---------------------------------------------------");
              Debug.Log("<color=yellow>★TrackTime:</color>" + track0_Main.TrackTime);
              Debug.Log("<color=#00ffff>★AnimationTime:</color>" + track0_Main.AnimationTime);
              Debug.Log("<color=green>★Duration:</color>" + track0_Main.Animation.Duration);
              Debug.Log("<color=red>★AnimationEnd:</color>" + track0_Main.AnimationEnd);
      
              //track0_Main.TrackTime = (restartFlame/30);
              //track0_Main.TrackTime = 0;
      
              Debug.Log("--------- track0_Main.TrackTime = NoChange or 0 or RestartFleame;-------------------------------------------------");
              Debug.Log("<color=yellow>★TrackTime:</color>" + track0_Main.TrackTime);
              Debug.Log("<color=#00ffff>★AnimationTime:</color>" + track0_Main.AnimationTime);
              Debug.Log("<color=green>★Duration:</color>" + track0_Main.Animation.Duration);
              Debug.Log("<color=red>★AnimationEnd:</color>" + track0_Main.AnimationEnd);
      
              track0_Main.Loop = false;
              //Time.timeScale = 0f;
          }

      私は検証のため上記のコードを書き、DebugLogに出力しました。
      Loop = trueの状態で呼び出されたログはTrackTime:12.49917 となっており、これは元々のアニメの総時間を超えています。

      ◆Question2
      TrackTime = 0 == AnimationStartですか?

      TrackTimeを0にするとAnimationTimeが1になりました。
      TrackTime == 0 == AnimationStartの時間に戻るように見えます。

      私は当初、TrackTime/FPSを指定すればエディタのドープシートの指定したフレームへ飛べると考えていました。しかし厳密には違うように見えます。
      AnimationStartが0のままであれば、エディタのドープシートのフレーム数と一致しますが、
      このアニメではAnimationStart=30に変更したので、
      元々のアニメの95Fからアニメを再開したい場合、restartは65Fを指定するか、
      AnimationStartを0に戻した上でrestartは95Fに指定する必要があるように見えます

      ◆Question3
      AnimationEndが変更された時点でTrackTimeは適用されますか?
      単にループ終端を変更するだけで現在再生中のアニメの描画フレームは変更されないと思っていましたがそうではないようです。

      Otherwise you receive the start-offset twice.

      とはAnimationEndが変更された時点でTrackTimeは適用される事を意味していますか?
      もしそうであれば、Loop = trueでもTrackTimeが増え続けるのは問題を起こします。

        koyu2 最方便的方法是,在spine把动画切割成三段,开始-循环-结束,从【开始】切换成【循环】之后,让他重复播放就行了

        • koyu2 đã trả lời bài viết này.

          yugutou

          たしかにStartアニメ、Loopアニメ、Endアニメの3つに分割すれば話は簡単です。
          しかし、アニメの数は3倍になり、動きに修正が入るとその度に複数のアニメを修正しなければなりません。アニメ数の増加は管理を難しくします。

          アニメ分割ではなく、一つのアニメをTrackEntryでループ管理したい理由は、そこにあります。

          @koyu2 Regarding the question in the screenshot:
          "Is this what TrackTime is?"

          No. The play-head at frame 35 is more related to TrackEntry.AnimationTime, however it may increase past the end on non-looping animations.

          TrackTime is described here:
          https://esotericsoftware.com/spine-api-reference#TrackEntry-trackTime
          "Current time in seconds this track entry has been the current track entry. The track time determines animationTime."
          In other words, TrackTime is relative to TrackEntry.AnimationStart.

          TrackTime is ever increasing, relative to AnimationStart and together with loop (and AnimationEnd if looping) it determines whether the result AnimationTime timepoint is inside your animation, or uses AnimationEnd as last frame when not looping and exceeding the animation duration.

          koyu2 ◆Question1
          TrackTimeはLoop = trueでも増え続けていませんか?
          Doesn't TrackTime keep increasing even with Loop = true?

          Yes, TrackTime is ever increasing. See the documentation description: "Current time in seconds this track entry has been the current track entry."

          koyu2 はい。設定しています。しかしfalseになるのは30-95Fのループを解除する命令を呼び出した時です。
          上記の画像のように最初のSetAnimationはLoop=trueで呼び出されており、30-95Fをループしています

          I'm afraid machine translation failed to translate this part of the question properly, unfortunately I don't understand this part. If the question is still unresolved, could you please describe this again in more detail or other words?

          koyu2 ◆Question2
          TrackTime = 0 == AnimationStartですか?

          Yes, see the referenced documentation above.

          koyu2 元々のアニメの95Fからアニメを再開したい場合、restartは65Fを指定するか、
          AnimationStartを0に戻した上でrestartは95Fに指定する必要があるように見えます
          If you want to restart the anime from 95F of the original anime, specify 65F for restart, or
          It looks like you need to set AnimationStart back to 0 and then specify restart to 95F.

          See above. You may also want to check out the code of AnimationTime here, likely code is easier to read than prose text.

          koyu2 Question3
          AnimationEndが変更された時点でTrackTimeは適用されますか?
          単にループ終端を変更するだけで現在再生中のアニメの描画フレームは変更されないと思っていましたがそうではないようです。
          Is TrackTime applied once AnimationEnd is changed?
          I thought that simply changing the end of the loop would not change the drawing frame of the currently playing anime, but that doesn't seem to be the case.

          TrackTime remains the same if Time.Scale is 0, it adds 0 to TrackTime every frame. However, as long as the SkeletonAnimation component is updated normally, the resulting AnimationTime will be evaluated, leading to a different result if you change parameters like loop, AnimationStart, etc.

          もしそうであれば、Loop = trueでもTrackTimeが増え続けるのは問題を起こします。
          If so, it would be a problem if TrackTime keeps increasing even when Loop = true.

          It is not increasing with Time.Scale = 0 since 0 is added each frame, it however remains e.g. 12.3 if it exceeded the animation end before and was 12.3 when setting Time.Scale = 0.

          • koyu2 đã trả lời bài viết này.

            Harald
            こんにちはHaraldさん。
            詳しい解説ありがとうございます!

            "Is this what TrackTime is?"
            No. The play-head at frame 35 is more related to TrackEntry.AnimationTime, however it may increase past the end on non-looping animations.

            把握しました!
            私はTrackTimeの値はこのplay-headが指す現在のフレームで、それを直接操作できるとリファレンスから読み取っていました。
            しかしこのplay-headはTrackTimeではなく、AnimationTimeで、それはTrackTimeを通して操作することがわかりました。

            
                public void _FlameLoop_Remove_Refine()
                {
                    float currentAnimationTime = track0_Main.AnimationTime;         // For some reason, when AnimationEnd is changed, AnimationTime is also changed; it must be cached before End or Start is changed.
                    track0_Main.AnimationEnd = track0_Main.Animation.Duration;      // AnimationTime must be cached before changing AnimationEnd or AnimationStart
            
            
                    // -----★ If you want to change only the end of the loop without changing the drawing frame of the looping animation
                    track0_Main.AnimationStart = 0; 
                    track0_Main.TrackTime = currentAnimationTime;
            
            
                    // -----★ If you want to change the loop end and then move to a specific frame
                    track0_Main.AnimationStart = 0;
                    track0_Main.TrackTime = restartFlame / fps;
            
                    track0_Main.Loop = false;
                }

            教えてもらったことを手がかりにして、私のやりたいことが上記コードで実現できました
            すでにリファレンスを何度も読んで試行錯誤をしていましたが、解説なしでリファレンスの意図を正確に把握するのは非常に困難でした。

            1.TrackTimeはドープシートのplay-headではない。特定のフレームに移動するためには、単純にTrackTimeに移動したいフレームを代入するだけではできない
            2.TrackTimeを代入して得られるAnimationTimeはStartからの相対値ということ。そのため、AnimationStartを変更していた場合、0にリセットしなければTrackTimeを代入しても期待したフレームに移動しない
            3.AnimationEndが変更されるとなぜかAnimationTimeが即時変更される。そのためEndやStart変更する前にキャッシュしなければ、ループ終端変更前の正しいcurrentAnimationTimeを取得できない
            4.TrackTimeはLoop=falseでも値は増え続ける。エディタ上のplay-headのようにループ中は数値が戻らず、単純にAnimationEndを変更するだけではループ中溜まり続けたTrackTimeが次のUpdateで適用される

            特にこの4つは詳しい仕組みを把握するのが難しかったので、助言を頂けて助かりました!
            ありがとうございます!

            • Harald đã thích điều này.

            @koyu2 Glad to hear you've figured it out, thanks for getting back to us!