#基礎編
IF文とLerpとは
◆IF文とは
…もし●●なら■■する。(そうでないなら▲▲する)
◆Lerpとは
…入力された値が0なら■■、1なら▲▲(0~1の間はブレンド)
→Lerp の入力された値を制御することでIF文として使える!
(入力された値●●が0なら■■する。●●が1なら▲▲する)
NiagaraでIF文を作る
ここでは、パーティクルの位置をノーマライズ(0~1に変換)し Lerp に入力
「0ならパーティクルのサイズを2、1ならパーティクルのサイズを20にする」
という処理を行っています
この処理を Niagara 上で表示したときには次の画像のようになります
上から順番に説明していきます
◆Vector 2DFrom Float
…パーティクルのサイズはX軸Y軸の2軸(Vector 2D)で決まっているので、
両方に同じ数値を入れます という処理
◆Lerp Float
…入力された値●●が0なら■■する。●●が1なら▲▲する
Aには0の時の値(■■)、Bには1の時の値(▲▲)を入れる
Alpha には入力する値●●を入れる
◆Normalize Float
…設定された Max , Min の数字に応じて Value の値を正規化する
この例の場合は Max が200、Min が0なので Value が200の時に1 (400の時は2) になる
◆NDIS Abs
NDIS Abs の中身はこのようになっています
絶対値を取るだけの Niagara Dynamic Input Script です
◆Subtract Float
…引き算です
◆Make Float From Vector
…Vector に入力された値から Channel でX,Y,Zのいずれかを選びその数値を使います
Vector に(10,15,20)という値が入っている時に Channel で Z を選ぶと20になります
ここでは Particle Position から原点(Simulation Position)の値を引いて距離を取っています
まとめると
原点からパーティクルの位置を取ってから絶対値を取って正規化
それを Lerp の条件として使用し、距離に応じてサイズを変更
ここで Niagara に使い慣れている人は
「いちいち Lerp 使わずに距離だけをとって足し算掛け算したらいいのでは?」
と思うかもしれません
この例においてはその通りです
なので応用編では Lerp の IF 文を使わないとできない例を出して解説します
#応用編
##用意
鳥が**「一定以上の速度で上方向に向かって飛んでいる場合にのみ羽ばたかせたい」**
応用編ではこの例を用いて説明していきます
どこからどう見ても鳥ですね UV も適当に開いておきます
羽ばたかせる用のテクスチャも作っておきましょう
World Position Offset で羽ばたかせるためのマテリアルを用意します
マテリアル内ではベクトル計算についての知識が必要ですが、
ここでは説明せずに Niagara 内で計算する際に説明します
こちらが今回のマテリアルになります
最低限の構成なので簡単に作れると思います
##羽ばたかせる
###Niagaraの設定
ここから Niagara の作業に入っていきます
モデルにマテリアルを適用しスポーンさせてランダム方向に飛ばします
Facing Mode はとりあえず Velocity にしておいてください
各パラメーターの説明を行います
◆WingFlap
...0~1で往復させることで羽ばたくようになっています
◆AgreeX,Y,Z
...今向いてる方向に対してそれぞれの方向にどれだけ羽ばたくかというパラメーターです
画像では0となっていますが、1を入れておいてください
◆WPOStrength
...羽ばたく大きさです
モデルの大きさに合わせて適当な数値を入れておいてください
###羽ばたきを考える
ではまずWingFlapから調整をしていきます
羽ばたくときは0~1を往復してほしいのでまずSinを使います
羽ばたかないときは羽を水平に保ってほしいので、WingFlap に0.5が入っていてほしい・・・
そうですね、Lerp ですね
今回は羽ばたく条件として
・上向きの速度が40以上の時に羽ばたく
とします
###Velocityに応じて羽ばたかせる
Niagara で実際に組むとこのようになります
軽く説明すると、Z軸のベクトルの最小を39.99...・最大を40として正規化
それを0~1で Clamp することで39.99...未満は0・40以上は1になります。
(Max と Min に同じ数字を入れると、ちょうどその数字が出た時に NaN になってしまうので少しずらします)
それを Alpha に入れて Lerp の AB を設定することで Z 方向に40以上で飛んでいる場合にのみ羽ばたくようにできます
###成分毎に分解して羽ばたかせる
次にAgreeXYZを計算していきます
さっきより かなり 面倒ですが、ここを理解できれば Cos・内積はある程度扱えるようになります
Cos というのは円を描いた時の中心からの位置をX軸Y軸に分けたものの片側です
内積は2つのベクトルからなる角の Cos を計算することができます
Sin・Cos については下記のサイトの gif を見ていただければ理解できると思います
https://qiita.com/rakusan/items/2f60f2ae6af208dda40f
それではまずは X から考えていきます
羽が X 方向に移動してほしいときというのは、鳥が Z 軸方向を向いている時かつ鳥が X 軸方向も向いている時になります
(一番左は縦方向を向いていないので、横軸方向には羽ばたかない)
Z軸方向に Velocity がかかっている・X軸方向に Velocity がかかっている
という2つの条件があっても Lerp で対応できます
A には0を入れておき、B で X 軸方向の条件を、Alpha で Z 軸方向の条件を満たしていきます
まず B については Niagara 内ではこのようになります
順番に説明していきます
◆Dot Product
...内積の計算を行ってくれます
A と B にベクトルを入れます
ここでは A に Velocity、B には X 軸を反転させたものを入れています
X 軸を反転させる理由ですが、後程説明します
◆Divid VectorとCector from FloatとVector Length
Dot にそのまま Velocity を入れると1以上の数値が出てしまいます
そこで
(a・b)/(|a||b|)=Cosθ
このような式に変換してベクトルの大きさを正規化することで Cosθ(-1~1)を出すことができます
※式の意味は理解しなくても大丈夫です 2ベクトルから-1~1の値を出すことができるとだけ覚えておいてください
向いている方向によって何かをしたいって時はよく使うので NDIS 化しておいてもいいかもしれません
...ちなみに今回の場合は(a・b)/(|a||b|)の式において、片方のベクトルは単位ベクトル(大きさが1)でX成分にのみ数値が入っているため、
このように書き換えることもできます
次に Alpha はこのようになります
上の B で行った内積の簡略化した式を使っています
今回は上方向なので Channel が Z になっています
(Alphaには-1~1の値が入っていますが、動きには問題ないので無視しています)
これで上方向を向いているかつ X 軸方向を向いている時に羽ばたくという設定ができました
AgreeY も同じように設定していきましょう
Channel が X だったところを Y に変更しただけになります
そして AgreeZ です
これはとても簡単です
Z 軸を向いている時に羽ばたかなければいいだけです
Z 軸方向を向いていれば羽ばたくという処理を One Minus を通すことで
Z 軸を向いていれば羽ばたかないという処理に変更してるだけです
AgreeXY で-1をかけていた理由
これは Sin・Cos の性質で
Sin は0~π(180度)の時に正・π~2πの時に負
Cos は-π/2~π/2の時に正・π/2~3π/2の時に負
となります
そのため-1をかけておかないと一定方向に動いた際に(鳥にとっての)縦ではなく横に羽ばたいてしまいます
試してみてください
##回転を考える
まだおかしいところがありますね
移動方向に対して回転している場合
このようにオフセット方向が対応しきれていません
解決方法は2つあります
1.Velocity に合わせてメッシュを回転させて正しい方向を向かせる
2.回転方向も考慮して羽ばたく方向を変える
簡単なのは1の方...というよりも2はかなり面倒なので1を採用します
###クォータニオンを変換する
Roll Pitch Yaw でいうと Roll に回転が入っていなければ正しく羽ばたいてくれそうです
そこでMesh Orientationというモジュールを使います
このモジュールはクォータニオンという4つの数字で回転を制御しています
クォータニオンが何かを簡単に説明すると、
「回転する軸を設定・回転量を設定」って感じです
クォータニオンのまま制御するよりも Roll Pitch Yaw に変換したほうが制御が簡単なので、変換する NDIS を作ります
これだけです
X が Roll、YがPitch、ZがYaw として扱えるようになります
(このモデルはX軸方向を前として作成しているため)
Facing Mode が Velocity のまま回転させると中指を立てたくなる挙動をするので、
Default に戻しておきます
###回転させる
今回は Roll が回転しなければ綺麗にいきそうなので、Xには0を入れておきます
初めにPitchを設定していきます
必要そうなものだけ説明していきます
◆NDIS Vector to Quat
...先ほど作った NDIS で Vector4 を Vector に変換します
◆Break Vector
...Roll Pitch Yaw でそれぞれ計算したいので、Vector を3つの Float に分解しています
◆Normlized float
...Cos は-1~1の値を取るので、-1を0、1を1として扱うように正規化しています
後は内積を簡略化させたものです
Lerp の AB の値が-0.25~0.25なのは Cos が0の時に正面を向くからです
次にYawを設定していきます
Yaw の設定は X 軸との内積だけではなく、Y 軸方向に対して正負の判断が必要になります
そのため、数値を出すための条件が増えます
Y 方向の Velocity が正で X 方向の Velocity が正であれば0
Y 方向の Velocity が負で X 方向の Velocity が正であれば0
Y 方向の Velocity が正で X 方向の Velocity が負であれば0.5
Y 方向の Velocity が負で X 方向の Velocity が負であれば-0.5
となります
(1で一周するため0.5と-0.5は同じ方向を向きますが、回転方向が違います)
Niagara 上ではこのようになります
ここで AB の値が ±0.5~0 ではなく ±0.25~0 である理由ですが、
AB の Alpha の中の計算は Cos を求めているので -1~1 の値を取ります
なので Alpha が1の時は0、0の時は±0.25、-1の時に±0.5となるようにしています
Pitch の時と同じように-1~1で Normalize すれば±0.5と書くことができます が面倒なのでやりませんでした
Alpha については、Clamp&Normalize をすることで
0以上で1・-0.0001以下で0となるようにして疑似的に正負を取っています
###蛇足
Roll に回転が入るのは基本的に鳥がカーブする時です
内側に背が来るように回転します
上から見た時に時計周りの場合に0~0.25、反時計周りの場合に0~-0.25の数値が入れば
今以上にきれいに飛んでいるように見えます
しかし Roll に回転が入っている場合、その回転量を鳥の羽ばたく向きに対応させなければなりません
先ほど「かなり面倒」と言った計算方法ですね
簡単に説明すると...
・1F 前の Velocty と今の Velocity から加速度を取得できるNDISを作成
・その加速度から回転方向を取得
・向きによってLerpで回転させる → Float で参照できるようにしておく
・回転量から鳥の背中の方向を取得し Z 方向との Cos を出す
・Cos を考慮して AgreeXYZ にさらに Lerp
といった具合になります
現状(4.24.1)加速度を取得できないと思います
なのでどうしても Roll 回転させたい場合は、鳥の動きを Sin Cos なので制御して、
その周期もしくは回転方向に合わせて Roll を回転させてやる、というのが現実的です
Epic さん、Niagara 内で Acceleration を取得できるようにしてください
Particles.Acceleration という Vector を用意してください
#最後に
斬撃や爆発や魔法といった普通のエフェクトではこんな面倒な計算をする必要はないと思います
しかし「速度や大きさや寿命ごとに何かをしたい」といった少し特殊な表現を行う場合には
今回の Lerp を使うことで表現できる場合があるかもしれません
また Sin Cos は初めは取っつきにくいかもしれませんが、
実際には汎用性の高い素晴らしい関数です
ベクトル計算も理解するまでは「???」かもしれませんが、
方向・速度・場所などの Vector を用いて表現する場合には必須になってきます
ここまでついてこられた方は、Lerp の使い方・内積についてはある程度理解できたのではないかと思います
この記事が少しでも皆様のお役に立てたのであれば幸いです
この記事は以上です
お疲れ様でした
それにしても、こんな難しい例じゃなくて
もうちょっと簡単なものはなかったんでしょうかね(他人事)
書き始めた時は数学初心者向けに書こうと思ってたんですけど、
数学初心者さんはここまで残ってるのでしょうか・・・
残ってなかったらごめんなさい
残ってても理解できなかった人がいらっしゃいましたらごめんなさい
Twitter などでいつでも質問してくださって構いません
また間違いなどありましたらそちらもお願い致します