community-apps/applications/voting/vote.js
armored-dragon 71ed32cb19
Handle ties.
Fix running without casted votes causing exception.
2024-09-18 06:18:30 -05:00

461 lines
No EOL
14 KiB
JavaScript

//
// vote.js
//
// App to simplify the tallying of votes for Community Meetings
//
// Created by Armored Dragon, 2024.
// Copyright 2024 Overte e.V.
//
// Distributed under the Apache License, Version 2.0.
// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html
/* global Script Tablet Messages MyAvatar AvatarList Uuid SoundCache*/
// TODO: Documentation
// FIXME: Make the candidate name have reserved value "-1", or change to uncommon name.
(() => {
"use strict";
let tablet;
let appButton;
let active = false;
const debug = false;
let poll = {id: '', title: '', description: '', host: '', question: '', options: [], canHostVote: false}; // The current poll
let pollStats = {iterations: 0, responses: {}, winnerSelected: false, winnerName: "", votesReceived: 0, votesCounted: 0 }; // Sent by host
let pollClientState = {hasVoted: false, isHost: false};
let activePolls = []; // All active polls.
let selectedPage = ""; // Selected page the vote screen is on. Used when the host closes the window.
let voteEngine = Script.require("./vote_engine.js");
const url = Script.resolvePath("./vote.qml");
const myUuid = generateUUID(MyAvatar.sessionUUID);
Messages.messageReceived.connect(receivedMessage);
Messages.subscribe('ga-polls');
tablet = Tablet.getTablet("com.highfidelity.interface.tablet.system");
appButton = tablet.addButton({
icon: Script.resolvePath("./img/icon_white.png"),
activeIcon: Script.resolvePath("./img/icon_black.png"),
text: "VOTE",
isActive: active,
});
// When script ends, remove itself from tablet
Script.scriptEnding.connect(function () {
console.log("Shutting Down");
tablet.removeButton(appButton);
deletePoll(true);
});
AvatarList.avatarSessionChangedEvent.connect(_resetNetworking);
// Overlay button toggle
appButton.clicked.connect(toolbarButtonClicked);
tablet.fromQml.connect(fromQML);
tablet.screenChanged.connect(onTabletScreenChanged);
function toolbarButtonClicked() {
if (active) tablet.gotoHomeScreen();
else tablet.loadQMLSource(url);
active = !active;
appButton.editProperties({
isActive: active,
});
}
function onTabletScreenChanged(type, newUrl) {
if (url == newUrl) {
active = true;
if (poll.id != '') {
return _findWhereWeNeedToBe();
}
// Request a list of active polls if we are not already in one
return getActivePolls();
}
else active = false;
appButton.editProperties({
isActive: active,
});
}
function _findWhereWeNeedToBe(){
// Vote has been completed
if (pollStats.winnerSelected) return _emitEvent({type: "switch_page", page: 'poll_results', poll: poll, pollStats: pollStats, isHost: pollClientState.isHost});
// Has voted already
if (pollClientState.hasVoted) return _emitEvent({type: "switch_page", page: 'poll_results', poll: poll, pollStats: pollStats, isHost: pollClientState.isHost});
// Has not voted yet, is not the host
if (!pollClientState.hasVoted && !pollClientState.isHost) return _emitEvent({type: "switch_page", page: 'poll_client_view', poll: poll, pollStats: pollStats, isHost: pollClientState.isHost});
// Has not voted yet, is the host
if (!pollClientState.hasVoted && (pollClientState.isHost && poll.canHostVote)) return _emitEvent({type: "switch_page", page: 'poll_client_view', poll: poll, pollStats: pollStats, isHost: pollClientState.isHost});
_emitEvent({type: "switch_page", page: selectedPage, poll: poll, pollStats: pollStats, isHost: pollClientState.isHost});
}
// Functions
// Get a list of active polls
function getActivePolls() {
// Sends a message to all hosts to send a list of their polls
Messages.sendMessage('ga-polls', JSON.stringify({type: "populate"}));
}
// Create a new poll for others to join
function createPoll(pollInformation) {
console.log("Creating a new poll");
// Check if we are already hosting a poll
if (poll.id != '') return;
// Set our active poll data
poll.id = generateUUID();
poll.host = myUuid;
poll.title = pollInformation.title;
poll.description = pollInformation.description;
console.log(`Active poll set as:\nid:${poll.id}\ntitle:${poll.title}\ndescription:${poll.description}`);
pollStats.responses = {}; // Clear any lingering responses
// Update Client State
pollClientState.isHost = true;
// Send message to all clients
Messages.sendMessage("ga-polls", JSON.stringify({type: "active_poll", poll: poll}));
console.log("Broadcasted poll to server");
// Subscribe to our own messages
Messages.subscribe(poll.id);
// Update the UI screen
_emitEvent({type: "create_poll"});
}
// Closes the poll and return to the main menu
function deletePoll(bypassPrompt){
// Check to see if we are hosting the poll
if (poll.host != myUuid) return; // We are not the host of this poll
// We are in a poll
if (poll.id == '') return;
// Confirm to user if they want to close the poll
if (!bypassPrompt) {
var answer = Window.confirm('Are you sure you want to close the poll?');
if (!answer) return;
}
console.log("Closing active poll");
// Submit the termination message to all clients
Messages.sendMessage("ga-polls", JSON.stringify({type: "close_poll", poll: {id: poll.id}}));
// Update the UI screen
_emitEvent({type: "close_poll", poll: {id: poll.id}, changePage: true});
// Clear our active poll data
_resetNetworking();
}
// Join an existing poll hosted by another user
function joinPoll(pollToJoin){
// TODO: Check if poll even exists
// Leave poll if already connected to one
leavePoll();
// Save the poll information
poll = pollToJoin;
// Subscribe to message mixer for poll information
Messages.subscribe(pollToJoin.id);
// Send join notice to server. This will cause the host to (re)emit the current poll to the server
Messages.sendMessage(pollToJoin.id, JSON.stringify({type: "join"}));
// Log the successful join
console.log(`Successfully joined ${poll.id}`);
_emitEvent({type: "switch_page", page: "poll_client_view"});
}
// Leave a poll hosted by another user
function leavePoll() {
if (!poll.id) return; // No poll to leave
let pollToLeave = poll.id;
// Clear poll
_resetNetworking();
console.log(`Successfully left ${pollToLeave}`);
}
// Cast a vote on a poll
function castVote(event) {
console.log(`Casting vote to ${poll.id}`);
// Check if poll is valid
if (poll == undefined || poll.id == '') return;
// Check if a winner was already chosen
if (pollStats.winnerSelected) return;
// Send vote to users in poll
Messages.sendMessage(poll.id, JSON.stringify({type: "vote", ballot: event.ballot, uuid: myUuid}));
pollClientState.hasVoted = true;
}
// Emit the prompt question and options to the clients
function emitPrompt(){
if (!pollClientState.isHost) return; // We are not the host of this poll
console.log(`Host: Emitting prompt: ${JSON.stringify({type: "poll_prompt", poll: poll, pollStats: pollStats}, null, 4)}`);
Messages.sendMessage(poll.id, JSON.stringify({type: "poll_prompt", poll: poll, pollStats: pollStats}, null, 4));
}
// Take the gathered responses and preform the election
// FIXME: Recursive function call
function preformElection(){
// Format the votes into a format that the election engine can understand
let votesFormatted = [];
const allVoters = Object.keys(pollStats.responses);
allVoters.forEach((voterId) => {
votesFormatted.push(pollStats.responses[voterId]);
});
const winner = voteEngine.preformVote(votesFormatted);
if (!winner) return;
const winnerName = winner?.tie ? winner.name.join(', ') : winner.name;
console.log(`Winner: ${winnerName}`);
// Update the stats
pollStats.winnerName = winnerName;
pollStats.winnerSelected = true;
pollStats.votesCounted = Object.keys(pollStats.responses).length;
pollStats.iterations = winner.iterations;
// Synchronize the winner with all clients
Messages.sendMessage(poll.id, JSON.stringify({type: "poll_winner", pollStats: pollStats}));
pollStats.responses = {};
return;
}
// Create a UUID or turn an existing UUID into a string
function generateUUID(existingUuid){
if (!existingUuid) existingUuid = Uuid.generate(); // Generate standard UUID
existingUuid = Uuid.toString(existingUuid); // Scripts way to turn it into a string
return existingUuid.replace(/[{}]/g, ''); // Remove '{' and '}' from UUID string >:(
}
function _debugDummyBallot() {
if (!debug) return; // Just incase...
let ballot = getRandomOrder(...poll.options);
const indexToRemove = Math.floor(Math.random() * ballot.length);
ballot.splice(indexToRemove, ballot.length - indexToRemove);
const responsesKeyName = Object.keys(pollStats.responses).length.toString();
pollStats.responses[responsesKeyName] = ballot;
function getRandomOrder(...words) {
for (let i = words.length - 1; i > 0; i--) {
const j = Math.floor(Math.random() * (i + 1));
[words[i], words[j]] = [words[j], words[i]];
}
return words;
}
}
// Reset application "networking" information to default values
function _resetNetworking(){
if (poll.id) Messages.unsubscribe(poll.id);
poll = {id: '', title: '', description: '', host: '', question: '', options: [], canHostVote: false};
pollStats = {iterations: 0, responses: {}, winnerSelected: false, winnerName: "", votesReceived: 0, votesCounted: 0 };
pollClientState = {isHost: false, hasVoted: false};
activePolls = [];
}
function _emitSound(type){
switch (type) {
case "new_prompt": {
let newPollSound = SoundCache.getSound(Script.resolvePath("./sound/new_vote.mp3"));
Audio.playSystemSound(newPollSound, {volume: 0.5});
break;
}
}
}
// Communication
function fromQML(event) {
if (!active) return;
console.log(`New QML event:\n${JSON.stringify(event, null, 4)}`);
switch (event.type) {
case "create_poll":
createPoll(event.poll);
break;
case "join_poll":
joinPoll(event.poll);
break;
case "cast_vote":
castVote(event);
break;
case "close_poll":
deletePoll();
break;
case "prompt":
poll.question = event.prompt.question;
poll.options = event.prompt.options.filter(String); // Clean empty options
poll.canHostVote = event.canHostVote;
pollStats = {iterations: 0, responses: {}, winnerSelected: false, winnerName: "", votesReceived: 0, votesCounted: 0 };
emitPrompt();
break;
case "run_election":
// Debug: Create a lot of fake ballots
if (debug) {
for (let i = 0; i < 26; ++i) {
_debugDummyBallot();
}
}
pollStats.iterations = 0;
preformElection();
break;
case "page_name":
selectedPage = event.page;
break;
case "leave":
leavePoll();
break;
}
}
/**
* Emit a packet to the HTML front end. Easy communication!
* @param {Object} packet - The Object packet to emit to the HTML
* @param {("create_poll"|"initial_settings")} packet.type - The type of packet it is
*/
function _emitEvent(packet = { type: "" }) {
tablet.sendToQml(packet);
}
function receivedMessage(channel, message){
// Not for us, ignore!
if (channel !== 'ga-polls' && channel !== poll.id) return;
message = JSON.parse(message);
switch (channel) {
case "ga-polls":
// Received a request to see our poll
if (message.type == "populate") {
// Send our poll information to the server if we are hosting it
if (poll.host == myUuid) {
Messages.sendMessage("ga-polls", JSON.stringify({type: "active_poll", poll: poll}));
}
}
// Received an active poll
if (message.type == "active_poll") {
if (poll.id != '') return; // We are in a poll, don't populate the list
if (activePolls.indexOf(message.poll.id) != -1) return; // We already have that poll in the list
_emitEvent({type: "new_poll", poll: message.poll});
activePolls.push(message.poll.id);
}
// Polls closed :)
if (message.type == "close_poll") {
var isOurPoll = poll.id == message.poll.id;
// Tell UI to close poll
_emitEvent({type: "close_poll", changePage: isOurPoll, poll: {id: message.poll.id}});
// Unregister self from poll
if (isOurPoll) leavePoll();
activePolls.splice(activePolls.indexOf(message.poll.id), 1); // Remove from active list
}
break;
case poll.id:
// Received poll information
if (message.type == "poll_prompt") {
console.log(`Received new prompt from host:\n${JSON.stringify(message.poll, null, 4)}`);
// TODO: This is still silly. Try using UUIDs per prompt and check if we are answering the same question by id?
// Don't recreate the prompt if we already have the matching question
if (message.poll.question == poll.question && !pollClientState.isHost) return;
if (pollClientState.isHost && !poll.canHostVote) return;
// update our poll information
poll = message.poll;
pollClientState.hasVoted = false;
pollStats.winnerSelected = false;
_emitSound("new_prompt");
_emitEvent({type: "poll_prompt", poll: poll, pollStats: pollStats});
}
if (message.type == "vote_count") {
_emitEvent({type: "received_vote", pollStats: message.pollStats});
}
// Winner was broadcasted
if (message.type == "poll_winner") {
pollStats = message.pollStats;
_emitEvent({type: "poll_winner", pollStats: message.pollStats});
}
// Received a sync packet
if (message.type == "sync"){
if (pollClientState.isHost) return; // Host doesn't need to sync to itself
console.log("Got sync packet!");
poll = message.poll;
pollStats = message.pollStats;
_findWhereWeNeedToBe();
}
// Host only -----
if (poll.host != myUuid) return;
// Received a ballot
if (message.type == "vote") {
// Check if we are the host
if (poll.host != myUuid) return;
// Record the ballot
pollStats.responses[message.uuid] = message.ballot;
// Emit a echo so the voter knows we have received it
// TODO:
// Broadcast the updated count to all clients
pollStats.votesReceived = Object.keys(pollStats.responses).length;
Messages.sendMessage(poll.id, JSON.stringify({type: "vote_count", pollStats: pollStats}));
}
// Received poll request
if (message.type == "join") {
if (!pollClientState.isHost) return;
// Send a sync packet
console.log("Sending sync packet.");
Messages.sendMessage(poll.id, JSON.stringify({type: "sync", poll: poll, pollStats: pollStats}));
emitPrompt();
}
}
}
})();