1
3

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

アバターの物を出し入れできるようにするEditorを作る(Animator編)|VRC用の拡張Editorを作ってみよう(6)

Last updated at Posted at 2021-12-09

この記事は前回の続きになります
VRC用の拡張Editorを作ってみよう(5)

#目次
1.準備
2.コードを書く
3.動作確認
4.次回

#1.準備
##usingを追加する
今回はアニメーターを扱うため

using UnityEditor.Animations;

を追加します
##メソッドを定義する
アニメーターを追加するためのメソッドを定義します
privateな戻り値無しのメソッドを定義します
今回の名前はAddLayerにします

private void AddLayer()
{

}

メソッドが定義できました

#2.コードを書く
##変数を宣言する
今回はAddLayerの中に変数を宣言していきます
AnimatorControllerの変数、AnimationClipの変数を二つ宣言します
AnimatorControllerの変数名はanimatorControllerにします
animatorControllerには、アバターのベースレイヤー(FXLayer)のアニメーションクリップを代入します
AnimationClipの変数名は、アニメーションオンの時、オフの時と、二つ作りたいので
onAnimation、offAnimationにします
onAnimation、offAnimationにはAddAnimationメソッドで作ったアニメーションをAssetsから取得して代入します

AnimatorController animatorController = avatar.baseAnimationLayers[4].animatorController as AnimatorController;
AnimationClip onAnimation = AssetDatabase.LoadAssetAtPath(animFolderPath + "/" + animationName + "On.anim", typeof(AnimationClip)) as AnimationClip;
AnimationClip offAnimation = AssetDatabase.LoadAssetAtPath(animFolderPath + "/" + animationName + "Off.anim", typeof(AnimationClip)) as AnimationClip;

これで宣言ができました
##レイヤー内を検索する
同じレイヤーが作成されていないか検索するためのコードを書いていきます
まずは、カウント用の変数を定義します
整数を扱う変数を定義します
今回の名前はsameLayerCountにします
変数は初期化しておきましょう(変数に0を代入します)

int sameLayerCount = 0;

変数はできたので、検索するコードを書いていきます
繰り返し処理をするため、for文を使って書きます
まずは、繰り返し回数をカウントする整数型の変数layerCountを定義して、layerCountに初期化のための0を代入します
次に、セミコロンで区切って、繰り返される条件を書きます
layerCountがanimatorControllerのレイヤーの最大に達したらループをやめるようにします
最後に、セミコロンで区切って、カウントアップされる条件を書きます
今回はlayerCountに+1します

for (int layerCount = 0; layerCount < animatorController.layers.Length; layerCount++)
{
	
}

繰り返し処理ができました
繰り返す内容を書いていきます
animatorController.layer[layerCount]でアニメーターコントローラーのレイヤーを繰り返し検索します
今回は、その中にAnimatorEditorの名前があったらカウントするコードを書きます
ですが、AnimatorEditorのままではかぶってしまうので、AnimatorEditorの後にsameLayerCountを追加します

if (animatorController.layers[layerCount].name == "AnimatorEditor" + sameLayerCount) sameLayerCount++;

繰り返し処理ができました
##レイヤーを追加する
カウントした値を元に、新しくレイヤーを追加します
AnimatotControllerLayer型の変数newLayerを作り、インスタンスを作成します
今回の値の代入はオブジェクト初期化子を使います
AnimatorControllerLayerに代入するのは
レイヤーの名前、ウェイトの値、インスタンス化したAnimatorStateMachineです
(なぜインスタンス化したAnimatorStateMachineが必要だったのかは覚えてないし今見返してもなんでかわからない。だれかわかる人教えて)
はい、追加していきます...

AnimatorControllerLayer newlayer = new AnimatorControllerLayer
{
	name = "AnimatorEditor" + sameLayerCount,
	defaultWeight = 1,
	stateMachine = new AnimatorStateMachine()
};

