2012年12月5日水曜日

MAYA トリッキーなShadingSwitch の使い方




図のようなシチュエーションから、pSphereがpPlane1に落とす影は現状のままクッキリとしてていいが、pPlane1がpPlane2に落とす影だけは柔らかく調整したい!といったケース。
いや、めったにない特殊なケースだという事は分かってますが・・・、同僚から相談されちゃいましたので考えてみました。

ちなみにライトを2種類作ってライトリンクでうんぬん・・とか、パスを分けてうんぬん・・・という以外で!という条件つき。めんどくさがりめっ。


まず、おもむろに出してみたのはShadingSwitchノード



このノードは、複数のオブジェクトで同じマテリアルを共用し、かつそれぞれに違ったテクスチャを割り当てたり出来る優れものです。

このノードにpPlaneのシェイプをつないでアレコレしてから、outputをディレクショナルライトのLight Angleへとつないでみます。

結果からいうと





出来た!
逆ももちろん この通り!





ではコネクト方法。

アトリビュート内容は下図の通りす。
こいつの In Shape に pPlane 2枚を追加したいわけですが、pPlaneを選んでAddSurfaceを押してみてもいっこうに追加されません。
このボタンを使う方法もあるのですが時間がかかるので、今回はコネクションエディター or ノードエディターで直接つないじゃいます。


