10
10

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 5 years have passed since last update.

【UE4】L4DのAIDirectorを真似てメタAIを作ってみる その2

Posted at

#続きものです
ここの内容はこちらの続きとなっておりますので併せてお読みください。

#ここでの内容
前回はEQSを用いて

  • プレイヤーから遠い位置
  • プレイヤーから見えない位置
  • プレイヤーの元へとたどり着ける位置
    の3要件を満たす敵の出現処理を実装しました。
    今回は**プレイヤーの感情強度(Emotional Intensity)**というパラメータを用意して動的な出現頻度調整を実装していきます。

#必要な機能のリストアップ
動的な出現頻度調整処理に必要な機能をリストアップします。

-感情強度(Emotional Intensity)に基づく敵の出現頻度動的調整
-感情強度が上がる要因は以下の2つ
--敵からダメージを受けた
--プレイヤーの近くで敵が死亡した(プレイヤーが追い詰められている)
-感情強度が最大になると3~5秒間感情強度を最大に維持し続ける
-維持後は徐々に感情強度を下げていく(この間も要因による感情強度上昇あり)
-感情強度が下がりきった後の30~45秒間は敵の出現は行わない

L4DのAIDirectorには感情強度を元にした**4つのステート(状態)**があります。
Build Up State - 感情強度が上昇する状態
Sustain Peek State - 感情強度が最大になり数秒間それを維持する状態
Peek Fade State - 維持後の徐々に感情強度を減衰させる状態
Relax State - 感情強度が下がりきった後の数十秒間敵を出現させない状態
一見リストアップした機能は出現処理に比べると複雑に見えるかもしれませんが各機能は状態ごとに分ける事が出来ます。
列挙した各状態に合わせた処理を実装することで意外とシンプルに実装出来ます。

#Build Up State の実装
ここから各状態ごとに実装していきますが、その前に各状態を表す列挙型を用意します。
コンテンツブラウザの新規追加 > ブループリント > 列挙型を選択します。名前はENUM_PaceStateとします。
27.PNG
ENUM_PaceStateを開くとエニュメレーターとあり右端に新規ボタンがありますのでボタンを4回押します。
すると以下のようになるはずです。
28.PNG
上から順に名前を
BuildUp
SustainPeek
PeekFade
Relax
と変更します。
以上で各状態を表す列挙型の用意が出来ました。

次はAIDirectorのコントローラーブループリントを開きます。(前回で敵の出現処理を実装したブループリントです)
カスタムイベントを追加します。名前はUpdate Stateとします。
イベント Tickと繋げるためにイベントを OnQueryFinishedEvent とバインドのピンを引っ張りUpdate Stateと検索し選択します。
それが出来たらENUM_PaceState型の変数を作成します。名前はPaceStateとします。
変数を作成出来たらPaceState変数をブループリント上にドラッグ&ドロップしてゲットを選択します。
ゲットしたPaceStateのピンを引っ張りSwitchと検索欄に打ち込みます。候補にENUM_PaceState をオンと出ますのでそれを選択します。
29.PNG
以下のようになればOKです。
30.PNG

次はBuildUp状態の処理を実装するためにカスタムイベントの追加と必要な変数を追加していきます。
カスタムイベントの追加を選択し名前をBuildUpとします。
Float型の変数を2つ追加します。名前をそれぞれSurviverIntensityMaxSurviverIntensityとします。
31.png
MaxSurviverIntensityですがデフォルト値100.0に変更してください。

さてBuildUp状態ですが、この状態の仕事は上昇するプレイヤー感情強度の監視という役目を担います。常に感情強度をチェックし、それが最大まで達するとSustain Peek状態へと自身の状態を変更するというお仕事になります。
では、感情強度が最大まで達したらSustainPeek状態へと変化するようにします。
カスタムイベントBuildUpのピンを引っ張りブランチを選択します。
SurviverIntensityとMaxSurviverIntensityを2つともゲットして、SurviverIntensityのピンを引っ張り**>=**と打ち込み選択します。

=ノードの余った所にMaxSurviverIntensityを繋ぎ、返り値ピンをブランチのConditionに繋ぎます。
ブランチノードの真ピンを引っ張りSet PaceStateと打ち込み選択します。
新たにセットする状態はSustain Peekです。
すると以下の様な形になるはずです。
32.PNG
以上でBuildUp状態の実装は終了です。内容はかなりシンプルですね。常にブランチでチェックしてTRUEなら状態を変えるというだけです。
それと忘れずにカスタムイベントUpdateStateの方にカスタムイベントBuildUpを呼び出すようにしてください。
33.PNG