これで新しいレイヤーはできましたが、まだ追加ができていないので追加します
追加はAnimatorController.AddLayer(追加するレイヤー)で追加できます
今回はnewLayerを追加します

animatorController.AddLayer(newlayer);

これでレイヤーの追加ができました...?
このままレイヤーの追加ができてほしかったんですけど、このままではなぜかレイヤーに名前がはいってない...ので、追加します

//なぜかLayerの中に名前が入らないため追加
newlayer.stateMachine.name = "AnimatorEditor" + sameLayerCount;

これでやっと完成です
##アニメーターパラメーターを追加する
アニメーターを操作するためのパラメーターを追加します
animatorController.AddParameterで追加できます
追加する内容は、名前とパラメータータイプです
名前はAnimatorEditor + sameLayerCountにして、
アニメーションタイプはboolにします

animatorController.AddParameter("AnimatorEditor" + sameLayerCount, AnimatorControllerParameterType.Bool);

これでパラメーターの追加ができました
##StateとTransitionとConditionを追加する
アニメーションを設定するためのState、アニメーションを移動するためのTransition、移動する条件のConditionを追加していきます
ここでは面倒なので一度で全部説明します

1.  newlayer.stateMachine.AddState(offAnimation.name);
2.  newlayer.stateMachine.AddState(onAnimation.name);
3.  newlayer.stateMachine.states[0].state.motion = offAnimation;
4.  newlayer.stateMachine.states[1].state.motion = onAnimation;
5.  newlayer.stateMachine.states[0].state.AddTransition(newlayer.stateMachine.states[1].state);
6.  newlayer.stateMachine.states[1].state.AddTransition(newlayer.stateMachine.states[0].state);
7.  newlayer.stateMachine.states[0].state.transitions[0].hasExitTime = false;
8.  newlayer.stateMachine.states[1].state.transitions[0].hasExitTime = false;
9.  newlayer.stateMachine.states[0].state.transitions[0].AddCondition(AnimatorConditionMode.If, 0, "AnimatorEditor" + sameLayerCount);
10. newlayer.stateMachine.states[1].state.transitions[0].AddCondition(AnimatorConditionMode.IfNot, 0, "AnimatorEditor" + sameLayerCount);

###1.
newLayerのstateMachineに、offAnimationのステートを追加する
###2.
newLayerのstateMachineに、onAnimationのステートを追加する
###3.
states[0](offAnimation)のstateのmotionに、offAnimationを代入する
###4.
states[1](onAnimation)のstateのmotionに、onAnimationを代入する
###5.
states[0]からstate[1]へ遷移する
###6.
states[1]からstate[0]へ遷移する
###7.
States[0]のtransitions[0](上で追加したやつ)のhasExitTimeを無効にする
###8.
States[1]のtransitions[0](上で追加したやつ)のhasExitTimeを無効にする
###9.
States[0]のTransitions[0]に条件を追加する
###10.
States[1]のTransitions[0]に条件を追加する

これらを追加すると、StateとTransitionとConditionを追加できます
##Layerを保存する
このままでは、Unityを閉じたときに全て消えてしまうので、消えないように保存します
保存するにはAssetDatabaseのAddObjectToAssetを使います
追加する内容は、新しく作ったレイヤーのステートマシーンです
保存先はanimatorControllerにします

AssetDatabase.AddObjectToAsset(newlayer.stateMachine, animatorController);

これで保存ができるようになりました

##ボタンからメソッドを呼び出す
前作ったボタンにメソッドを追加します

if (GUILayout.Button("Animation作成"))
{
	AddAnimation();
	AddLayer();
}

追加ができました

#3.動作確認
保存してEditorを開いて
image.png
新しくアニメーションを作ります
image.png
追加されていますね
image.png
パラメーターも大丈夫そうです
次は、複数作ってみましょう
image.png
image.png
image.png
問題なさそうです
今回はこれでおわりです
お疲れさまでした

#4.次回
次回はExpression編です

VRC用の拡張Editorを作ってみよう(7)
coming soon

1
3
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
3

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?