pPlane(シェイプ) の instObjGroup[0] → singleShadingSwitch  input[#].inShape

プレーン2枚ともつなぎます。
でたでた。



つぎに、ディレクショナルライトのLight Angleに代入したい数字をそれぞれのIn Single に入れたいわけですが、これも直接入力できません。。
かわりに単色のrampを2つ作ってそれぞれをコネクションエディターでつなぎます。

ramp#.outColorR → singleShadingSwitch.input[#].inSingle

あとはrampの色でボケ具合を調整できます。

下の画像では分かりませんが、ramp1 = 0  ranmp2 = 30 の色を持たせています。
ですので pPlane2 にのみ LightAngle 30 のやわらかい影が落ちます。




ちなみにデプスマップシャドーのフィルターにコネクトしても同じような結果が得られますが、mental ray では動作しませんでした。。。(レイトレシャドーは mental ray でも動作確認済みデス)

2012年8月14日火曜日

パーティクルインスタンサー オブジェクトの法線を拾う

「パーティクルがサーフェイスに沿う」表現はGOALで実現できますが、そのパーティクルをインスタンスで任意のオブジェクトにして、「表面を這う」なんていう表現は・・・


needParent UV を ON の状態でパーティクルエミッタ(emit from surface)を作成。
パーティクルとサーフェイスでgoalを設定。

perParticleAttributeにparentU parentVを追加し、パーティクルエクスプレッションに下記の通り書き込みます。

goalU = parentU
goalV = parentV

これで、パーティクルはサーフェイス上での個々が発生した場所に留まります。

ここでパーティクルにオブジェクトをインスタンスしてみます。
インスタンスの回転は0の状態ですが、それぞれがサーフェイスのノーマルに従って回転した状態で配置されるようにしてみます。

パーティクルのアトリビュートエディターの「Goal Weights and Objects」セクッションにて、「Create Goal World Normal 0 PP」ボタンを押すと法線情報を持ったアトリビュートが出来ます。
注 このボタンはgoalの設定をしていないと表示されません。
instancer の Aim Direction を出来たアトリビュートに設定。
 
とりあえずノーマルに沿って配置されたはず!

ですが、サーフェイスのノーマルとオリジナルオブジェクトの+X軸が揃った状態になっているので、思ってたのと違う!となる場合が多いでしょう。
そんな時は、追加でUserVectorPPなどのperParticleAttributeを作り、オフセット処理をしましょう。ついでにランダムな感じを足したい時はノイズなども同時に計算しちゃいましょう。
注 その計算はruntime after dynamicsでしないと、おかしくなる時があるそうです

インスタンスされるオブジェクトの+X方向がサーデイスのノーマル方向に向きます。

2012年6月28日木曜日

ランダム ストリーム

再現性のある乱数処理の方法

まずは例、

//seed宣言
seed 2;
rand 0 5;
// Result: 2.064494 //
rand 0 5;
// Result: 1.76712 //
rand 0 5;
// Result: 2.612228 //
rand 0 5;
// Result: 0.615584 // 
//seed宣言し直し
seed 2;
rand 0 5;
// Result: 2.064494 //
rand 0 5;
// Result: 1.76712 //
rand 0 5;
// Result: 2.612228 //
rand 0 5;
// Result: 0.615584 // 


rand関数はデフォルトではメインストリームが使われている為、seed値を設定しなおせば、ストリーム配列の初めからrand値を出しなおす。よって、rand結果は再現性のあるものである。
また配列はrand関数が実行される度に1つ進むと解釈してよさそう。。
では、たとえばオブジェクト「a」、オブジェクト「b」に毎フレーム同じrand結果を出させたい場合。
このままだと配列はrand 実行毎に進んでいくので、


//seed宣言
seed 2;

//frame1
setAttr a.ty `rand 0 5`;
// Result: 2.064494 //
setAttr b.ty `rand 0 5`;
// Result: 1.76712 // 

 //frame2
setAttr a.ty `rand 0 5`;
// Result: 2.612228 //
setAttr b.ty `rand 0 5`;
// Result: 0.615584 //


となり、1フレーム目では「a」に配列1、「b」に配列2、が返され同じ結果にはならない。
そこで、任意のstringにてオブジェクト個々に独自のランドストリームを持たせる事で、結果を同調させます。


//seed宣言
seed st_a 2;
seed st_b 2;

//frame1
setAttr a.ty `rand st_a 0 5`;
// Result: 2.064494 //
setAttr b.ty `rand st_b 0 5`;
// Result: 2.064494 // 

//frame2
setAttr a.ty `rand st_a 0 5`;
// Result: 1.76712 //
setAttr b.ty `rand st_b 0 5`;
// Result: 1.76712 // 

//frame3
setAttr a.ty `rand st_a 0 5`;
// Result: 2.612228 //
setAttr b.ty `rand st_b 0 5`;
// Result: 2.612228 // 

//frame4 
setAttr a.ty `rand st_a 0 5`;
// Result: 0.615584 //
setAttr b.ty `rand st_b 0 5`;
// Result: 0.615584 // 
 

毎フレーム同じ結果になりました。
ストリームはrand実行毎に進むので、「b」が1コマ遅れでついていくなんて事も出来ます。


//seed宣言
seed st_a 2;
seed st_b 2;

//frame1
setAttr a.ty `rand st_a 0 5`;

// Result: 2.064494 //
//frame2
setAttr a.ty `rand st_a 0 5`;
// Result: 1.76712 //
setAttr b.ty `rand st_b 0 5`;
// Result: 2.064494 // 

//frame3
setAttr a.ty `rand st_a 0 5`;
// Result: 2.612228 //
setAttr b.ty `rand st_b 0 5`;
// Result: 1.76712 // 

//frame4 

setAttr a.ty `rand st_a 0 5`;
// Result: 0.615584 //
setAttr b.ty `rand st_b 0 5`;
// Result: 2.612228 //


エクスプレッションの場合、seedの宣言は1フレームに一度だけ、
パーパーティクルエクスプレッションの場合はcreationにて行なう。


以下、perParticleでの例 (書籍 テクニカルアーティストスタートキット Chapter6 .p266)
同じ数のパーティクル数ならば、完全に同調した動きをします。

particleShape1
creation :
seed pShape1 2;
runtime before dynamics :
particleShape1.velocity = `rand pShape1 <<-1,0,-1>><<1,0,1>>`;

particleShape2
creation :
seed pShape2 2;
runtime before dynamics :
particleShape2.velocity = `rand pShape2 <<-1,0,-1>><<1,0,1>>`;

関連コマンド

randstate:ストリームの状態を調べる。

例)
randstate st_a;
// Result: <<63847, 14317, 25624>> //

