2018年12月10日月曜日

失われたTransformを取り戻す。

こちらはHoudini Advent Calender 2018 10日目の記事です。
https://qiita.com/advent-calendar/2018/houdini

ジオメトリの形状からTransform(位置、回転、スケール)を取得(作成)し、別ジオメトリを追従させる方法です。
この間同僚にやり方を聞かれ、これまで何人かに同じ事聞かれたので記事にする事にしました。
本当はこのような場面(Transformデータが無い)は出くわすのを避けるべきですが、
データが連番ジオメトリしか残っていないっ!なんて場合に使えます。
(基本AlembicやFBXでTransformデータが残っている場合はパラメーターをchで繋げるのみです。)

以下は結果です。
ドーナツ(Torus)ジオメトリのアニメーションに、ぬいぐるみ(rubbertoy)が付いて行っています。



Transformをつくる全体を簡単に説明すると、

ポジションは全体の中心座標、回転とスケールは任意のポリゴン一枚からN、up、 pscaleを計算し、copy to points sopで流し込むという流れです。

(このサンプルではTorusを適当にアニメーションしたものをObject引っ張り、Transformのデータは無いと仮定して進めています。)
まずはグリーンの部分、
位置(中心座標)の取得ですが、ジオメトリをpack化する事で1point のみになり自動でpointの位置も中心にくるので楽で良いです。
add sopで1pointつくり でHscriptのcentroidを使用しても良いと思います。

次に回転、スケールを計算するオレンジの部分の説明です。

Torusから任意のポリゴンを一枚選び残します。
(今回は解説の為見やすいようビューポートで一枚選択してキーボードのdeleteキーを押し作成されたblast sopのオプションをDelete Non Selectedにしていますが、  blastのGroupに0とすればインプットのジオメトリを変えてもそのままの仕組みが流用しやすくなります。)


次にNormal SopでPrimitiveにNを作ります。

ここでpointタイプに作成してしまうと、次につくるupと直交した物にならないので要注意です。
直交したNとupがあれば最後のcopy to points sopで回転として上手く作用します。

次にUpとpscaleの作成です。
vector dir = point(0,'P',0)-point(0,'P',1); //ポイント番号0のポジションから番号1番のポジションを引き算しベクトルを作ります。
v@up = normalize(dir);  //ノーマライズして長さが1のupというベクトルにします。
@pscale = length(dir)*3.5;  //エッジの長さの変化をスケールとして扱うので、ノーマライズ前のベクトルdirのlengthをとりpscaleとします。
この手法の欠点は、ぬいぐるみジオメトリの初期状態をキープは出来ないので大きさや
向きを調整してあげる必要があります。pscaleに3.5をかけているのはその為です。

これでNとUp、pscaleができましたがNがprimitiveアトリビュートのままでは次のattribute copyでpack化した中心のpointに移せないので、attribute promoteでpointアトリビュートに変換します。

このNとUp、pscaleを左側のpack化したpointにattribute copyで渡します。

このポイントにcopy to points sopでぬいぐるみジオメトリを配置すると
ドーナツ(Torus)と同じように移動、回転、スケールが反映されます。

今回の様にポリゴンが斜めを向いている場合は、あらぬ方向を向いてしまうので、ぬいぐるみの初期の向きをTransform sopで調整しています。

今回はオブジェクトの中心にN , up , pscaleを持たせましたが任意のポイントもしくは、プリミティブをpackにすれば、ポイントコンストレインやポリゴンコンストレインにも置き換えられます。

以上です。

Houdiniアドベントカレンダーは2016年から続いています。Houdiniのほとんどのテクニックは古くならずいつまでも使えるものが多いので是非読んでみて下さい。




2018年12月3日月曜日

分布の種類をコントロールする。

こちらはHoudini Advent Calender 2018 3日目の記事です。
https://qiita.com/advent-calendar/2018/houdini


スキャッターされる複数ジオメトリを配置具合に重みをつける方法を紹介します。

ちなみにH17のHeight field scatterでは配置するオブジェクトにweightアトリビュートをを持たせる事で似たようなことができますが、
こちらの方法はscatterポイントのアトリビュートをコントロールする事でより柔軟な表現が可能になります。


全体のネットワークです。3種類のジオメトリをpointに配置しています。
半径10のcircleにscatterし、そのpointにweight アトリビュートを持たせていきます。
今回は原点からの距離をweightにしています。weightのコントロールをrampで行いやすいよう、0-1の間に収めるよう10で割っておきます。
確認の為、Cdにweightコピーしてビューポートで見えるようにしています。
(今回は半径10のcircleに配置したので10で割るという簡単な方法をとりましたが、最大値、最小値を自動で求めて0-1にfitしたい場合はattribute promoteで出来ます。)

次にweightのコントロールし、そこからidアトリビュートを作ります。

float weight = chramp('weight',@weight); //下のランプで重みをコントロールしています。
@Cd = weight; // ビューポートで確認。
i@id = rint(fit01(weight,0,npoints(1)-1)); // weightからid アトリビュートを作ります。
0-1 のweightアトリビュートfitでを0-N のidアトリビュートを作ります。
Nは今回は3種類のジオメトリを配置しているので、3種類をそれぞれpacked化しmergeしたものを2番目のインプットにさしnpointsを使用する事でNを求めています。
今回は3つのジオメトリがあるのでそれぞれのptnum(ポイントナンバーは)は0, 1 , 2 となり最大値は2。
npointsは何個ポイントあるかが返ってくるので3となるので1を引いています。
最後にrintで四捨五入し整数値にしています。
上記の様にカーブで小さい値が多くなる様に制御した状態。これでidが小さいほうのジオメトリが多く配置されます。

最後にcopy sopでstampを使用可能にし、blast でポイントナンバーがidにマッチしないジオメトリを消すことで
スキャターされたpointが持つidにマッチしたジオメトリが配置されます。

以上です。

Houdiniアドベントカレンダーは2016年から続いています。
Houdiniのほとんどのテクニックは古くならず、いつまでも使えるものが多いので是非読んでみて下さい。