この記事はこの記事はUnreal Engine (UE) Advent Calendar 2023の12日目の記事です。
概要
GameplayAbilitySystem(以後GASと表記)をC++に頼らずに
BPだけで攻撃アクションを実行できるようにします。
ゲームを1回作ってみて、スキルなどの実装がごちゃごちゃしてしまった…。
という人に刺さるやもしれないシステムです。
GASとは?
公式ドキュメント
簡単に言うと「タグ」を付けたり外したりすることで
キャラクター側が意識せずにスキルなどのアクションを実行できるという便利なものです。
~場合この攻撃で、~の場合でなければこの攻撃は出なくて…といった複雑なフラグのような条件を
タグという形で一括管理させることで、記述をシンプルにできるのが強みとなります。
注意点として、タグをうまく管理できないと非常に厄介な事になるため、
事前にどういった行動パターンがあるかなどの構想を立てたうえで実装しましょう。
セットアップ
UE5のエディタの[編集]→[プラグイン]からプラグインのウィンドウを開き、
検索で[GamePlayAbility]と入力するなどして[GameplayAbilities]プラグインにチェックを入れエディタを再起動します。
これでGASを使える準備が完了しました。簡単だね。
まずは最低限動かしてみる
今回はThirdPersonTemplateで説明していきます。
(このバージョンはUE5.1ですが、他のバージョンでも大差ないはずです)
PrintStringで文字列を表示する簡単なAbilityを作ってみる
まずコンテンツブラウザを右クリックしてGameplayAbilityブループリントを選択します。
なぜか最初から複数ありますが、GameplayAbilityを選択して作成
というわけでGA_PrintStringという、PrintStringを表示するためだけのAbilityをこれから実装します。
BPを開いたら最初から[ActivateAbility]と[OnEndAbility]の二つのイベントがあるので、以下のように繋ぎます。
[EndAbility]は終わり次第とりあえず呼んでおきましょう。お約束みたいなものです。
それが終わったら[クラスのデフォルト]を選択します。
詳細ウィンドウにいっぱい出てきますが、[AbilityTags]の右にある[編集...]を選択します。
出てきたウィンドウの[新しいゲームプレイタグを追加]を開くと名前を入力できるので
[Action.Test.PrintString]と入力して新しいタグを追加してください。
ドット[.]で区切る事である程度分別ができます。
こんな感じになればOKです。
この設定項目に関する事については後述します。
プレイヤーにAbilityを使う準備をする
まずはThirdPersonCharacterの左上にあるコンポーネントの[+追加]を押して
AbilitySystemコンポーネントを追加します。
追加したらAbilitySystemコンポーネントからGiveAbilityを呼び出して、先ほど作ったGA_PrintStringをAbilityClassに設定します。
それが終わったら任意のタイミングで同じく[AbilitySystemコンポーネント]から[TryActivateAbilityiesByTag]を呼び出し、[MakeGameplayTagContainerFromTag]ノードを以下のようにして設定してください。
このノードが基本的なAbilityを実行するノードとなります。
実行して動けば成功です。
意外と簡単でしょう?
小休憩
AbilitySystemを使ってPrintStringを表示することができましたが
「PrintStringを表示するだけでこんなに手間かける意味とは…?」
と思われるのではないでしょうか?
このシステムの本番は複数のAbilityを扱うようになってからなので、そう思うのは当然です。
というわけで、便利さの一つを体験してもらってこの記事の本文は終わりとさせてもらいます。
GASの便利さを体験しよう
ではGA_PrintStringに戻ってPrintStringの後に3秒間のDelayを挟み込みます。
そして[ActivationOwnedTags]に[Action.Test.PrintString]を設定します。
これでこのAbilityは3秒間持続したのちに終了するという挙動になります。
ここで言う「Abilityが持続する」というのは、この[ActivationOwnedTags]に設定されたタグである[Action.Test.PrintString]をキャラクターが所持している状態の事を言います。
新しいAbilityを作成する
ここで新しいAbilityを作成します。
作り方は最初に作ったものと大体同じですが、[AbilityTags]と[ActivationBlockedTags]に少し手を加えてあります。
[AbilityTags]には新しく作ったタグ(任意の名前でOK)を
[ActivationBlockedTags]には最初に作ったAbilityで使っていた[ActivationOwnedTags]のタグを指定します。
そして最初にやったのと同じようにキャラクターにAbilityを設定、実行できるようにします。
実行してみる
まず新しく作ったAbilityを実行する…普通に動きます。
古いほうのAbilityを実行する…普通に動きます。
しかし、古いほうのAbilityを実行してから3秒間の間、新しいAbilityが動いていない事がわかります。
これがAbilitySystemの良いところの一つで
「古いAbilityが実行中は新しいAbilityの実行を禁止する」という挙動を
AbilitySystem内だけで実装できたということです。
キャラクターで実装したのは
・Abilityを覚えさせる
・Abilityを実行させる
の2つだけであり、フラグもタイマーも作っていないのです。
ちなみに、キャラクターが現在所持しているタグは以下のコンソールコマンドによって可視化できます。
showdebug abilitysystem
具体的には何に使える?
アクションゲームで攻撃Abilityを作成したとします。
その攻撃Abilityはアイテム使用中には実行できない仕様でした。
アイテム使用Abilityの[ActivationOwnedTags]に[Action.Use.Item]と設定しておけば
攻撃Abilityの[ActivationBlockedTags]に設定することで、アイテム使用Abilityが終了するまで攻撃が実行できなくなります。
そしてその動作をプレイヤー側の実装を変えずに可能となります。
とても単純な言い方をすれば、キャラクターの状態に関するフラグをすべてタグに集約することで
キャラクターの中に大量の管理用変数ができる事を防止し
プレイヤーを敵が同じ基礎挙動(ジャンプや仰け反り、死亡など)を持っていれば同じタグを使いまわしたりもできます。
そしてタグの名前でAbilityを実行するため、攻撃そのものが差し変わってもキャラクターの実装には影響せず
キャラクターはただ「攻撃したい」と求めれば「今持っているAbilityの中で実行可能なもの」を行うという
非常にスッキリした実装が可能となります。
Abilityの設定タグの大まかな説明リスト
以下にざっくりとしたAbilityのタグの挙動を置いておきます。
- Ability Tags
ここに設定したタグをTryActivateAbilityiesByTagで呼び出すと動く
複数設定すると、いずれかを呼び出せば動く
呼び出し用の名前って事 - Cancel Abilites With Tag
ここに設定したタグを持つアビリティをキャンセルする
必殺技による割り込み、被ダメージでのダウン等で活躍かな - Block Abilities With Tag
このアビリティ実行中は該当タグ(AbilityTags)を持つアビリティの実行を禁止する。
後述のActivationBlockedTagsと同じ実行禁止効果だが、受動と能動という関係。
ダウン中は他の行動を禁止したいって場合はこっちを使ったほうがミスがない。 - Activation Owned Tags
アビリティ実行時に設定したタグをオーナーに付与する
今このアクション実行中だよ!ということを明示するために必要
別になくても動くっちゃ動くけど、普通は設定する - Activation Required Tags
ここで設定したタグを持っていないと実行しなくなる。
例えばダッシュ中のタグが付いていないとダッシュ攻撃は出せない。みたいな。
ダッシュ攻撃作るなら、通常攻撃はダッシュ中発動できない制限がいるのかな?
変身中なんかの行動変化はこれで切り分けできそう - Activation Blocked Tags
これに設定されたタグを持っている間はアビリティが実行できなくなる
↑のダッシュ中は通常攻撃が出ない。とか、攻撃中は攻撃できないとか。
これちゃんと設定しないと、攻撃を攻撃でキャンセルとかいうおかしな挙動ができちゃう - Target Required Tags
Activation Required Tagsと参照する対象が違うみたいだけど、違いがわからん…
どっち使えばええの…? - Target Blocked Tags
Activation Blocked Tagsと参照する対象が違うみたいだけど、違いがわからん…
どっち使えばええの…?