2012年6月26日火曜日

luminance depth

レンダーレイヤーを使用したluminance depth の出し方。

デプス出力したいオブジェクトを新規レンダーレイヤーに登録する。
レイヤーを右クリックでアトリビュートを出す。
アトリビュートエディタのプリセットからluminance depthを選択。

出来たset rengeユーティリティでカメラ深度を設定し、十分なビット数でレンダリング!

2012年6月19日火曜日

イメージプレーンの誤レンダリング防止

レンダリングが遅いと思ったら、いらないイメージプレーンを消し忘れてた、という凡ミスをなくすためのMEL

string $list[] = `lsType imagePlane`;
for ($node in $list)
    if ($node != "<done>"){
    setAttr ($node+".displayMode") 0;
}

で全てのイメージプレーンをNONEに出来ます。
これをpre render mel に入れればいいのだけど、長いのでプロシージャにして呼び出します。

global proc WB_IPhyde()
{
    string $list[] = `lsType imagePlane`;
    for ($node in $list)
        if ($node != "<done>"){
            setAttr ($node+".displayMode") 0;
        }
}

これをWB_IPhyde.melとして保存すれば、レンダーセッティングのpre render mel にWB_IPhydeと入れるだけで、レンダリング前に全てのイメージプレーンをNONEにしてくれます。

ただ、このままではMAYA GUIからのRender Current Frameの時も同様にNONEに設定されてしまいます。それでは困る場合の方が多いので、BatchRenderの時のみpre render melが適用されるように「about -b」を使ってif文を足します。

global proc WB_IPhyde()
{
    if(`about -b`==1){
        string $list[] = `lsType imagePlane`;
        for ($node in $list)
            if ($node != "<done>"){
                setAttr ($node+".displayMode") 0;
            }
    }
}

for in

変数の配列の数だけループできる便利なfor文!

[例]選択したオブジェクトのx値を0に

$sel = `ls -sl`;
for($node in $sel){
    setAttr ($node+".tx") 0;
}

2012年1月26日木曜日

AE BOXの回転移動


四角形のレイヤーをコロコロと転がすアニメーションです。

フッテージの高さ/幅ピクセルサイズを計算に使いますので、素材読み込みは「コンポジション-レイヤーサイズを維持 」で。
もし余白がある場合は四角形と同じコンポジションサイズにプリコンポします。
余白があると回転する時に地面から浮いてしまいます。



素材をコンポジションに入れたら、 「アンカーポイント」と「位置」にそれぞれエクスプレッションを追加します。

アンカ-ポイント用
count = Math.floor(rotation / 90);
    num = count + 1000;
    val = ( num % 4);
    if (val == 0) {
      [width, height]
    }
    else if (val == 1) {
      [width, 0]
    }
    else if (val == 2) {
      [0, 0]
    }
    else
      [0, height]
 

位置用
val = Math.floor(rotation / 90);
widthP=width*(scale[0]/100);
heightP=height*(scale[1]/100);
gap=Math.abs(widthP-heightP);
val2=Math.ceil(val/2);
valX=val * widthP;
valY=val2 *gap;
if(widthP>heightP){

    x = value[0]+ valX-valY;
}
else{

    x = value[0]+ valX+valY;
}
y = value[1];
[x, y]

レイヤーの回転に数値を入れてアニメーションさせてみると、回転に応じて右へ左へ転がっていきます。
あとはレイヤーを好きな位置に動かして完成!


2012年1月24日火曜日

AE ボールの自動回転

転がるボール、車のタイヤなどのアニメーションを簡単に作るエクスプレッションです。

アフターエフェクトで用意されている円周率関数(PI)を使って、移動をアニメーションさせるだけで地面を転がっているように物理的に正しく回転させます。


まず、フォトショップでネタを作ります。


AEに 「コンポジション-レイヤーサイズを維持 」で読み込みます。
レイヤーサイズを維持しているので、レイヤーサイズ=ボールの直径になっています。



レイヤーに移動のキーフレームアニメーションを作り、
回転にエクスプレッションを追加して下記の文をコピペします。

