はじめに (モチベーション)
巨大なビルの破砕シミュレーションだって捌いてしまうHoudiniですが、「2D的」「モーション的」な使い方をしようとすると、シンプルなものでも意外ととっかかりに苦労します。そもそも作例があまり見つからず、あれこれ工夫しても前後にはみ出たり傾いてしまったりと、他のアプリで簡単にできるようなことすら中々実現できなかったりするのです。
せっかくHoudiniを触り始めたモーション仲間が、初手で躓いてしまってそのまま離脱…
というのはあまりに悲しいので、
- 落とす (自由落下)
- 発生させる (パーティクル)
- 動かす (アニメーション追従)
の3つの基本的なセットアップを、初手のテンプレとして用意してみました。
作例ファイル
2D_RBD.hipnc (Non-Commercial)
肝となる2D拘束の部分はいずれも共通で、コピペしてそのまま他のセットアップにも使える形にまとまっています。良ければご活用下さい。
アプローチ (血も涙も2Dモードもない)
私が(血眼になって)探した限りでは、HoudiniのRBDで2D形状を2Dのまま扱う方法は存在しません。基本的には全ての形状を押し出し、立体物としてシミュレーションしたものを強引に平面上に封じ込め、平面的にレンダリングします。
具体的には
- 位置をXY平面上に抑える
- 角度をZ軸回転のみにする
の2つを毎フレーム補正していきます。
方法1: コンストレイントを作る
使い方
いつものRBDセットアップに紫ブロックの部分をコピペして挟むだけ!簡単ですね。
もう少し詳しく
そもそもRBDの拘束(コンストレイント)とは
点2つで構成される線のプリミティブ群です。物体に相互関係を表現するために、何かと何かを一定のルールで繋ぐので、線なのですね。シミュレーションする対称のオブジェクト本体と合わせて、RBD Solverの2番目に入力してあげます。
自由度の拘束 (Degree of Freedom)
拘束には様々な種別やオプションがありますが、今回はHard Constraintの Degree of Freedom(自由度) を使います。
具体的にはZ軸を基準に、位置と回転の拘束を作っていきます。
- Z軸方向にはみ出さないようにする → 位置拘束
- Z軸以外で回転しないようにする → 回転拘束
これに則って考えると、2D平面に拘束するには、
-
condir
をどちらも Z軸({0,0,1}) に -
condof
を 位置拘束=1 (平面移動), 回転拘束=2 (1軸回転)
にすれば良さそうです。VEXで書くと
// 位置拘束
v@condir = {0, 0, 1};
i@condof = 1;
// 回転拘束
v@condir = {0, 0, 1};
i@condof = 2;
という感じですね。
2D拘束の設定
今回の拘束はオブジェクト単体に作用するので、対となるオブジェクトが存在しません。そのような場合、もう片方の拘束先には空欄を指定します。(ワールドの任意の点にピン留めしたいような場合なども同様ですね)
前述したcondif
, condir
はPrimitiveではなくPointへの設定が必要な点にも注意しましょう。
似たタイプのコンストレイントなので、最初は共通項目を同じ線に設定した上で、あとから分岐して位置/回転それぞれ異なる部分を設定し、最後にマージしています。
方法2: Post Solveを使う
使い方
RBD Solver内のPOSTSOLVEに、黄色いノードをコピペして繋ぐだけ!お手軽ですね。
もう少し詳しく
Post Solveは毎フレーム、シミュレーション後の結果に適用できるもので、ここでズレた位置と角度を強引に修正してしまおうという寸法です。
// --位置--
// Z位置をリセット
@P.z = 0;
// --回転--
// 現在の姿勢(クォータニオン)を回転行列へ変換
matrix3 m = qconvert(p@orient);
// 変換した行列で今向いている方向Nをつくる
vector N = {0,0,1} * m;
// Nを本来向くべき正面方向へ向かせるクォータニオンをつくる
vector4 qrot = dihedral(N, {0, 0, 1});
// qrotを使ってp@orientを正面に向かせる
p@orient = qmultiply(qrot, p@orient);
正直コンストレイントに比べてケアすべきことも少なくずっと楽なので、こちらで済む場合はその方が良さそうです… (期待せずちょっと調整を入れたら想像以上に安定してしまった)
自分もまだ検証し切れていないのと、拘束やDOFの話は応用範囲も広いので、一応上の方法も残すことにしました。
場合によっては両手法を組み合わせるのもアリだと思います。
各作例のポイント
細部は割愛しますが、特にポイントとなる点を書きます。
1. 自由落下
床(Ground Plane)をONにした程度の基本形で、2D拘束以外は特に何もしていません。
2. パーティクル
シミュレーションの途中で動的にオブジェクトを追加しています。RBD SolverのEmit RBDs
にチェックを入れましょう。
通常はオブジェクトを増やすと、よしなに関連コンストレイントもシミュレーションに追加されるのですが、ユニークな名前に置換されるためか、今回のような空欄指定のコンストレイントがあると上手く追加されません。
少しトリッキーですが最初はコンストレイントの両ポイントに同じ名前を指定しておき、生成直後にRBD Solverの中に作ったSOP Solver内で、片側のs@name
を空欄に書き換えています。
3. アニメーション追従
途中までアニメーションに従いながらも、衝突に合わせて解放されて自由に動きます。
最初のRBD Configureとその後のノードで、グリッド/円に対してそれぞれ独自のプロパティを設定しています。
ざっくり説明:
- 動かす方は
i@active=0
,i@animated=1
を指定。止まっている方は0/1を逆に。 - RBD Configure内の
Min Activation Impulse
を設定することで、一定の力以上にがかかったところでactive
に切り替わるよう指定。 - グリッドの方は最初に
Pin Constraint(Glue)
を設定し、次のRBD Constraint Properties
ノードで壊れる閾値を設定。
最後に
正直今回の作例程度のことであれば、他のソフトの方がよほどサクッと作れるかもしれません。(3つ目とかはやや難しかったりしますが)
でもひとたび基礎がHoudiniで作れたのなら、これを数万個にスケールすることも、サブステップ数をぶち上げて超高速な物体を処理することも、複雑な拘束を張り巡らせて破壊を何段階にも分けることも、想像しうる大抵のことは自由自在です。ツールの制約から開放されるべく、最後のツール習得と思って頑張りましょう。
参照 (謝辞)
今回の内容は、ほぼ以下両者の説明やノウハウを2D拘束という視点でゆるくラップ&再構成したものとなっています。(いつも大変お世話になっております...)
アニメーション追従のより詳しい記事など