GUI (Graphical User Interface)

Boot the server to get going.

Windows

The class Window is used to create GUI windows. To create a window, we use the class method Window. There are several arguments for the window:

Creating Windows of Different Sizes

To change the size of a window, we need to use an instance of the class Rect. Rect specifies the a rectange on the screen where the window will appear. It can also be used to draw a rectange in a window. But for the moment, we will just use it specify the bounds of the window.

Drawing on a Window

Create a window to draw on.

Windows

Each window has a drawFunc method that can be used to place elements on the screen. The class Pen is used to draw on the screen.

The objects are drawn with the class Pen, which has numerous class methods that can draw all sorts of figures (termed paths) on the window.

When objects are placed on a window or any subcontainer in a window, coordinates are relative to the upper left corner of the window. Positive y-coordinates go down.

Rectangles

Rectangles are some of the simplest figures we can draw but also are useful as a way of expressing dimensions for other GUI objects. Rectangles take four arguments:

When placing rectangles in windows or other GUI objects, these coordinates are relative to the upper-left corner.

We can also fill objects using Pen as well.

Exercise 1: Bee Cells

Draw the following picture on a 400x400 Window (remember that is the default size) and call the Window "Bees". See below for the picture to emulate.

drawing

Note that the width here of the Pen stroke is 1.

Exercise 2: Shocked Emoji

Draw the shocked emoji. To do so, you will need to use the Pen method called .drawOval which can draw any circle or oval. Note that .drawOval takes a Rect which specifies the box with which the oval should fit inside.

See the picture below for the image to emulate.

drawing

Note that I set the width here to be 5.

Views

In SuperCollider, views are the GUI objects that are placed on windows. Some views (called containers) are capable of holding other views. Example: Each window has a view that is used to hold all of the GUI objects within itself.

When a view "A" holds a view "B", we say that "A" is the parent view and "B" is the child view. The view occupies a rectangular space of the window within which it draws itself to display some data or to indicate a mode of interaction between the user and the program.

Views receive keyboard and mouse events generated by the user and respond to them by controlling the behavior of the program. They also display information about the state of the program and the data on which it operates.

Slider

The class Slider is a view for a moveable slider that can be mapped to values for some other aspect of your program.

The .value method will get the value of the slider. However, we often want sliders, knobs, or other interactive devices to change some aspect of the sound in real-time. Therefore, it is better to have a function that is evaluated each time the slider is moved.

.action

Each view has an action function that can be set to be evaluated for each change to the view.

Now when we move the slider the values get posted as the slider changes.

Changing the Cutoff Frequency

Here's a practical example where we can change the cutoff frequency using a slider, similar to the Moog assignment.

Below is an example to show how to change the cutoff frequency. Notice that we need to map the linear values from 0 to 1 of the slider to appropriate values for the cutoff frequency.

Smoothing Values

If you listen carefully to the sawtooth wave when the slider is changed, there are discrete blips. Our ears are sensitive enough to hear a "jump" or "glitch" when the cutoff frequency is moved. Remember that the slider is producing discrete values.

To fix this, we can use Lag to smooth out the transitions. If the lag time is reasonably small we will not hear any delay.

Knobs

The class Knob is another view that can be used to control SuperCollider code.

Below we will use a knob to control the volume of the robot noises made from this SynthDef below.

Exercise: Saw GUI

Take the example from below that has arguments for sustain time and mix and create a GUI to control both of those parameters. Use a knob to control the sustain time and a slider to control the mix of the reverb. Below is a picture of what an example GUI might look like.

As a reminder, here is what that SynthDef sounded like.

Write your code here. You should do the following:

  1. Create a new window
  2. Create a knob whose action sets the sustain time. Map the value of the sustain time to the range 0 to 0.3.
  3. Create a slider whose action sets the mix of the reverb. No need to map the value because the mix should range between 0 and 1.
  4. Create the snyth to play it.
  5. Bring the window to the front to be displayed.

Write two lines to free the saw Synth and close the window.

Buttons

The class Button is useful in music GUIs usually to trigger events. Buttons in SuperCollider can have any number of states valued from 0 to one less than the number of states. The states of a button are an array of arrays where each inner array is a button state of text, text color, and background color.

Like the other views, we can set the action function to grab the value from each state.

StaticText

The class StaticText is used to display non-editable text.

The class Font can be used to change the Font for any view that displays text like StaticText.

You can get an array of all the available fonts using the method below.

Optional and Advanced Concepts

Below are optional things you may want to consider if building your own GUI, particularly for the final project.

Composite Views

The class Window contains a container view called TopView. Container views are views that are capable of holding other views. The TopView class is only for Window. If you want a parent view to organize several children views, use the class CompositeView.

Properties of CompositeView:

Advantages:

Composite views do not have any special/unique methods. They are simply a container/organizer for other views.

Notice how all the 2D sliders are moved simply by moving the composite view.

Decorators

We have seen how views can be arranged by hardcoding position and sizes within container views like Window’s container view TopView and CompositeView. Decorators automatically handle the positioning of views within a container view. There is one decorator in SuperCollider called FlowLayout. A decorator can be assigned to a View through the view’s .decorator method or by using the instance method .addFlowLayout for CompositeView or Window.

Here we let the decorator control the layout of the four spaces.

Layouts

Layouts offer a middle ground, providing the user more flexibility than a decorator but automating some of the process. Layouts distribute the amount of space given to the view on which they are installed among the children of that view.

Types of layouts:

Layouts can be added to a view by setting the layout with .layout method for the specific view. You should not combine decorators and layouts. Behavior is undefined.

The End!