d=width*(scale[0]/100);
pi=Math.PI;
circ=d*pi;
rot=360/circ;
movex=position[0];
rot*movex;

プレビューするとボールが自動で回転します。



----エクスプレッションの補足---

レイヤーサイズ=球の直径 になっていない素材を使いたい場合
1行目の「width」を直径(ピクセル)の数値に書き換えて下さい。

AE ウィグル

画ブレなど、プロパティにランダムな値を入れたい場合、ウィグラーが便利ですが、キーフレームを作るよりもエクスプレッションで行なうと作業がかなりやり易くなります。
試しにレイヤーの位置にエクスプレッションを追加して、

wiggle (5,20);

と打ってみると、レイヤーがランダムに振動しはじめると思います。
( )内の「,」で区切られている数字のうち、初めの「5」は周波数で、大きくなると振動のスピードが激しくなります。2つ目の数字「20」は振動幅で、この例だと最大20ピクセルの移動範囲で振動します。
2つの数字を書き換えて再生すれば、すぐに変更が反映されています。
もちろんこの文はそのまま回転やエフェクトの値なんかにも使えます。

もう1ステップ、
このままだと、いちいちエクスプレッション文を開いて調整しないといけないので少し面倒です。
エフェクトのスライダーとエクスプレッション内の各数字を関連付ける事でさらに調整がスムーズになります。
なにやら面倒くさそうですが、アニメーションプリセットを参照で簡単に出来ますので、 よければ使ってみて下さい。

ウィグル位置.ffx
ウィグル回転.ffx

--オプション説明--
●(ON/OFF) オンオフ切り替え。キーフレームを打って途中から振動を止めるor始める事も可
●freq:周波数 上の説明の同じ
●amp:振幅 上の説明と同じ
●octaves:オクターブ ノイズ
●ampMult:振幅 ノイズの振幅


また、振動をつけた上で、キーフレームアニメーションも付けれますので、細かくブレながら飛ぶハエの動きなんかは簡単に出来ますね。



本文
a=effect("wiggle (amp:振幅)")("スライダー");
f=effect("wiggle (freq:周波数)")("スライダー");
o=effect("wiggle (octaves:オクターブ)")("スライダー");
on=effect("wiggle (ON/OFF)")("チェックボックス");
mult=effect("wiggle (ampMult:振幅)")("スライダー");
num=effect("wiggle (ON/OFF)")("チェックボックス").numKeys;

if(num==0){
    if(on!=0){
        wiggle(f,a,o,mult,time);

    }
    else{
        value;
    }
}
else{

    t = 0; //ループカウンター、初期化   

    timeGo=0;
    while (t <= time){   
        nKey = effect("wiggle (ON/OFF)")("チェックボックス").nearestKey(t);
        (t >= nKey.time ) ? P1 = nKey.index : P1=nKey.index-1;
        onoff = effect("wiggle (ON/OFF)")("チェックボックス").key(P1).value;

            if(onoff==1){
                timeGo+=1/30;
            }

        t+=1/30
    }

wiggle(f,a,o,mult,timeGo);
}
 

2012年1月16日月曜日

AE 連番文字をパカパカ


文字がランダムにパカパカ切り替わり→決まり文字へ、というよくあるパターン。
タイムリマップとエクスプレッションを使ってやってみます。

例えば、アルファベットの場合、A~Zの連番26枚を用意し読み込みます。



コンポジションに入れて、タイムリマップを適用します。
タイムリマップにエクスプレッションを適用し、↓をコピペします。

 fps=30;
sn=source.name;
du=footage(sn).duration*fps;

start=1; //ランダム範囲の開始フレーム
end=du; //ランダム範囲の最終フレーム
koma=1; //何コマ落ちでパカパカするか

if(marker.numKeys){
    marker1=marker.key(1).time;
    kimari=marker.key(1).comment;
}
else{
    marker1=100;
    kimari=1;
}

start=start-1;
hold=koma/fps;
seed = Math.floor(time/hold);
seedRandom(seed,true);
run=random(start,end)/fps;
fix=(kimari-1)/fps;