##Intensityの上昇
BuildUp状態の実装が終わったので次は実際にIntensityを上げる処理を実装していきます。
Intensityが上昇する要因は以下の2つです。

  • 敵からダメージをもらう
  • 敵死亡時のプレイヤーとの距離の近さ
    まずは敵からダメージをもらうとIntensityが上昇するという処理を実装します。
    AIDirectorのコントローラーブループリントに新しく関数を追加します。名前はAddIntensityとします。
    AddIntensity関数の実装は以下の通りです。
    34.PNG
    MinFloatを通すことでIntensityの最大値である100を超えないようにしています。
    次にプレイヤーであるTopDownCharacterブループリントを開きます。
    イベントグラフの下の方にAnyDamageイベントがあるはずです。
    AnyDamageイベントが呼ばれるとAIDirectorのコントローラーブループリントのAddIntensity関数を呼び出すようにします。
    35.PNG
    AddIntensity関数のValueを0から変更することを忘れないで下さい。
    これでプレイヤーが敵からのダメージを受けるとIntensityが上昇するようになりました。
    実際に敵が攻撃をする処理は下地プロジェクトを落とされた方なら既に実装されています。
    実装内容は簡単で敵がプレイヤーに一定距離以内に近づくとApplyDamageを呼び出しているだけです。

次に敵死亡時のプレイヤーとの距離の近さでIntensityが上昇するという処理を実装します。
敵キャラのブループリントBP_Infectedを開くとイベントグラフの下の方にOnComponentHitイベントがあります。幾つかの処理がピンで繋がっているので、一番最後のDeadノードから以下のように処理を追加します。
36.PNG
敵とプレイヤーの距離を反比例の式を通す事で死亡時にプレイヤーに近いほど多くのIntensityが生じるという処理が実装出来ます。

##Build Up状態の動作確認
BuildUp状態の実装と2つの要因によるIntensityの上昇の実装が出来ましたので動作確認をしてみます。
AIDirectorのコントローラーブループリントにPrintStringでIntensityの値とSustain Peek状態に遷移した時に「Peek !!」と表示するようにします。
37.PNG
画像のように実装出来たら実際にプレイをして、動作確認をします。
敵が最接近すると値が1ずつ上がり、敵を倒した距離に応じて上昇値が変化します。
そしてIntensityが100に達した時画面に「Peek !!」の文字が表示されればOKです。
38.PNG

#Sustain Peek状態の実装
Sustain Peek状態はBuild Up状態で最大にまで達した感情強度を数秒間最大値を維持し続ける仕事をします。
感情強度が最大まで達した時L4Dでは50~100体ほどのゾンビが常にレベル上に存在するようにスポーンしプレイヤーを襲います。
こちらでも同じように感情強度がピークに達した時3~5秒間レベル上に常に50体は存在するようにスポーンさせるようにします。

Build Up状態の手順と同様にカスタムイベントを追加します。名前はSustain Peekとします。ここで新しく変数を追加します。Float型で名前はPeek Durationとします。デフォルト値5にしてください。PeekDurationはピーク状態の維持を何秒間続けるかを表す変数です。
イベントと変数が追加出来たら以下のように実装します。
39.PNG
Sustain Peekイベントに入るとまずはブランチを通ります。Peek Durationのデフォルト値は5なので真の処理に移行します。そこではSurviver IntensityをMax Surviver Intensityの値に置き換え1秒毎にPeek Durationを1ずつ減らすといった内容です。ちなみに**--ノードDecrement**と打てば候補に上がります。
Peek Durationが0未満になるとブランチの偽に移行します。これはPeek Duration秒経過したということを意味しますのでPeek Durationにデフォルト値5をセットしてPeek StateをPeek Fadeにセットします。

続いて常にレベル上に指定数の敵を存在させる出現処理を実装します。
カスタムイベントを追加し名前をEnemy Spawningとしてください。
そして以下のように実装します。
40.PNG
内容は単純です。呼ばれる度に敵キャラを検索しその個数を指定したMax Spawnと比較します。
真であればMax Spawnから検索結果の個数分引き、不足分を求めてForLoopでSpawn AIFrom Classを呼び出し出現させます。

