millis()
We quickly reviewed some concepts from last week and then went over how to do the second part of the homework. The code that we worked on is here:
We've already seen how to draw shapes and make them move on their own. This motion would typically start when the program started running and continue forever. Or maybe it would be triggered or changed by user interaction in some way. Today we are going to talk about timing: how to create motion that is in some way scheduled, chorreographed, triggered with some delay, or that repeats at some interval.
Time-based media are most typically thought of as sound and the moving image, although digital media like video games and motion graphics are also included. Today we will see some coding techniques to implement time-based behavior in interactive computer programs of the sort we've been working on, and we will think about how to work with the domain of time in ways that are analagous to how we have so far been working within the domain of space.
One incredible example of early time-based media is the work of Oskar Fischinger. Fischinger was an artist in the first part of the 20th century and a pioneer in animation and motion graphics. A classic example of his work is the animated film Studie nr 8, 1931.
You might be inclined to think of the history of animation as preceding that of cinema, but in fact there is a historical case to be made for understanding these in the opposite order. Several modern artists and illustrators in the early 20th century were inspired by cinema to create animation, and used the principle of cinema — a continuous strip of film, divided into cells of static imagery that flicker by the viewer's eyes — to create early animation and motion graphics. In fact, many of these innovators actually worked on film strips, drawing and etching into the cells by hand.
Fischinger actually contributed to Disney's Fantasia but quit uncredited because of creative differences. In 2012, the Whitney museum showed an exhibition of his artwork called "Space Light Art". There is great documentation for this show online and I highly recommend taking a look!
millis()
Toward the end of the day back in week 4 ("Making Things Move") we saw how you could use variables and some basic arithmetic to make things move "on their own".
We looked at this pattern:
circleX = 300 def setup(): size(600,600) stroke(50,50,150) fill(200,200,255) def draw(): global circleX background(255) ellipse( circleX,300, 50,50) circleX = circleX + 1That draws a single circle, and to make the circle move horizontally, it create a variable which gets used to specify the x position of the circle. The value of this variable is then changed a little every frame, by incrementing or decrementing the variable.
We expanded on this example using conditionals to ask questions about the position of the circle, and changed its movement based on that. For example:
circleX = 300 def setup(): size(600,600) stroke(50,50,150) fill(200,200,255) def draw(): global circleX background(255) ellipse( circleX,300, 50,50) circleX = circleX + 1 if circleX > width: circleX = 0
Today we are going to build on this, but instead of using conditionals to ask questions about movement in space, we're going to use them to ask questions about time.
Unlike other time-based digital tools — like Adobe Premiere, After Effects, Audacity, or others — Processing does not give you an explicit, visual timeline. If you want to create timed events, you have to think in terms of numbers.
Processing gives us a command just for this
purpose: millis()
(check the reference)
This command returns a number that corresponds to the number
of milliseconds since the program started running. A
millisecond is one thousandth of a second, so 1000
milliseconds = 1 second. millis()
is like a
stopwatch that starts when your program starts, and keeps
running as long as the program is running.
Even though this isn't a visual timeline, we can imagine a
visual timeline in our thinking. millis()
returns a number that represents a playhead or a marker,
moving along a timeline. Thus, we can use this like a
variable to ask questions about the status of our sketch,
similar to mouseX
and mouseY
. We
can also use it to "save" or remember a position of this
playhead on the timeline.
This one basic command is enough to implement many different types of timed events, from simple things up to more complicated ones.
Do something for the first three seconds my sketch is runningWe could implement that with:
if millis() < 3000: # Do somethingSimilarly, think about this pseudocode:
Wait three seconds, then do something for two secondsWhich we could implement with this:
if millis() > 3000 and millis() < 5000: # Do something
What should Do something
be in this case? Well
let's say we want to draw a shape and only have it move
according to the above timing. We could combine movement and
timing like this:
circleX = 300
def setup():
size(600,600)
stroke(50,50,150)
fill(200,200,255)
def draw():
global circleX
background(255)
ellipse( circleX,300, 50,50)
if millis() < 3000:
circleX = circleX + 1
if circleX > width:
circleX = 0
Notice how now, the circle's position is being updated just
like before, but only when that conditional
about millis()
is True
. In other
words, only during the interval described by the Boolean
expression millis() < 3000
We can get a bit more complicated by introducing variables to use for saving time values. For example:
circleX = 300 startTime = 3000 def setup(): size(600,600) stroke(50,50,150) fill(200,200,255) def draw(): global circleX background(255) ellipse( circleX,300, 50,50) if millis() > startTime and millis() < startTime + 2000: circleX = circleX + 1 if circleX > width: circleX = 0When would this move the circle?
This doesn't seem that interesting since I'm
setting the variable startTime
to a hard-coded
value. But because I am using a varibale as a placeholder
for that value, that means that the value could change.
circleX = 300 startTime = 3000 def setup(): size(600,600) stroke(50,50,150) fill(200,200,255) def draw(): global circleX background(255) ellipse( circleX,300, 50,50) if millis() > startTime and millis() < startTime + 2000: circleX = circleX + 1 if circleX > width: circleX = 0 def keyPressed(): global startTime startTime = millis()Now when would this move the rectangle? Think about when the
startTime
variable gets
set. Whenever any key is pressed, startTime
gets
set to the current value of millis()
— or
in other words, it marks where the imaginary playhead is at
when the user presses a key. Then, each
time draw()
runs, it checks if the current value
of millis()
is greater
than startTime
, and less
than startTime
plus 2000, or two seconds. So, in
pseudocode we could say:
Every time the user presses any key, move a circle to the right for two seconds.
Notice that the circle still starts moving at 3 seconds and
moves for 2 seconds. This is because of the initial value
that we are using for startTime
, still carried
over from the previous code snippet. What could you set the
initial value of startTime
to so that it would
not appear like this? Think about when that conditional
is True
and what value would make it
not True
. Highlight for
answer: startTime = -2000
Wait a while, then do something foreverIn this example we wait two seconds, and then start moving the square and never stop
position = 0 def setup(): size(600,600) smooth() stroke(50,50,150) fill(200,200,255) def draw(): global position background(255) ellipse(position,300,100,100) # wait two seconds, then move forever: if millis() > 2000: position = position + 1
Wait a while, then do something for a while, then stopIn this example we wait two seconds, then start moving the square, move the square for one second, and then stop.
position = 0 def setup(): size(600,600) smooth() stroke(50,50,150) fill(200,200,255) def draw(): global position background(255) ellipse(position,300,100,100) # only increase the position if millis() is greater than two seconds but # less than three seconds if millis() > 2000 and millis() < 3000: position = position + 1
Do something for a little while, then stop Wait a little while and repeat the aboveThis example is similar to example 2 above, but it has a variable which gets added to the timing values. Then another
if
statement periodically resets that
variable. This causes the whole timing process to reset,
doing this forever.
position = 0 startTime = 0 def setup(): size(600,600) smooth() stroke(50,50,150) fill(200,200,255) def draw(): global position, startTime background(255) ellipse(position,300,100,100) # move for a little while, then stop if millis() > startTime and millis() < startTime + 2000: position = position + 1 # wait a little while, then reset the startTime variable, # so the above timing starts over: if millis() > startTime + 4000: startTime = millis()
So far we have been using variables either for numeric things
(like shape sizes, positions, or colors) or for
Boolean True
/False
values. All of
these things are ways of keeping track of what the program is
doing in a given moment.
Today we saw how we could use timing
and millis()
to affect how the program might
change over time.
But what if we want the program to change over time in a way that is not explicitly tied to timing. For example, what if we wanted the user to be able to change what the program was doing? Or if we want the program to operate in different phases or modes, and move through those?
In computer science, the term for keeping track of the status of the program is state. We might ask, what state or phase is the program in? This could be used to implement the levels of a video game for example.
The way to do this is by using variables. This is not very different from anything that we have been doing so far. I just want to show a few examples that emphasize how you can use variables in some slightly different ways.
'n'
key (for "on") the light
goes on, and when the user presses the 'f'
key
(for "off") the light goes off.
def setup(): size(800,800) def draw(): background(0)
How should we implement the "on" key? Well we could use the
Boolean keyPressed
variable, like this:
def setup(): size(800,800) def draw(): background(0) if keyPressed and key == 'n': background(255,255,200)But that only works if the user is holding down the
'n'
key. What if we want this to work not
like a button that must be held down, and more like a switch
that can be flipped?
Could we improve on things by using the def
keyPressed()
block? That would look like this: block,
def setup(): size(800,800) def draw(): background(0) def keyPressed(): if key == 'n': background(255,255,200)That doesn't really seem to help things. Now, whenever the user presses the
'n'
key, the light flashes on
for a split second. It still doesn't stay on.
We need a way to keep track of whether the light
has been pressed or not. In other words, we need to keep
track of the state of the light
switch. We'll do this with a variable. Since this variable
will only have two values (on, or off) we can use a Boolean
variable for this purpose. Initially I will set the variable
to False
to signify that the light is off.
switchState = False def setup(): size(800,800) def draw(): if switchState: background(255,255,200) else: background(0)Now, whenever the variable
switchState
is True
, I will draw the light as on, and
whenever the variable is False
, I will draw the
light off. But this is not doing anything yet. Why? Because
I am not changing this variable anywhere! Let's try to
change it.
When the user presses the 'n'
key, let's make
the variable True
:
switchState = False def setup(): size(800,800) def draw(): if switchState: background(255,255,200) else: background(0) def keyPressed(): global switchState if key == 'n': switchState = TrueOK! This is getting is somewhere. Now when the user presses the
'n'
key, switchState
is set
to True
, and so the light will be drawn on.
The last thing is to let the user turn it off. Have a look at this:
switchState = False def setup(): size(800,800) def draw(): if switchState: background(255,255,200) else: background(0) def keyPressed(): global switchState if key == 'n': switchState = True if key == 'f': switchState = FalseGreat! So now we have a variable (
switchState
)
that keeps track of whether the user has pressed a
certain key or not. In other words, it is keeping track of
the state of the program. And we can
use if
statements to change the value of that
variable.
Of course, we can create programs with even more states than just on / off. As a simple example, we could repeat the pattern from the above example to have several different on / off variables. Let's create a program that let's the user turn three shapes on or off:
rectOn = False ellipseOn = False triangleOn = False def setup(): size(800,800) rectMode(CENTER) def draw(): background(255) if rectOn: fill(255,155,155) rect(200,400,50,50) if ellipseOn: fill(155,255,155) ellipse(400,400,50,50) if triangleOn: fill(155,155,255) triangle(600,375, 625,425, 575,425) def keyPressed(): global rectOn, ellipseOn, triangleOn if key == 'q': rectOn = True if key == 'a': rectOn = False if key == 't': ellipseOn = True if key == 'g': ellipseOn = False if key == 'o': triangleOn = True if key == 'l': triangleOn = FalseNotice that I've named my variables using the pattern
triangleOn
. This creates a nice way of
reading your code so that you're looking at the if
statements, you can read it like If the
triangle is on
.
What if we didn't want to have different keys for on and off, but instead wanted the same key to turn each shape on and off? We call this a toggle. And it looks like this:
rectOn = False ellipseOn = False triangleOn = False def setup(): size(800,800) rectMode(CENTER) def draw(): background(255) if rectOn: fill(255,155,155) rect(200,400,50,50) if ellipseOn: fill(155,255,155) ellipse(400,400,50,50) if triangleOn: fill(155,155,255) triangle(600,375, 625,425, 575,425) def keyPressed(): global rectOn, ellipseOn, triangleOn if key == 'q': rectOn = not rectOn if key == 't': ellipseOn = not ellipseOn if key == 'o': triangleOn = not triangleOnNow, in those
if
statements, I'm setting each
variable to the opposite of whatever it currently is. This
creates the toggle effect.
Lastly, we can keep track state that is more than on / off. Have a look at this code:
dayEveningNight = 1 def setup(): size(800, 800) def draw(): if dayEveningNight == 1: background(155,155,255) elif dayEveningNight == 2: background(25,25,75) elif dayEveningNight == 3: background(0) def keyPressed(): global dayEveningNight if key == 'q': dayEveningNight = dayEveningNight + 1 if dayEveningNight > 3: dayEveningNight = 1Here, I am using a variable
dayEveningNight
that is holding the values 1
, 2
,
or 3
. I am using an if
statement
in the draw()
block that checks what the value
of this variable is and draws something accordingly. And
then, in the keyPressed()
block, I am
incrementing that variable based on what the user has
pressed. If the variable gets larger than 3
, I
am reseting back to its initial value of 1
.
You can use these same principles to keep track of many kinds of state within your program. For example, if a user is entering a password, or the levels of a game.
Functions are a way to organize your code.
Now that you've started thinking about the midterm project, you will be working on a computer program that is a little bit longer and a little bit more complicated. You need a way to keep this organized and manageable. Functions give you a technique for how to do that.
(Another strategy for code organization involves a technique called object-oriented programming, which uses things called classes and objects. We might talk about this later in the semester if there is time and interest.)
A function takes any sequence of commands, groups them together into a block, and gives that block a name. Then, just by using that name, you can automatically run all those commands.
New syntax. Let's say I have this sketch I'm working on:
def setup(): size(600,600) def draw(): # Pretend that in here # I have many many commands # to draw a landscape.
To use a function, I must first define it. I pick any name that I'd like (as long as it's not a special Processing reserved word) and then I create a new block like this:
def drawLandscape(): # Pretend that in here # I have many many commands # to draw a landscape.The term
drawLandscape()
is arbitrary. I could
have called it spaghetti()
, but like with
variables, I recommend that you use informative function
names that describe what the function does and that will
help you remember and understand later what the function is
doing.
This syntax is called the function definition or implementation, and this bit of code is described as defining or implementing a function.
Now you would use this new function simply by specifying its name like this:
drawLandscape()This is referred to as calling or invoking the function. Now, just specifying this function name is equivalent to invoking all the commands contained within the function definition.
If calling a function looks familiar to you, that is because nearly everything that you've been doing all semester has been calling functions. We just weren't refering to it that way. I've been referring to them as commands. All the commands that you have been using thus far are actually functions that Processing has already defined for you in advance.
rect()
, background()
, map()
:
these are all functions that Processing has defined for you,
that you are able to use simply by
calling. The len()
command is a function that
Python defines for you. The setup()
and draw()
blocks are also functions, but they
are special functions that Processing requires you
to define. When you run a sketch, the Processing system
starts by first calling your setup()
function,
then calling your draw()
function many times,
as we talked about in week 3.
Putting all these pieces together, my initial sketch would now look like this:
def setup():
size(600,600)
def draw():
drawLandscape()
def drawLandscape():
# Pretend that in here
# I have many many commands
# to draw a landscape.
Here you can see the definition and
the invocation of this new function
called drawLandscape()
.
Some notes on this new syntax:
You may be curious about why I
am calling the function before (or
above) where I am defining it. The
order does not matter. When you run your sketch,
Processing (and Python) looks through your entire
program for all function definitions and stores them
in memory, ready and waiting to be invoked. Only after
all functions are defined does it then automatically
invoke the setup()
function for you,
starting your sketch running. In other words,
functions can be defined in any order.
But, make sure that all your functions are defined in global space. Functions cannot be defined inside other functions. (They actually can, but that is a more advanced topic that we will not touch on this semester. Javascript programmers might be familiar with this technique, as it is more common in that language.)
If you really want to keep your code organized and manageable, consider putting some functions in different tabs. You can make tabs with helpful and informative names, and then put different function definitions in them. Perhaps you have a tab called "User input" and a tab called "Draw code". Experiment with whatever works for you.
IMPORTANT UPDATE:
Initially, I neglected to mention that if you want to
use multiple tabs, and include other code in your tabs
like function definitions, you need to use
the import
command in your main tab to be
able to access those functions. Examples of how to use
this is below.
This usage of functions is what is
called modularity: breaking down a big task
into smaller modules, that are themselves each more
manageable. Remember that we read about
modularity in the Lev Manovich chapter. You
could imagine that I might continue expanding the above
example with additional functions, and
my draw()
block might look like this:
def draw(): drawLandscape() drawClouds() drawCar() moveClouds() moveCar() }Looking at this code, we don't know what each of those functions does, but we can start to get an abstract, high level understanding of what is going on in this sketch.
Functions can work really well with the topic
of state variables from last
week. For example, if you were doing the game option for the
midterm, you could implement your different levels like
this: NOTE: Please pay
attention to how I'm using the import
command in the main tab to import the
function definitions from the other two tabs.
# First, main tab from level_1 import * from level_2 import * level = 1 def setup(): size(800,800) def draw(): # ... other code here ... if level == 1: drawLevel1() elif level == 2: drawLevel2() # ... more code down here ...
# Tab: level_1.py def drawLevel1(): # draw code goes here ...
# Tab: level_2.py def drawLevel2(): # draw code goes here ...
Keep this in mind as you work on the midterm!