A Routine runs a Function and allows it to be suspended in the middle and be resumed again where it left off. This functionality is supported by the Routine's superclass Thread. Effectively, Routines can be used to implement co-routines as found in Scheme and some other languages.
A Routine is started the first time -next is called, which will run the Function from the beginning. It is suspended when it "yields" (using Object: -yield within the Function), and then resumed using -next again. When the Function returns, the Routine is considered stopped, and calling -next will have no effect - unless the Routine is reset using -reset, which will rewind the Function to the beginning. You can stop a Routine before its Function returns using -stop.
When a Routine is scheduled on a Clock (e.g. using -play), it will be started or resumed at the scheduled time. The value yielded by the Routine will be used as the time difference for rescheduling the Routine. (See -awake).
Since Routine inherits from Thread, it has its own associated logical time, etc. When a Routine is started or resumed, it becomes the current thread.
Routine also inherits from Stream, and thus shares its ability to be combined using math operations and "filtered".
Creates an instance of Routine, passing it the Function with code to run.
func |
A Function with code for the Thread to run. |
stackSize |
Call stack size (an Integer). |
The Function class provides its own shortcut method r
that calls Routine.new
, thus one can also write:
This method performs differently according to the Routine's state:
inval
argument.inval
argument as the return value of yield
.thisThread
: Since Routine inherits from Thread, it will become the current thread when it is started or resumed; i.e. thisThread used in the Routine Function will return the Routine.
Time: Just before next
is called, thisThread
points either to another Routine, or the top-level Thread. During the next
call, this becomes the parent thread (see Thread: -parent). next
then evaluates on the parent's clock, at the parent's logical time.
Synonyms for next
are -value and -resume.
nil
, if the Routine has stopped.When a Routine is started by a call to this method (or one of its synonyms), the method's argument is passed on as the argument to the Routine Function:
After the Routine has yielded (it has been suspended at the point in its Function where yield
is called on an Object), a call to this method (or its synonyms) resumes executing the Function and the argument to this method becomes the return value of yield
. To access that value within the Function, you have to assign it to a variable - typically, the argument of the Function is reused:
Typically, a Routine yields multiple times, and each time the result of the yield is reassigning to the argument of its Function.
Equivalent to -next.
Equivalent to -next.
Equivalent to the Routine Function reaching its end or returning: after this, the Routine will never run again (the -next method has no effect and returns nil
), unless -reset is called.
Causes the Routine to start from the beginning next time -next is called.
If a Routine is stopped (its Function has returned or -stop has been called), it will never run again (the -next method has no effect and returns nil
), unless this method is called.
A Routine cannot reset itself, except by calling Object: -yieldAndReset.
See also: Object: -yield, Object: -alwaysYield
Schedules the Routine on the given Clock, at a time specified by quant
. At that time, the Routine will wake up (by calling Routine: -awake), setting the clock and time and evaluating the Routine. If the Routine yields a number, this number of beats will be added to the current time and the Routine will be rescheduled. (This behavior is compatible with scheduling a Stream or a Function.)
clock |
a Clock, TempoClock by default |
quant |
see the Quant helpfile |
using Object: -idle within a routine, return values until this time is over. Time is measured relative to the thread's clock.
If a Routine is currently scheduled on a clock, it will be expected to "awake" at a specific time, on a specific clock. reschedule
allows you to change to a different clock or to a later time.
argClock |
The new clock on which to run the Routine. If |
quant |
A quantization specifier, identifying a time later than the next-scheduled time. If |
When switching the clock, reschedule
will first calculate the new "awake" time, in beats, based on the given quant
. Then it converts that beats
value into the new clock's beats
for the same instant in time. If the old clock's tempo is 1 and the Routine is waiting for 1 beat, the next "awake" should happen after one second. When rescheduling without a quant
, the next "awake" will still happen after one second -- but the beats
value will adjust for the new clock.
quant
will resolve to a time later than the Routine's current nextBeat
. If you try to force it earlier, there is likely to be some discontinuity. This is because the Routine cannot reschedule until it wakes up normally. The workaround is to switch the Routine into a different Task wrapper.This method is called by a Clock on which the Routine was scheduled when its scheduling time is up. It calls -next, passing on the scheduling time in beats as an argument. The value returned by next
(the value yielded by the Routine) will in turn be returned by this method, thus determining the time which the Routine will be rescheduled for.
inBeats |
The scheduling time in beats. This is equal to the current logical time (Thread: -beats). |
inSeconds |
The scheduling time in seconds. This is equal to the current logical time (Thread: -seconds). |
inClock |
The clock which awoke the Routine. |
Convert the routine to a (subclass of) Pattern.
A Prout that (in response to asStream
) acts as a generator of independent copies of the original routine (the receiver of p
).
Any subclass of Pattern returns a Stream in response to asStream
. However, the exact subclass of Stream returned depends on the class of the Pattern. In particular, a Prout returns a Routine in response to asStream
. Thus, a way to make an independent copy of a Routine is to make a Prout from it with the method p
, and then create a new Routine from this Prout. In the example immediately below, this second step is done explicitly by calling asStream
, but in the context of passing arguments to other Patterns, this latter step usually happens automatically, as shown in later examples.
.p.asStream
run the original routine's function from the beginning. There is no copy of the internal state of the original routine, only its function is copied. On a related note, Routine.copy
(inherited from Thread
) does not create a new, separate Routine; it just returns the receiver.The Routine method p
is mostly useful in the context of using the result of Routine-returning expressions as arguments to Patterns, when the intent is to use the Routine's result as a pattern, i.e. multiple occurrences acting independently of each other.
In the following example, directly reusing the same Routine object twice in a Pseq
fails to duplicate its output, because the first embedding fully consumes the routine's (non-nil) output.
Routine inherits from Thread, which allows access to some of its state:
The elapsed beats (logical time) of the routine. The beats do not proceed when the routine is not playing.
The elapsed seconds (logical time) of the routine. The seconds do not proceed when the routine is not playing, it is the converted beat value.
The thread's clock. If it has not played, it is the SystemClock.