Clocks

Supriya provides a collection of musical-time-aware clocks which permit scheduling callbacks relative to both seconds, beats and measures, and which - by extension - understand tempo, time signatures and downbeats.

These clocks come in both threaded and asynchronous flavors, as well as online and offline.

Threaded clocks handle callbacks in their own thread, while asynchronous clocks hook into an asyncio event loop. Use threaded clocks when experimenting interactively in the terminal or building simple applications. Use asynchronous clocks when building more complex applications which integrate with other asyncio-aware libraries like aiohttp, python-prompt-toolkit or pymonome.

Offline clocks implement the same interface as their online counterparts, but process their callbacks as fast as possible, regardless of the amount of real time consumed. This makes them suitable for unit-testing logic expecting online clocks, or for implementing non-realtime rendering of usually-online musical patterns.

Lifecycle

Creating

Instantiate a threaded clock with:

>>> clock = supriya.Clock()

Starting and stopping

Start a threaded clock with:

>>> clock.start()

Stop a threaded clock with:

>>> clock.stop()

Scheduling

Let’s consider a simple callback function:

>>> def callback(context):
...     print(f"The current offset is: {context.current_moment.offset}")
... 

All clock callbacks need to accept at least a context argument, to which the clock will pass a ClockContext object.

Scheduling

We can schedule that procedure with:

>>> clock.schedule(callback)
0

Scheduling callbacks returns an event ID.

Canceling

We can use the event ID to cancel the callback if it hasn’t already occurred:

>>> clock.cancel(0)
CallbackCommand(event_id=0, event_type=<EventType.SCHEDULE: 1>, quantization=None, schedule_at=0.0, time_unit=<TimeUnit.BEATS: 0>, args=None, kwargs=None, procedure=<function callback at 0x7f2d4d6aae80>)

Cueing

We can also cue callbacks, scheduling them to occur at some quantized offset in the future, e.g. on the next beat or downbeat:

>>> clock.cue(callback)
1

Callbacks

Let’s consider a more complicated callback function:

>>> def callback(
...     context,
...     salutation,
...     delta=0.25,
...     max_repeats=0,
...     time_unit=supriya.clocks.TimeUnit.SECONDS,
... ):
...     print(f"{salutation}: {context.current_moment.offset}")
...     if max_repeats and context.event.invocations < max_repeats:
...         print("Re-scheduling")
...         return delta, time_unit
...     else:
...         print("Bailing!")
... 

Arguments

  • positional arguments

  • keyword arguments

Deltas

  • none

  • single float

  • pair of float and time unit or int

Contexts, moments, events

  • contexts

  • moments, current and expected

  • events, invocations

Musical time

Like callbacks, changes to tempo and time signature can be triggered “immediately”, scheduled at an absolute point in the future, or cued at some quantization relative the current time.

Tempo

Time signatures

Async clocks

Async clocks have identical APIs to threaded clocks with three differences:

  • their start() method is async,

  • their stop() method is async, and

  • scheduled callbacks may also be async.

Instantiate an async clock with:

>>> clock = supriya.AsyncClock()

Offline clocks

Debugging