以上でSustain Peek状態とレベル上に常に指定数敵が存在するようにするという処理が出来ましたので呼び出し部分に追加していきます。
以下のようになります。
41.PNG
線で囲まれている部分が追加部分です。
Sustain Peekの処理の追加に伴ってBuild Up状態の後にもEnemy Spawningを呼び出しています。これによりOnQueryFinishedEvent_イベント0に書いていた敵の出現処理が不要になります。
42.PNG
線で囲まれていた部分は繋がっていましたが、それを切りました。

では実際にプレイしてみて動作確認をします。
初めは3体が連なってこちらに向かって移動し、攻撃をしてきます。
こちらが敵を倒したり攻撃を受けたりするうちにIntensityが上昇し最大まで達した時、一気に大量の敵がこちらへ向かってくるはずです。
43.PNG
44.PNG
AIDirectorはPeek時には5秒間敵の総数を最大に維持し続けるので、倒しても倒してもワラワラとプレイヤーの元へと集まってきます。しかしPeekが終わるとパタッと敵が出現せず最後にはレベル上に敵が存在しない状態になります。

#Peek Fade状態の実装
Build UpとSustain Peekを実装したことでなかなか面白いゲーム展開が出来たのでは無いでしょうか。しかし、これだけの内容だと敵は常に出現しプレイヤーを追いかけ続けてしまいます。倒しても倒しても無限に湧き続けるという状態はプレイヤーに対し大きなストレスを与えてしまいます。L4Dではそういった大きなストレスを与え続けないようにしています。それが、これから実装するPeek Fade状態とRelax状態です。

ではPeek Fade状態を実装していきます。
Peek Fadeの内容はかなり単純です。Intensityが0になるまでIntensityを減算し続けるだけです。
まずはIntensityを減算する処理を作ります。関数を追加し名前をSubIntensityとします。内容は以下の通りです。
45.PNG
AddIntensity関数の実装と反対ですね。Max Floatを通すことでIntensityが0未満になることを防いでいます。
次にカスタムイベントを追加します。名前をPeek Fadeとします。
内容は以下の通りです。
46.PNG
SubIntensityでIntensityを0.1ずつ減算し0以下になった時にPace StateをRelaxに遷移させています。
これでPeek Fade状態の実装が終わりましたので呼び出し部分に追加していきます。
47.PNG
線で囲まれた部分が追加部分です。

実際に動作確認をしてIntensityの値を確認すると徐々に減っていくのがわかると思います。
Sustain Peekが終了した時点で敵の出現はさせていないので変化したのが実感しづらいですが、そんなものです。徐々に敵の出現数を減らすという処理を入れるとより実感出来るかもしれません。

#Relax状態の実装
Peek FadeでIntensityを減らす処理が出来ましたので次はIntensityが減りきった時の状態であるRelax状態を実装していきます。
Relax状態では指定秒数間は敵の出現処理を一切行なわず、プレイヤーの感情強度も上昇しません。
L4Dでも敵の出現がパタッと止んで静かになるタイミングがあります。その時レベル上にはAIDirectorが生成した敵とは別に事前にランダムで生成されているぼーっとしてたり道端に座り込んでいたりするゾンビが存在しているだけです。

ではRelax状態を実装していきます。
カスタムイベントを追加します。名前はRelaxとします。続いてFloat型の変数を追加します。名前はRelax Durationとします。
カスタムイベントRelaxは以下のようになります。
48.PNG
Sustain Peekとほぼ同じ実装内容ですね。異なる点は使用する変数がRelax Durationになっている所とSurviver Intensityを常に0にセットする所、Pace StateがBuild Upにセットされている所の3点です。

Relax状態の実装が終わった所で呼び出し部分にも処理を追加します。
49.PNG
線で囲まれている部分が追加された部分です。

動作確認をするとPeekが過ぎた後しばらくの間敵の出現が止み、再度敵がポツポツと出現するというのを繰り返すようになります。

以上でプレイヤーの感情強度による動的な敵出現頻度調整の実装が終了しました。

#まとめ
その1ではEQSを用いて敵の出現位置を求め、ここでは敵の出現頻度を動的に調整する処理を実装しました。
これで最初のほうで定義した要件の2つを満たすメタAIが実装できました。
実際のL4Dとは仕様がかなり単純になっているので想像していた結果とは異なる結果となってしまったかもしれません。
L4Dを遊んでみるとよりダイナミックでドラマチックな内容となっていますのでプレイ動画を見るなり実際に買ってAIDirectorに注目しながら遊んでみるとより面白いかもしれません。

お疲れ様でした。

10
10
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
10
10

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?