5
0

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 1 year has passed since last update.

HoudiniAdvent Calendar 2022

Day 20

カスタム座標とカスタムフィールドとポイントからフィールド直読み

Posted at

この記事は上級者向け(呪術廻戦でいう一級に相当)で、扱っているのは以下の三つです。


カスタム座標とカスタムフィールドについて説明します。
カスタム座標とはxyzではない座標のことで当方による造語です。ノイズとかフィールドを作るときにたまにほしくなりますがあんまり使わないですしほしいときに調べて数時間つぶれたりします。その時間が無駄すぎるのでここに書きました。ほぼコピペで済む感じになってます。


カスタムフィールドはdensityとかvelとかのあらかじめ用意されているものではないフィールドのことです。漂う流体を作ってその中だけ減衰したいときとかに使えます。gas advect fieldを使うとカスタムフィールドもvelで動くようになります。もっと有用な使い方もありますがコンプラ的に言っちゃダメなやつな気がするので普通に調べられる範囲のものだけ書いてます。
あんまり詳しく書いてないので詳細は付属ファイルを見てください。


ボリュームを使わずポイントからフィールドを読む方法
あんまり詳しく書いてないので詳細は付属ファイルを見てください。

sample file

カスタム座標

1極座標

距離と角度を使った座標系で r θ(シータ)φ(ファイ)の三項で表されますが式はこんな感じになってます。
θ= acos(z/radius);
φ = atan2(y,x);

これをwrangleで書くとこんな感じです。
theta = acos(z/rad);
phi = atan2(y,x);

あとこの記事では使わないですが極座標から直交座標の変換式はこんな感じです。
x = radsin(θ) * cos(φ);
y = rad
sin(θ) * sin(φ);
z = rad*cos(θ) ;

極座標系を使ってノイズを作るとこういう感じになります。(色は別)
polar_coord.png

(右)

vector p = v@P;
float rad = length(p);
float theta = acos(p.y/rad);
float phi = atan2(p.z,p.x) ;
vector polar = set(rad , theta , phi) ;

float cd = anoise(polar);
@Cd *= cd ;

位置によってノイズが途切れるのですが元々のxyzを入れ替えると回避できる場合があります。例えばyとzを入れ替えると底面の跡切れを回避できます。あらゆる箇所で回避するのは無理です。
(真ん中)

vector p = v@P;
float rad = length(p);
float theta = acos( p.z /rad);
float phi = atan2( p.y ,p.x) ;
vector polar = set(rad , theta , phi) ;

float cd = anoise(polar);
@Cd *= cd ;

また、距離をいじると横長の極座標になります。
(左)

vector p = v@P + set(10,0,0) ;
float rad = length(v@P * set(1,2,2) );
float theta = acos(p.y/rad);
float phi = atan2(p.z,p.x) ;
vector polar = set(rad , theta , phi) ;
float cd = anoise(polar);
@Cd *= cd ;

角度は原点をベースに作られるので原点以外で同じようなノイズが欲しいときは極座標を作る元座標にそういう処理をしてください。

ところで極座標はvopでpolarというノードを使ってもつくることができますのでvop派の人はそっちを使った方が楽でしょう。当方は触るたびにvopに潜るとかめんどくさいのでwrangleで書きます。

2 カーブ座標

カーブ座標なんて名前の座標系はたぶん存在しないんですがカーブをもとに座標系を作ってるのでここではそう呼びます。応用すればサーフィス座標とかもつくれると思いますがここではやってません。

カーブの座標系を書く前に微妙に準備が必要です。

1 resampleSOPをつくって下の方にあるcurveuとtangentuにチェックを入れる
2 polyframeSOPをつくってのtangentvにチェックを入れる
3 wrangleをつくって以下のスクリプトをコピペ

v@tangentv = normalize(v@tangentv) ;
v@tangentu = normalize(v@tangentu);
v@up = normalize(cross(v@tangentv , v@tangentu));

ここでつくられたアトリビュートはtangentu , tangentv , up , curveuの四つです。最初の3つは座標系を作るのに必要です。

これを使って三つ座標系を作りました。
1 カーブ座標のxyとcurveuを使った座標

int pr;vector uv;
xyzdist(1,v@P , pr,uv);
float cu = primuv(1,"curveu",pr,uv);

vector tangentv = primuv(1,"tangentv",pr,uv);
vector up = primuv(1,"up",pr,uv);
vector tangentu = primuv(1,"tangentu",pr,uv);
vector p = primuv(1,"P",pr,uv);

matrix3 m3 = set( tangentv , up , tangentu );
vector pos = (v@P -p) * m3 ;

pos *= 2 ;
pos.z = cu * 13 + @Time * 1 ;
float cd = anoise(pos);
@Cd *= cd *31;
curve_coord1.png

2 curveuと半径と角度を使った座標

int pr;vector uv;
xyzdist(1,v@P , pr,uv);
vector p = primuv(1,"P" , pr,uv);
vector tangentv = primuv(1,"tangentv" , pr,uv);
float cu = primuv(1,"curveu" , pr,uv);

vector p2 = v@P -p ;
float rad = length(p2);

float theta = acos(dot(tangentv , -normalize(p2))) ;
theta = degrees(theta)/360 * 30;
cu = 9;
vector coord = set(cu , rad , theta ) ;
coord
=2;
coord.x += @Time ;
float cd = anoise ( coord );
@Cd += cd ;

curve_coord2.png

3 カーブの最近接ポイントのポジションとcurveuを使った座標

int pr;vector uv;
xyzdist(1,v@P , pr,uv);
vector p_= primuv(1,"P" , pr,uv);
float cu = primuv(1,"curveu" , pr,uv);

vector p = v@P -p_ ;
p*=3;
p.x = cu*12+@Time;

float cd = anoise(p );
@Cd += cd ;

curve_coord3.png

端的な解説
全部同じに見えますし止めだと実際同じなんですがtimeとかで動かすと違いが出てきます。
xuzdistとprimuvでカーブの最近接ポイントのアトリビュートを持ってきてます。
1は元の座標をカーブ座標に変換したうえでノイズをかけてます。
(座標変換は変換先のマトリクスを掛けるだけでできます。)
2はcurveuを時間で動かすとカーブ方向に流れ、距離を動かすと内側から広がるような動きをします。角度を動かすと回転しますがここでは180度までしか考慮されてないので思った感じには回りません。

カスタムフィールド

つくるのに苦戦した気がしたのですがやってみたら普通にvolumesource作れましたがせっかく調べたので別の方法も載せます。
方法は三つあります。

1 volumesourceにそのまま入れる

2 gasmatchfieldを使ってからそのフィールドをwrangleかなんかでいじる。gasmatchfieldだけだとフィールドはつくられないみたいです。

3 sopvectorfieldとかsopscalarfieldを使う
つくられはするんですが機能しないです。

ポイントからフィールドを直読みする方法

sopgeoを作りsoppathを指定しset initialにする。
そのあとgasparticlefieldを使ってそれぞれのフィールドを発生させる。ただしこの方法だとカスタムフィールドは作れない様子。
それとこの方法だと1フレ目が元ジオメトリのサイズを考慮しないのでこれをさけるためにgasresizefluiddynamicを使います。max bounds のチェックを外し、boundsタブのtrack objectをdopにしてからdop pathをGeometryと打ちます。

pointToField.png

5
0
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
5
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?