WebController - Enter Frame

The enter frame section contains mostly methods that are triggered each frame or in a similar way.

onNewFrame

onNewFrame(frameSource)

The onNewFrame method performs various tasks that need constant updating. We call it every frame for the Genvid client we created.

  1. First we update the overlays to adapt to the composition of the video stream through updateOverlays(compositionData). There, we check the received composition data for the number of sources and composition data layout types. It enables us to update the overlay accordingly for multi-source(ChromaKey/PiP) and single source streaming.

  2. We then call updateStreamsInfoFromSession(session) to extract the game data streams out of the given frame. The extracted data, devided in strams such as Positions, Colors, and Camera is used to update the web page overlay.

  3. Finally, we query the GenvidClient to update the UI with such information as the latency, the delay offset, volume visibility, etc.

onStreamsReceived

onStreamsReceived(dataStreams)

The onStreamsReceived method uses loops to get the game data. We check the game data for its format. If the format is UTF-8, we check the stream id to make sure the character string we received is the copyright data and, if so, we display it in the console. We expect the rest of the data streams to be formated in JSON. In such a case, we look for the “Names” stream. Once found, we can pass the names data as a parameter to initPlayerTable(cubeNames) so that the info pannel can be constructed. Then, we proceed to check the streams data for annotations. If we find a JSON formatted annotation with a “Colors” id, we extract the data to update the overlay about which cubes changed color.

initPlayerTable

initPlayerTable(cubeData)

The initPlayerTable creates a table under the stream window for each object.

  1. We start with a loop that repeats for each object’s information, based on received game data.

  2. We clone a node that we later append to an HTML tag.

  3. We then add the event listeners for the table, cheer, and reset click functionality.

  4. Finally, we perform a loop for each color available and add a click-event listener.

        // Method used to display the appropriate number of players with their proper buttons
        initPlayerTable(cubeNames) {
            const cubePanel = document.getElementById("cube_panel_prototype");
    
            // The prototype panel gets removed after init.
            if (cubePanel === null) {
                // We already have real pannels. No need for more.
                return;
            }
    
            this.cubeNames = cubeNames;
    
            for (const idx in cubeNames) {
                const name = this.cubeNames[idx];
                let cubePanelClone = cubePanel.cloneNode(true);
                cubePanelClone.id = idx;
                cubePanelClone.getElementsByClassName("cube_name")[0].innerText = name;
    
                let cheerButton = cubePanelClone.querySelector(".cheer");
                cheerButton.addEventListener("click", () => this.onCheer(name), false);
    
                let cubeDiv = cubePanelClone.querySelector(".cube");
                cubeDiv.addEventListener("click", () => this.selectCube(parseInt(idx)), false);
                this.cubePanelDiv.push(cubeDiv); // will stop triggering initPlayerTable from onNewFrame() indefinitely
    
                let resetButton = cubePanelClone.querySelector(".reset");
                resetButton.addEventListener("click", () => this.onReset(name), false);
    
                for (let colorSelect of this.tableColor) {
                    let colorButton = cubePanelClone.querySelector("." + colorSelect[0]);
                    colorButton.addEventListener("click", () => this.onColorChange(name, colorSelect[1]), false);
                }
                document.querySelector(".gameControlsDiv").append(cubePanelClone);
    
                const panel = {
                    panel: cubePanelClone,
                    x: cubePanelClone.getElementsByClassName("position_x")[0],
                    y: cubePanelClone.getElementsByClassName("position_y")[0],
                    z: cubePanelClone.getElementsByClassName("position_z")[0],
                    popularity: cubePanelClone.getElementsByClassName("cheer_value")[0],
                };
                this.panels.push(panel);
            }
    
            // We don't need the prototype panel anymore. We now have real panels.
            cubePanel.remove();
        }
    

onNotificationsReceived

onNotificationsReceived(message)

The onNotificationsReceived method checks if any notification’s ID is Popularity. When it is, we convert the JSON data into JavaScript and update the overlay. We call this method when the website receives a notification.

    // Upon receiving a notification, gets the notification content
    onNotificationsReceived(message) {
        for (let notification of message.notifications) {
            switch (notification.id) {
                case "RESET":
                    try {
                        let obj = JSON.parse(notification.data);
                        this.showNotification(`${notification.id} - Velocity, position, and orientation were reset for ${obj.cube}.`);
                    } catch (e) {
                        genvid.log("Error when parsing JSON:", notification.data);
                    }
                    break;

                case "SPEED":
                    try {
                        let obj = JSON.parse(notification.data);
                        this.showNotification(`${notification.id} - Speed ${obj.speed.toPrecision(3)} applied to ${obj.cube}.`);
                    } catch (e) {
                        genvid.log("Error when parsing JSON:", notification.data);
                    }
                    break;

                case "DIRECTION":
                    try {
                        let obj = JSON.parse(notification.data);
                        let direction = "|X:" + obj.direction[0] + " Y:" + obj.direction[1] + " Z:" + obj.direction[2] + "|";
                        this.showNotification(`${notification.id} - Direction ${direction} applied to ${obj.cube}.`);
                    } catch (e) {
                        genvid.log("Error when parsing JSON:", notification.data);
                    }
                    break;

                case "POPULARITY":
                    try {
                        let obj = JSON.parse(notification.data);
                        this.popularities = obj.popularity;
                    } catch (e) {
                        genvid.log("Error when parsing JSON:", notification.data);
                    }
                    break;

                case "events.summaries.cheer":
                    try {
                        let obj = JSON.parse(notification.data);
                        genvid.log(JSON.stringify(obj));
                    } catch (e) {
                        genvid.log("Error when parsing JSON", notification.data);
                    }
                    break;

                default:
                    genvid.warn("Unrecognized notification: " + notification.id);
            }
        }
    }

render

render

The render function uses the game data provided to update the scene. First, we check if we received the cube names. If so, we create the corresponding tags and circle sprites. We only do so once. We then use received camera data, cube positions data, and Three.js functionality to update the sprite and tag position in 2D space to visually match the postions of the cubes in the rendered frame. The position info is also used to update the player pannels. Next, we call the Three.js renderer to render the updated scene. Finally, we call requestAnimationFrame with render as the argument to keep updating the scene.