time<marker1? run:fix


再生してみると、1コマおきにランダム表示されていると思います。
次に適当な場所にレイヤーマーカーを1つ作って、決めで使いたいフレーム番号をコメントに入れます。例えばGを使いたい時は7。
も一度再生すると、コメントの位置にくるとGでストップします。

また、
本文の中、下の3行はオプションとして書き換えることができます。

start=1; //ランダム範囲の開始フレーム
end=du; //ランダム範囲の最終フレーム
koma=1; //何コマ落ちでパカパカするか

読み込んだ連番のすべてのフレームを使ってランダム表示したい時はそのままで良いのですが、
特定の範囲、例えばC~Fだけを使いたい!という場合は開始の「1」を「3」へ、最終の「du」を「6」へ変更してみてください。
またkomaはコマ落ち数で、5に変更すると5枚おきに文字が切り替わります。現状はフルフレームの「1」ですね。

もろもろ設定が終われば、レイヤーをコピーして、レイヤーの位置、マーカーを編集すればOK。

ちなみにリンクのムービー「PAKAPAKA」のタイムラインはこんな感じ↓






タイミングの編集くらいなら、レイヤー数が増えても比較的編集しやすいと思います。



 --追記--
今回は、外部で文字ファイルを用意する前提での方法ですが、
文字の扱い的にフォントにこだわる必要がなく、決まり文字のタイミングもおおざっぱで良い場合などは、AE内のテキストレイヤーでアニメーターオプションを使った方法がてっとりばやいですね。
一応、 AEサンプルです。

2012年1月13日金曜日

AE キーフレームのループあれこれ

キーフレームを1サイクル作ってそのアニメーションをループさせたい時、キーフレームをコピーするよりも、エクスプレッションの記述で簡単にできちゃいます。

loopIn(type = "cycle", numKeyframes = 0)
loopOut(type = "cycle", numKeyframes = 0)

上の文は、エクスプレッションを追加すると出てくる矢印ボタンから、メニューをたどると簡単に出せます。

矢印を押して出てくるメニュー、propertyの中にあります。

2タイプある文から、まず
loopIn(type = "cycle", numKeyframes = 0)
を適用してみた結果


グラフの実線が実際作ったキーフレームで、点線がエクスプレッションでループされているアニメーションです。
実際のキーフレームより前の時間だけループします。
お察しの通り、このloopInはあまり使う事はないでしょう・・


ちなみに、エクスプレッションの動きをグラフに点線で表示するにはココを押す





次が本命
loopOut(type = "cycle", numKeyframes = 0) 
結果は、


そうそうコレコレ。
ちなみに、前後にループを足すには少々ややこしい文が必要になりますので、特に理由がなければコンポジションの先頭でキーフレームを作って、このloopOutを使うと良いと思います。

loopIn loopOut の意味はそんなところです。
そして気になるのが()の中。type="cycle"とありますが、これはどのようにループするかというのを指定しています。

試しに「cycle」を「pingpong」と書き換えてみます。
loopOut(type = "pingpong", numKeyframes = 0) 
結果は、


いったりきたり、RAMプレビュー再生のオプションにもある、あの動きです。
んん、たまには使うかなぁ

次は、「cycle」を「offset」と書き換えてみます。

loopOut(type = "pingpong", numKeyframes = 0) 
結果は、


最初と最後のキーの値を差分して足したうえでループしてくれます。
階段をはずみながら落ちていくボール、など使うケースはそこそこあるかと思います。
このループ形式は、3Dソフトで歩行アニメーションの足の位置なんかで結構使ったりします。


次はとてもステキなループです。
とりあえず結果は、


タイムラインの最初に実際のキーフレームがあります。
そして、レイヤーにマーカーを付けて、コメントに数値を打ち込みます。
あら!マーカー位置から数値の回数分ループしてるじゃないですかっ!もちろんマーカーの位置を移すとアニメーションもついていきます。
マスクシェイプにも使えるので、キャラのまばたきなど利用するシチュエーションは無限大!
ちなみにキーフレームは実際のアニメーションに反映されませんので、 そこも動かしたい場合は1Fにマーカーを作りコメントを「1」とすると良いでしょう。

