Выражение random after effects
Randomness is one of the most asked about uses of expressions and After effects gives us a pretty good arsenal of tools with which to implement it. The basic random() method has several different flavors. Some examples would probably be helpful in understanding the different ways you can call random():
So you can see that the random() method is pretty flexible.
We'll illustrate how you might use this with a few examples. Let's say you wanted to randomly position a layer within the comp. Any of these expressions for Position will do the trick:
While the expressions above do indeed position the layer randomly, if you try any them you'll discover what seems to be a serious limitation of the random() method. You get a different result on each frame. This is great if you want to have a layer frenetically hop all over the place. Most of the time though, you want a more controlled randomness. This is where seedRandom() comes in.
Seed Random to the Rescue
seedRandom() is a clever method that Adobe came up with to let us control the rate of randomness. Actually it has more than one use. You can also use it just to change the sequence of random numbers generated by random(). seedRandom() has two parameters. The first one is the random seed, and the second is a true/false flag that tells After Effects whether or not the random numbers generated by random() will be "timeless". We'll get back to that "timeless" business in a minute, but first let's look at the other parameter. In its simplest form, seedRandom just seeds the random number generator to produce a particular sequence of random numbers that will change on every frame.
As an example, these two expressions will generate completely different sequences of random numbers (which will change on each frame):
OK - here comes the cool part. By setting the second parameter of seedRandom() to "true", the numbers generated by random() will be the same on every frame. Here's an example expression for Position that will move a layer to a random position and hold it at that location:
Remember that without the seedRandom() call, the layer would bounce around to a different Position on each frame. How does that help us? Having a layer move to a random Position and stay there is pretty boring, right? Yes it is. But all we need to do to get a new position is change the seed. So to get the layer to move around randomly in an orderly fashion, all we have to do is update the seed periodically and get a new random Position from random(). Let's construct an example. Say we want our layer to move to a new random Position every half second. All we need is some new code in our expression that will change the seed every half second. How would we do that? Remember that the Math.floor() JavaScript function rounds down to the nearest integer. We can use that to generate a seed that changes every half second by dividing the current time by .5 and rounding down to the nearest integer. Here's an example (you would change the value of "holdTime" to change the length of time each Position is held):
Now the layer still moves around randomly, but it does it a half-second intervals. Let's make a modification so that our motion stays in the "title safe" area of the comp. We'll do this by restricting the results of the calls to random() to the values between 10% and 90% of the width and height of the comp. Here's the modified code:
random position every .5 seconds
(Note that there's a slightly simpler way to do this by using the posterizeTime() method in conjunction with seedRandom(), but we'll look at that when we delve into using expressions to manipulate time).
OK, that's better, but what if we wanted the layer to move smoothly from one Position to the other? This brings us to one of the most important concepts for generating random motion, which is that whenever you set the seed to a particular number, the random sequence generated by random() will always be the same. For example if we set the seed to 1, random() might give us .45633. If we change the seed to 2, we might get .78341. But if we change the seed back to 1, we will get .45633 again. So how does this help us generate smooth random motion? Let's look again at our example above. For the first half second, the seed calculates out to be 0. So for every frame in the first half second, the call to random() generates the same random Position based on a seed of 0. After the first half second, the seed changes to 1 and the random Position changes. So what if we were to "peek ahead" by changing the seed to the value that we'll be using in the next half second to see what the next Position will be, and start heading there? The sequence would be to calculate the seed for this half second, use that to get the random Position for this half second (which will be our starting Position), bump the seed by one and use that to see what the random starting Position for the next half second will be (which we'll use as the ending Position for the current half-second segment). OK, here's the code:
random motion - synchronized
You'll notice in the demo movie we've now got multiple stars, all at different random positions. They all have identical copies of the random position expression, so you might be wondering why they have different positions. It turns out that for any given random seed, the random values generated will be unique to each layer. Internally After Effects somehow modifies the seed so that it is unique for each layer, comp, property, effect, etc. Most of the time this is a very handy feature, but there are some occasions where it would be nice to be able to force two different layers to generate the same random numbers.
Anyway, to recap, the seed is derived from the time and the segment duration and becomes our "segment number" where segment 0 starts at time = 0, segment 1 starts at time = .5, and so on. Once we have the seed, we seed the random number generator with it and call random() to get the starting Position for this segment. Then we bump the seed, seed the random number generator with the new seed and call random() again to get the starting Position of the next segment (which is, of course, the ending Position for the current segment). Then we just use ease() to interpolate between the two Positions so that the layer eases from one Position to the next. You can change the interpolation from ease() to linear() for a different effect.
So that's all you need to generate basic random motion in After Effects. Ah, if only life were so simple.
If you duplicate the layer several times and preview the comp, you'll see that the layers moves to different Positions, but the movement of all the layers is synchronized. That is, they all arrive at their new destinations at the same time. That can be a useful effect, but what if you want something more chaotic? We'll look at a couple of things you can do to fix this.
One solution is pretty simple and gives results that can be quite useful. The other solution is much more complex (in both code and theory) but really blows the doors off this thing as far as opening up endless possibilities for randomness in After Effects. First the simple fix. We can modify the expression so the duration used by each layer is a random number within a given range, and would be different for each layer. Here's the modified code to do that:
random motion - more chaotic
As you can see, the only change from the previous version is the code at the beginning that seeds the random number generator with the layer's index and then calls random() to get the segment duration. This simple change can add quite a bit of chaos to the mix.
However, if you look closely at the motion, you begin to notice that each movement of any given layer always takes the same amount of time. What if what you really want is for each movement to take a random amount of time? This brings us to a major roadblock and we have to come up with a new and very powerful concept to get around it. Before we solve it, we need to take a little side trip.
Expressions Have No Memory
The implementation of expressions in After Effects is stateless. Here's the good news: that's what allows you to move the time marker to any frame in the comp and After Effects can immediately begin to show you that frame because it doesn't have to know what any expressions applied have done in the past. So at any given frame After Effects just calculates the expression and applies the results to the value that the property would have without the expression (i.e. the original value of the property plus the effect of any keyframes.) The implications of this are enormous. An example here might help.
Let's say that you wanted to add a random value from 0 to 10 degrees to the Rotation of a layer on each frame. You would expect sporadic movement but you might also expect the layer's Rotation to always be increasing in the clockwise direction. The code would seem to be simple enough. It seems like this should do the trick:
random rotation - failed attempt
When we look at the result though, it's clear that After Effects is not accumulating the effects of the random() function from frame to frame. At each frame it starts anew with the original value of Rotation (which is 0) and adds the random number to that. This is clearly not what we wanted or expected. So what do we do?
Unfortunately, what we often have to do in situations like this is to do the frame-by-frame accumulation ourselves, with code that we add to the expression. In other words, our expression has to start at frame zero and recalculate everything that has happened since then. To accomplish this we'll have to use some JavaScript to construct a "while" loop that goes through the comp one frame at a time and adds up the random numbers. The code for the loop looks a little complex, but after you do it a few times and you start to get a feel for what's going on it gets easier. Here's what the code looks like:
random rotation with accumulating loop
Now we're getting what we expected. This is accomplished by the "while" loop that loops through the code between the curly braces until the variable "j" is greater than the current time. Each time through the loop, j gets incremented by one frame's worth of time (thisComp.frameDuration). When the loop finishes, the variable "accum" contains the sum of all the random numbers generated for this and all previous frames. Even though the code is not extremely complicated, all this comes at a cost. At each frame, the loop has to revisit all previous frames. As your comp increases in length, render times can begin to bog down as the expression has more and more to do on each frame. So there are practical limitations on the use of this technique, but for most situations (short comps) it will work just fine. In fact, as you will see shortly, many applications of this technique don't require a one-frame granularity and are much less expensive in terms of render time. What i mean is that you end up looping through the comp in chunks of time much larger than a single frame, which extends the usefulness of the technique even further. You'll see.
Back to the Task at Hand
OK - let's get back to the problem of random durations for our random motion. In the process we'll end up using the loop-through-the-comp technique that we just investigated. Let's go over what we need to do. We want our layer to move smoothly from one Position to the next and we want this to happen over a random amount of time (within a range that we will define). Armed with "seedRandom()" and "while", we're ready to tackle this thing. First, let's just go ahead and take a look at the finished code:
random motion with random durations
Let's take a look at what has changed since the previous version of our random motion code. The main difference is the setup and execution of the "while" loop. There are several things going on here. The "while" loop is where the length of each "segment" of motion is determined. The segment length will be a random number between "segMin" and "segMax" (.3 and .7 seconds in this case). We are going to loop through the comp, starting at time 0, adding up the random segment lengths until we reach the current time. So, for our purposes here, it will not be necessary to revisit each frame every time the expressions runs - we only need to look at the start of each segment (which will be, on the average, about every .5 seconds or so). Meanwhile, we've also been incrementing "j" for each segment. That makes "j" unique for each segment and a perfect candidate for the seed for "seedRandom()". So when we drop out of the loop, we know the start and end times of the current segment ("start" and "end") and we have the seed ("j") that was used to generate the end time. Outside the loop (after the closing curly brace) we know that the random number generator is still seeded with the index ("j") for the current segment. So we call random() again to get the ending Position for this segment. We now have everything we need (segment start time ("start"), segment end time ("end"), and ending Position ("endVal")) except the starting Position. We can get that by backing up to the previous seed, re-seeding the random number generator, throwing away the first random number, and retrieving the starting Position ("startVal"). (See the sidebar below for a more detailed explanation of why we have to throw one random number away). Now we just apply the same ease() statement as before to generate our random-duration, smooth, random motion.
Everything from here on out gets much easier because it's all just variations on a theme. In the next section we'll extend these concepts to randomizing other properties.
Throw-away Random Numbers
Remember that when you seed the random number generator with a particular seed using the seedRandom(seed,true) method, you will always get the same sequence of random numbers on subsequent calls to random(). In this example, as we go through the loop, the first call we make to random() after setting the seed is to get the duration of the current segment. After we fall out of the loop we call random() again to get the ending Position for this segment. That's three random numbers (segment duration, x coordinate, y coordinate). When we set the seed back to the index of the previous segment to pick up the ending Position of that segment (which is the starting Position of the current segment) we need to remember that we've been using the first call to random() after setting the seed to get a segment duration. Therefore, after we set the seed back to "j-1", we need to throw away the first call to random() because it just gets us the duration of the previous segment (which we don't need). Then the next call to random() will generate the proper x and y coordinates of the starting Position (which was the ending Position of the previous segment). That's the reason for the line of code that plugs the previous segment duration into the "dummy" throw-away variable. If we didn't do this, we wouldn't get the same x and y coordinates that were generated as the ending Position when we were in the previous segment. Make sense? It's an important concept to wrap your brain around, so you may want to come back to it later if you don't get it yet.
After Effects counting seconds with the time expression
The values generated by this expression can then be used to drive movement by connecting a property value to the expression.
In the example above I rigged a text layer to preview the value generated by the time expression. As the composition is playing you see the seconds being counted in the composition panel through that rigged text layer. All I did was used a simple time expression to have After Effects generate those values.
Note: toFixed() limits how many numbers are allowed after the decimal
How Does the Time Expression Work in After Effects?
For example, If I double the time expression using multiplication it would read 8 seconds within a 4 second composition time.
A faster time readout using the time expression
Just look at that little line go! In the first example we get 1 degree for every second. So if we want to get a full rotation every second we need to know how many degrees are in 1 full rotation; which is 360 degrees.
By multiplying the value time provides by 360 we are asking After Effects to speed up the process exponentially. It now is going to complete a 1 degree move 360 times within one second.
Cool Examples of the Time Expression in After Effects
Here is an example of looping rotations at different speeds. Imagine if you had a bunch of gears that needed to rotate, or an astroid field that needed slight rotations for those cold heavy rocks.
In a nutshell, valueAtTime is an expression that tells After Effects to pull a value from a property (scale, position, slider, etc.) for a declared time.
You can download the time expression project file for After Effects by clicking the download button below.
Time Expression Money Rig for After Effects
I hope you see how awesome the time expression can be. There are a lot of use cases outside of what I went over in this article!
If you want to learn more about using expressions in After Effects we have a ton of other great expression content here on School of Motion. Here are a few of our favorite tutorials:
начну пожалуй с самого популярного экспрешшона рунета, автором которого является Тимур Константинов. вот скрипт:
mp = .1;//Чем больше значение, тем больше амблитуда
freq = 5;//Чем больше значение, тем больше частота
decay = 7;//Чем больше значение, тем меньше задержка
n = 0;
if (numKeys > 0)<
n = nearestKey(time).index;
if (key(n).time > time)<
n--;
>
>
if (n == 0)<
t = 0;
>else<
t = time - key(n).time;
>
if (n > 0)<
v = velocityAtTime(key(n).time - thisComp.frameDuration/10);
value + v*amp*Math.sin(freq*t*2*Math.PI)/Math.exp(decay*t);
>else<
value;
>
точнее - вот так:
amp = .1;
freq = 5;
decay = 7;
n = 0;
if (numKeys > 0)n = nearestKey(time).index;
if (key(n).time > time)n--;
>
>
if (n == 0)t = 0;
>elset = time - key(n).time;
>
if (n > 0)v = velocityAtTime(key(n).time - thisComp.frameDuration/10);
value + v*amp*Math.sin(freq*t*2*Math.PI)/Math.exp(decay*t);
>elsevalue;
>
вот ссылка на несколько полезных выражений
вот те,которыми я пользуюсь:
Для оси X:
Math.cos(S*time)*360
Для оси Y:
Math.sin(S*time)*360
Для колес эволюции и оси Z:
time*S
amp = 15; //amplitude (pixels)
freq = 10; //frequency (cycles per second)
n = 0;
if (numKeys > 0)<
n = nearestKey(time).index;
if (key(n).time > time)<
n--;
>
>
if (n == 0)<
t = 0;
>else<
t = time - key(n).time;
>
if (n > 0)<
v = velocityAtTime(key(n).time - thisComp.frameDuration/10);
amp = .05;
freq = 4.0;
decay = 2.0;
value + v*amp*Math.sin(freq*t*2*Math.PI)/Math.exp(decay*t);
>else<
value;
>
,где amp - амплитуда, freq - частота, а decay - затухание
Серега маслов, это вообще не Тимура скрипт)) А Хэрри Фрэнка с грэй машин)Но экспрешн реально полезный)
а по моему - это просто код программирования.
зная язык, можно самому написать без Генри и Тимура
Просто этот экспрешн я видел давным давно еще на старой версии сайта грэй машин) А вообще Хэрри Фрэнк очень неплохо рассказывает, советы дельные)
origin = [10,10,10];//описываем стандартный размер массива (таблица)
dimX = 10;//к-во ячеек по X
dimY = 10;//к-во ячеек по Y
dimZ = 10;//к-во ячеек по Z
gap = 50;//максимально расстояние между ячейками
gridRate = 500;//скорость движения частиц
holdTime = .1;//максимальная задержка по времени
//описываем необходимые переменные, заранее обнулив их
start = 0;
startX = 0;
startY = 0;
startZ = 0;
endX = 0;
endY = 0;
endZ = 0;
deltaX = 0;
deltaY = 0
deltaZ = 0;
end = 0;
j = 0;//переменная, необходимая для проведения цикла и дальнейшей проверки значений
while (time >= end) //цикл, аналог цикла "for"
seedRandom(j,true); //генератор случайных чисел
//задаем начальные позиции элемента
startX = Math.floor(random(dimX))*gap + origin[0]; //по X
startY = Math.floor(random(dimY))*gap + origin[1]; //по Y
startZ = Math.floor(random(dimZ))*gap + origin[2]; //по Z
j +=1;//наращиваем значение j на единичку с каждым шагом, т.е. в следующем цикле
произойдет новое перемещение объекта
seedRandom(j,true) //снова рандомим
start = end; //присваиваем значению "end" значение "start" для просчета
конечной позиции элемента
//========«строчим все как для старрта»==========
endX = Math.floor(random(dimX))*gap + origin[0];
endY = Math.floor(random(dimY))*gap + origin[1];
endZ = Math.floor(random(dimZ))*gap + origin[2];
deltaX = Math.abs(endX - startX);
deltaY = Math.abs(endY - startY);
deltaZ = Math.abs(endZ - startZ);
//теперь конечная позиция будет менятся в каждом новом цикле
end += (deltaX + deltaY + deltaZ)/gridRate + 3*holdTime;
>
p1 = start + deltaX/gridRate; //определено рандомное расположение объекта и первое движение по X
p2 = p1 + holdTime; //второе движение c задержкой
p3 = p2 + deltaY/gridRate;//третье со смещением по Y
p4 = p3 + holdTime;// четвертое с задержкой
p5 = p4 + deltaZ/gridRate; //пятое со смещением по Z
//========«проверка на порядок действий (порядок движения) и движение по-этапное»=========
if (time < p1)
ease(time,start,p1,[startX,startY,startZ],[endX,startY,startZ]) //изменение позиции с мягким ключом (короче в виде часиков песочных :D)
>
else if (time < p2)
[endX,startY,startZ] //первое смещение
>
else if (time < p3)
ease(time,p2,p3,[endX,startY,startZ],[endX,endY,startZ]) //третье движение
>
else
if (time < p4)
[endX,endY,startZ] //четвертое
>
else
if (time < p5)
ease(time,p4,p5,[endX,endY,startZ],[endX,endY,endZ]) //пятое
>
else
[endX,endY,endZ]
>
//движение элемента теперь будет происходить рандомно по сетке по направлениям X,Y и Z
Читайте также: