Skip to content
This repository has been archived by the owner on Nov 1, 2021. It is now read-only.

UI Object Overview

janesconference edited this page Dec 8, 2011 · 19 revisions

UI Object

Tutorial

An UI object controls the relationship between one or more Graphical Elements and a Wrapper

With an UI object, you can associate a Graphical Element to a wrapper, set its depth (z-index) and visibility or chain n elements and set a filter between them. Let's see how.

I will use recursiontest as an example.

Creating an UI Object

First of all, we create a Wrapper object and an UI object, associating the first to the latter:

    var plugin_canvas = document.getElementById("plugin");
    var CWrapper = K2WRAPPER.createWrapper("CANVAS_WRAPPER",
                                               {canvas: plugin_canvas}
                                               );
    ui = new UI (plugin_canvas, CWrapper);

The wrapper is a CANVAS\_WRAPPER, so we know that every elements in our UI object will be rendered on an HTML5 canvas (we got the canvas element from the document via document.getElementById).

Notice also that we pass the same canvas object to the UI constructor. When you pass a DOM element to an UI object, the UI object adds some event listeners to the target. Current UI implementation adds listeners for mousedown, mouseup and mousemove events.

Adding elements to the UI

Now we can create an array of three Knob Elements to populate the UI object:

    var NKNOBS = 3

            for (var i=0; i < NKNOBS; i+=1) {

                knobArgs[i] = {
                    ID: "knob" + i,
                    top: 100,
                    left: i * 100,
                    imagesArray : loaderStatus.imagesArray,
                    sensivity : 5000,
                    onValueSet: function (slot, value, ID) {console.log (slot, value, ID); ui.refresh()}
                };

                knob[i] = new Knob(knobArgs[i]);

                ui.addElement(knob[i], {zIndex: 10});
                ui.setValue ({elementID: knobArgs[i].ID,
                              value: 0});
        }

knobArgs[i] is the variable we will pass to the Knob() constructor to build a new Knob element.

Its ID field is the unique ID of the element in the UI. No more than one element in an given UI instance can have the same ID (if you try to add two elements with the same ID, an exception is thrown).

top and left are the top, left position of the element in pixels, relative to the canvas origin.

imagesArray is the image array relative to the Knob element, ordered from the frame associated with the lowest value (0) to the frame associated with the highest (1).

sensitivity is how much the knob is sensitive to the mouse movement when it is clicked upon and dragged up and down.

onValueSet is very important: it is the function that is called back when the element state is changed (for example, when a Knob is dragged up or down with the mouse). It normally accepts three parameters: slot, value and ID. slot is always "knobvalue" in a Knob, since a Knob is a single-slot element (so to speak, it has only to remember the knob's state, a value from 0 to 1). Other elements could have more than one slot (think for example about a dialog that makes you draw a Beziér curve: it has to have at least three indipendent control points). value is the value the element assumed: if you drag the Knob down to its lowest point, for example, the value parameter received by the callback function will be 0. ID is the ID of the element in the UI that changed its state (this is particularly useful when we have a lot of elements and a single callback function to handle them all).

In this case, the callback doesn't do much: it just writes the parameters it received to the console and it calls refresh(). Calling refresh() redraws the elements in the UI: redrawing is not done automatically whenever an element's state is changed, so you probably want call refresh() at the end of a callback if you don't want your element to look "frozen" (there are situations in which you don't want to refresh as soon as your element's state is changed, but this simple example is not one of them).

addElement() adds the element to the UI. When your element is added to an UI, you can set its value and tell the UI how it must be drawn in respect to the other elements (not only: you can tell the UI if the element is active or inactive, or visible or invisible, for example).

addElement() has 2 parameters: one is the element previously created with its constructor and arguments, the other is an optional object that specifies extra-details about the element. Currently, this optional object supports only one parameter: zIndex.

Finally, setValue() is a function that makes you set the value (= the internal state) of an element associated to a given UI. setValue() has a number of options, but only elementID (the unique ID associated to the element) and value (the value to set) are mandatory. Here we set the initial value of the three knobs to 0.

Adding depth: zIndex

zIndex is a value >= 0 that represents the "depth" of a particular graphic element in the UI: the bigger zIndex is, the more that particular graphic elements is "on top" (towards the user). This affects the way UI refreshes the elements when their value gets changed.

In particular, when refresh() is called, all elements are re-drawn in their zIndex order, from the lowest to the highest. So, if two elements overlap, the one with the bigger zIndex value will be drawn top of the other (if you assign the same zIndex to two overlapping elements, there is no safe way to predict what element will be drawn on top of the other).

If an element does not define its zIndex, zIndex is set to 0 by default. If zIndex is defined, but it is not a number or it is < 0, an exception is thrown.

Since in this example the three knobs don't overlap, we can assign te same zIndex value (10) to each one of them. Since they're in top of nothing and nothing overlaps them, we could have omitted completely the {zIndex: 10} parameter.

Chaining elements in the UI

Now, we want to chain our three knobs: when one changes value, we want the others to change value as well, and we also want to tell each element how to change value in relation to the others' value.

In particular, here's what we want:

  • Everytime Knob 0 changes its knobvalue, Knob 1 has to change its knobvalue automatically. But we want Knob 1 to be the inverse of Knob 0: when Knob 0 increases, Knob 1 decreases, and vice versa. This has to be true whether the value-changing action (e.g. the user manipulating the knob) is started by Knob 0 or by Knob 1.
  • Similarly, everytime Knob 1 changes its knobvalue, Knob 2 has to change automatically. But we want Knob 2's range to be [0,0.5] instead of [0,1].
  • To close the loop, we want Knob 2 to change the value of Knob 0. This time, Knob 0's value must be the equal to Knob 2's.

Using UI.connectSlots() and the filter callback, we can accomplish this in four lines of code:

    // 0 -> 1
    ui.connectSlots("knob0", "knobvalue", "knob1", "knobvalue", {callback: function (value) {return 1-value;}});
    // 1 -> 0
    ui.connectSlots("knob1", "knobvalue", "knob0", "knobvalue", {callback: function (value) {return 1-value;}});
    // 1 -> 2
    ui.connectSlots("knob1", "knobvalue", "knob2", "knobvalue", {callback: function (value) {return value * 0.5;}});
    // 2 -> 0
    ui.connectSlots("knob2", "knobvalue", "knob0", "knobvalue", {callback: function (value) {return value;}});

UI.connectSlots()

UI.connectSlots takes five parameters. Its signature is:

    UI.connectSlots  = function (senderElement, senderSlot, receiverElement, receiverSlot, connectParameters)

senderElement and senderSlot are strings. They specify where the chain we want to add "starts".
Similarly, receiverElement and receiverSlot are strings, too. They specify where the chain we want to add "ends".

Calling UI.connectSlots means "When the value of slot senderSlot of the element senderElement changes, make the value of slot receiverSlot in the element receiverElement change accordingly".

The fifth parameter is an object containing optional parameters. The one that we need for now is the callback one.
callback is a callback function that acts as a filter when UI activates a particular connection. This callback simply takes the value of the senderSlot as its only parameter, and returns the value that should be put in the receiverSlot.

We used this callback to invert the knob value between Knob 0 -> Knob 1 and viceversa:

    {callback: function (value) {
        return 1-value;
        }
    };

But we can use it to make an hypothetical element with numeric values (a Knob or a Slider, for example) to automatically change a slot value of another element which handles strings (maybe a Label), or any other complex-as-you-like filter / translation you may need.

Infinite loops

Slot connections is a directed graph, and chaining slots can introduce loops. If Knob 0 changes Knob 1, Knob 1 changes Knob 2 and Knob 2 changes Knob 0 in turn, will this configuration generate an endless feedback between knobs?

UI "breaks" these infinite chains storing the history of recursively changed element:slot pair. When an element:slot is seen for the second time in the same chain, UI knows the loop is endless and breaks the chain. This way, if you modify the value of the first knob (either with setValue() or by using the mouse), the change is propagated to the second and the third knob, but not "fed back" again to the first knob, that is the change originator.

Clone this wiki locally