エクスプレッション文は

loopTime = 0;
numLoop = 0;
keyN = 0;
shiftTime=0; 
cycleTime =key(numKeys).time; 

if (marker.numKeys){
keyN = marker.nearestKey(time).index;
if (marker.key(keyN).time > time) keyN--;
}

if (keyN > 0) {
numLoop = parseInt(marker.key(keyN).comment);
shiftTime = time - marker.key(keyN).time;

if (shiftTime < numLoop * cycleTime ) {
loopTime = shiftTime % cycleTime ;
}
}

valueAtTime(loopTime )
 

これはとあるサイトから見つけてきた文で、ボクは理解していません・・・
難しく考えずにコピペで使っちゃいましょう!

↓レイヤーマーカーではなく、コンポジションのマーカーを参照できるよう書き換えてみました。
複数レイヤーを同じタイミングでループさせたい場合はこちらの方がラクですね。


loopTime = 0;
numLoop = 0;
keyN = 0;
shiftTime=0;
cycleTime =key(numKeys).time;

if (thisComp.marker.numKeys){
keyN = thisComp.marker.nearestKey(time).index;
if (thisComp.marker.key(keyN).time > time) keyN--;
}

if (keyN > 0) {
numLoop = parseInt(thisComp.marker.key(keyN).comment);
shiftTime = time - thisComp.marker.key(keyN).time;

if (shiftTime < numLoop * cycleTime ) {
loopTime = shiftTime % cycleTime ;
}
}

valueAtTime(loopTime )
 


AE マーカーの位置でブルブル

このエクスプレッションはココのサイトからの転載です。

スケールにエクスプレッションを追加し、下記文をコピペ
レイヤーにマーカーを追加して再生してみて下さい。

n = 0;
t = 0;
if (marker.numKeys > 0){
n = marker.nearestKey(time).index;
if (marker.key(n).time > time) n--;
}
if (n > 0) t = time - marker.key(n).time;
amp = 15;
freq = 5;
decay = 3.0;
angle = freq * 2 * Math.PI * t;
scaleFact = (100 + amp * Math.sin(angle) / Math.exp(decay * t)) / 100;
[value[0] * scaleFact, value[1] / scaleFact];

タイムライン上でマーカー位置のタイミングに合わせて、スケールがしばらくブルブルしておさまります。
ブルブル具合は、エクスプレッション内の「amp」「freq」「decay」で調整できます。

amp : 大きいほど数値の増減がまします
freq : 周波数 大きいほど小刻みになります
decay : おさまるまでの時間 大きいと短い時間でおさまり、小さいと長い時間ブルブルします

また、最後の行の「/」を、「*」に変更すると、これまた違った種のブルブルになります。


2012年1月12日木曜日

AE 値の参照

エクスプレッション
他レイヤーの値を参照。

同じ動きをさせたり、タイミングをずらして追従せたり、色々なことが出来ます。



例)

レイヤーを2つ用意します 





 緑のレイヤーに上下移動のキーフレームをつけます




青レイヤーの位置にエクスプレッションを追加し、縄のアイコンから緑レイヤーの位置にドラッグします。これで青が緑の位置を参照したことになり、同じ位置に移動し追従します。

式は
thisComp.layer("グリーン 平面 1").transform.position

「thisCopm.layer」「グリーン 平面1」の部分を書き換えれば、別のコンポジションのレイヤーからも参照できます。


式を下の通りに書き変えてみます

A=hisComp.layer("グリーン 平面 1").position;
[position[0].A[1]]

青は緑のy位置だけを参照する事になり、x移動は自由になります。

1行目で変数Aにグリーン平面の位置を代入しています。
2行目で、xにposition[0]と書く事で自身のx値をそのまま、yにはA[0]と書く事で緑のy値を指定しています。



式を下の通りに書き変えてみます

offset=3;
offTime=(time-offset/30);
A=thisComp.layer("グリーン 平面 1").position.valueAtTime(offTime);
[position[0],A[1]]

これで、青が緑のyの動きを3コマ遅れで 追従します!遅れるコマ数は1行目のoffset=の後の数値で指定できます。


3行目の値の参照の後に「.valueAtTime(offTime);」という文が足されていますが、
この「.valueAtTime(数値)」は、指定した数値の時間の値を参照できるという優れものです。
数値に1と入れれば、コンポジションの1秒目(フレームなら30)の値を持ってきてくれます。
この例では()の中に offTime つまり、現時間+1行目で指定したコマ数を入れているので、常に3コマ時間をさかのぼって値を参照しています。


青レイヤーをいっぱいコピーして、エクスプレッションの
offset = ○
だけを少しずつ書き換える事で、波打つように追従させるアニメーションが出来ます。
実際にキーフレームを持っているのは緑だけなので、編集も楽チンです。


2012年1月11日水曜日

AE テキストで現在のフレームを自動表示



ムービーのはじっこにフレームナンバーやタイムコードを入れたいことって結構あると思います。
そんな時、新規平面を用意しテキスト - タイムコードを使う方法もありますが、文字がアルファで抜けてなかったり、無駄な平面が増えたりとあまり良い事がありません。

新規テキストを作成し、ソーステキストにエクスプレッションを適用し下記文をコピペすれば見た目も美しく簡単かもしれません。

-- フレームの場合 --
timeToFrames(t = time + thisComp.displayStartTime, fps = 1.0 / thisComp.frameDuration, isDuration = false)

-- タイムコードの場合 --
timeToTimecode(t = time + thisComp.displayStartTime, timecodeBase = 30, isDuration = false)

また、フォント、塗り・線 などお好みに設定したうえで、上の画像のようにレイヤーの「テキスト」を選択した状態で、アニメーションプリセットを保存すれば、次回からは 新規テキスト作成 - アニメーションプリセットの適用 の2ステップで同じ状況を再現できます。


--余談--
ソーステキストのエクスプレッションは他のレイヤーの数値を引っ張ってくれば、


こんな事も出来たりと、楽しいです。
AEの性質上数値の上下が逆になってますが・・・



AE タイヤの自動回転



車体レイヤーの移動距離にしたがって、タイヤを自動で回転させるエクスプレッションです。

まずフォトショップでレイヤーに分かれたPSDを用意します。


AEに 「コンポジション-レイヤーサイズを維持 」で読み込みます。
レイヤーサイズを維持しているので、タイヤはそれぞれの直径サイズになっています。

 タイヤ「front」「back」の親を「body」にします。
「body」の位置にキーフレームを打ち、移動します。

タイヤ「front」「back」の回転にエクスプレッションを追加し、下記の文をコピペします。

d=width;
pi=Math.PI;
circ=d*pi;
rot=360/circ;
movex=thisComp.layer("body").transform.position[0];
rot*movex;

プレビューするとタイヤが回転しているのが確認できます。



----エクスプレッションの補足---

赤字のアルファベットは変数であり、好きな名前をつける事が出来ます。 
青文字各行の説明

d=width;       ← width(レイヤー自身の横幅ピクセル)変数dに代入
pi=Math.PI;     ← MathPI(円周率関数)を変数piに代入
circ=d*pi;      ← dpiをかけた数値を変数circに代入、つまりタイヤの円周がピクセルで入っています
rot=360/circ;    ← 360をcircで割った数値を変数rotに代入
movex=thisComp.layer("body").transform.position[0];   ← 「body」の位置xを変数movexに代入
rot*movex;     ← rotmovexをかけた数値を最終的な回転とする

別のプロジェクトでこの文を使いまわす場合は、5行目の("body")を書き換えるだけでOKです。

ちなみに、エクスプレッションを編集し易くする為に変数を使用してますが、シンプルに

movex=thisComp.layer("body").transform.position[0];
(360/(width*Math.PI))*movex;

の2行でも同じ動作をします。