Newer
Older
/********************************************************************************
This file contains the variables and functions that form the select screens of
the game. The parsing functions for the opponent.xml file.
********************************************************************************/
/**********************************************************************
***** Opponent & Group Specification *****
**********************************************************************/
/**************************************************
* Stores meta information about groups.
**************************************************/

ReformCopyright
committed
function createNewGroup (title) {

ReformCopyright
committed
opponents:Array(4)};
/**********************************************************************
***** Select Screen UI Elements *****
**********************************************************************/
/* main select screen */
$selectTable = $("#select-table");
$selectBubbles = [$("#select-bubble-1"),
$("#select-bubble-2"),
$("#select-bubble-3"),
$("#select-bubble-4")];
$selectDialogues = [$("#select-dialogue-1"),
$("#select-dialogue-2"),
$("#select-dialogue-3"),
$("#select-dialogue-4")];
$selectAdvanceButtons = [$("#select-advance-button-1"),
$("#select-advance-button-2"),
$("#select-advance-button-3"),
$("#select-advance-button-4")];
$selectImages = [$("#select-image-1"),
$("#select-image-2"),
$("#select-image-3"),
$("#select-image-4")];
$selectLabels = [$("#select-name-label-1"),
$("#select-name-label-2"),
$("#select-name-label-3"),
$("#select-name-label-4")];
$selectButtons = [$("#select-slot-button-1"),
$("#select-slot-button-2"),
$("#select-slot-button-3"),
$("#select-slot-button-4")];
$selectMainButton = $("#main-select-button");
$selectRandomButtons = $("#select-random-button, #select-random-female-button, #select-random-male-button");

ReformCopyright
committed
$selectRandomTableButton = $("#select-random-group-button");
$selectSuggestions = [
$("#opponent-suggestions-1"),
$("#opponent-suggestions-2"),
$("#opponent-suggestions-3"),
$("#opponent-suggestions-4"),
];
$suggestionQuads = [
[$("#opponent-suggestion-1-1"), $("#opponent-suggestion-1-2"), $("#opponent-suggestion-1-3"), $("#opponent-suggestion-1-4")],
[$("#opponent-suggestion-2-1"), $("#opponent-suggestion-2-2"), $("#opponent-suggestion-2-3"), $("#opponent-suggestion-2-4")],
[$("#opponent-suggestion-3-1"), $("#opponent-suggestion-3-2"), $("#opponent-suggestion-3-3"), $("#opponent-suggestion-3-4")],
[$("#opponent-suggestion-4-1"), $("#opponent-suggestion-4-2"), $("#opponent-suggestion-4-3"), $("#opponent-suggestion-4-4")],
]
/* individual select screen */
$individualSelectTable = $("#individual-select-table");
$individualNameLabels = [$("#individual-name-label-1"), $("#individual-name-label-2"), $("#individual-name-label-3"), $("#individual-name-label-4")];
$individualPrefersLabels = [$("#individual-prefers-label-1"), $("#individual-prefers-label-2"), $("#individual-prefers-label-3"), $("#individual-prefers-label-4")];
$individualSexLabels = [$("#individual-sex-label-1"), $("#individual-sex-label-2"), $("#individual-sex-label-3"), $("#individual-sex-label-4")];
$individualHeightLabels = [$("#individual-height-label-1"), $("#individual-height-label-2"), $("#individual-height-label-3"), $("#individual-height-label-4")];
$individualSourceLabels = [$("#individual-source-label-1"), $("#individual-source-label-2"), $("#individual-source-label-3"), $("#individual-source-label-4")];
$individualWriterLabels = [$("#individual-writer-label-1"), $("#individual-writer-label-2"), $("#individual-writer-label-3"), $("#individual-writer-label-4")];
$individualArtistLabels = [$("#individual-artist-label-1"), $("#individual-artist-label-2"), $("#individual-artist-label-3"), $("#individual-artist-label-4")];
$individualCountBoxes = [$("#individual-counts-1"), $("#individual-counts-2"), $("#individual-counts-3"), $("#individual-counts-4")];
$individualLineCountLabels = [$("#individual-line-count-label-1"), $("#individual-line-count-label-2"), $("#individual-line-count-label-3"), $("#individual-line-count-label-4")];
$individualPoseCountLabels = [$("#individual-pose-count-label-1"), $("#individual-pose-count-label-2"), $("#individual-pose-count-label-3"), $("#individual-pose-count-label-4")];
$individualDescriptionLabels = [$("#individual-description-label-1"), $("#individual-description-label-2"), $("#individual-description-label-3"), $("#individual-description-label-4")];
$individualBadges = [$("#individual-badge-1"), $("#individual-badge-2"), $("#individual-badge-3"), $("#individual-badge-4")];
$individualStatuses = [$("#individual-status-1"), $("#individual-status-2"), $("#individual-status-3"), $("#individual-status-4")];
$individualLayers = [$("#individual-layer-1"), $("#individual-layer-2"), $("#individual-layer-3"), $("#individual-layer-4")];
$individualImages = [$("#individual-image-1"), $("#individual-image-2"), $("#individual-image-3"), $("#individual-image-4")];
$individualButtons = [$("#individual-button-1"), $("#individual-button-2"), $("#individual-button-3"), $("#individual-button-4")];
$individualPageIndicator = $("#individual-page-indicator");
$individualMaxPageIndicator = $("#individual-max-page-indicator");

ReformCopyright
committed
$individualCreditsButton = $('#individual-credits-button');
/* group select screen */
$groupSelectTable = $("#group-select-table");

ReformCopyright
committed
$groupSwitchTestingButton = $("#group-switch-testing-button");
$groupNameLabels = [$("#group-name-label-1"), $("#group-name-label-2"), $("#group-name-label-3"), $("#group-name-label-4")];
$groupPrefersLabels = [$("#group-prefers-label-1"), $("#group-prefers-label-2"), $("#group-prefers-label-3"), $("#group-prefers-label-4")];
$groupSexLabels = [$("#group-sex-label-1"), $("#group-sex-label-2"), $("#group-sex-label-3"), $("#group-sex-label-4")];
$groupHeightLabels = [$("#group-height-label-1"), $("#group-height-label-2"), $("#group-height-label-3"), $("#group-height-label-4")];
$groupSourceLabels = [$("#group-source-label-1"), $("#group-source-label-2"), $("#group-source-label-3"), $("#group-source-label-4")];
$groupWriterLabels = [$("#group-writer-label-1"), $("#group-writer-label-2"), $("#group-writer-label-3"), $("#group-writer-label-4")];
$groupArtistLabels = [$("#group-artist-label-1"), $("#group-artist-label-2"), $("#group-artist-label-3"), $("#group-artist-label-4")];
$groupCountBoxes = [$("#group-counts-1"), $("#group-counts-2"), $("#group-counts-3"), $("#group-counts-4")];
$groupLineCountLabels = [$("#group-line-count-label-1"), $("#group-line-count-label-2"), $("#group-line-count-label-3"), $("#group-line-count-label-4")];
$groupPoseCountLabels = [$("#group-pose-count-label-1"), $("#group-pose-count-label-2"), $("#group-pose-count-label-3"), $("#group-pose-count-label-4")];
$groupDescriptionLabels = [$("#group-description-label-1"), $("#group-description-label-2"), $("#group-description-label-3"), $("#group-description-label-4")];
$groupBadges = [$("#group-badge-1"), $("#group-badge-2"), $("#group-badge-3"), $("#group-badge-4")];
$groupStatuses = [$("#group-status-1"), $("#group-status-2"), $("#group-status-3"), $("#group-status-4")];
$groupLayers = [$("#group-layer-1"), $("#group-layer-2"), $("#group-layer-3"), $("#group-layer-4")];
$groupImages = [$("#group-image-1"), $("#group-image-2"), $("#group-image-3"), $("#group-image-4")];
$groupNameLabel = $("#group-name-label");
$groupButton = $("#group-button");
$groupPageIndicator = $("#group-page-indicator");
$groupMaxPageIndicator = $("#group-max-page-indicator");

ReformCopyright
committed
$groupCreditsButton = $('#group-credits-button');
$searchName = $("#search-name");
$searchSource = $("#search-source");
$searchTag = $("#search-tag");
$searchGenderOptions = [$("#search-gender-1"), $("#search-gender-2"), $("#search-gender-3")];
$sortingOptionsItems = $(".sort-dropdown-options li");
$groupSearchGroupName = $("#group-search-group-name");
$groupSearchName = $("#group-search-name");
$groupSearchSource = $("#group-search-source");
$groupSearchTag = $("#group-search-tag");
$groupSearchGenderOptions = [$("#group-search-gender-1"), $("#group-search-gender-2"), $("#group-search-gender-3"), $("#group-search-gender-4")];
/**********************************************************************
***** Select Screen Variables *****
**********************************************************************/
/* hidden variables */
var mainSelectHidden = false;
var singleSelectHidden = false;
var groupSelectHidden = false;
/* opponent listing file */
var listingFile = "opponents/listing.xml";
var metaFile = "meta.xml";
/* opponent information storage */
var loadedOpponents = [];
var selectableOpponents = loadedOpponents;
var loadedGroups = [[], []];
var selectableGroups = [loadedGroups[0], loadedGroups[1]];
var loadingOpponents = Array(4);
var chosenGroupGender = -1;
var sortingMode = "Featured";
var sortingOptionsMap = {
"Newest" : sortOpponentsByMultipleFields("-release"),
"Oldest" : sortOpponentsByMultipleFields("release"),
"Most Layers" : sortOpponentsByMultipleFields("-layers"),
"Fewest Layers" : sortOpponentsByMultipleFields("layers"),
"Name (A-Z)" : sortOpponentsByMultipleFields("first", "last"),
"Name (Z-A)" : sortOpponentsByMultipleFields("-first", "-last"),
ReformCopyright
committed
"Targeted most by selected" : sortOpponentsByMostTargeted(),
var individualCreditsShown = false;
var groupCreditsShown = false;
/* consistence variables */
var selectedSlot = 0;

ReformCopyright
committed
var shownIndividuals = Array(4);
var shownGroup = Array(4);
var shownSuggestions = [Array(4), Array(4), Array(4), Array(4)];
/* Status icon tooltips */
var TESTING_STATUS_TOOLTIP = "This opponent is currently in testing.";
var OFFLINE_STATUS_TOOLTIP = "This opponent has been retired from the official version of the game.";
var INCOMPLETE_STATUS_TOOLTIP = "This opponent is incomplete and currently not in development.";
/**********************************************************************
***** Start Up Functions *****
**********************************************************************/
/************************************************************
* Loads all of the content required to display the title
* screen.
************************************************************/
function loadSelectScreen () {
loadListingFile();
updateSelectionVisuals();
}
/************************************************************
* Loads and parses the main opponent listing file.
************************************************************/
function loadListingFile () {
/* clear the previous meta information */

ReformCopyright
committed
var opponentGroupMap = {};

ReformCopyright
committed
var opponentMap = {};
var onComplete = function(opp, index) {
if (opp) {
if (opp.id in opponentMap) {
loadedOpponents[opponentMap[opp.id]] = opp;
}
if (opp.id in opponentGroupMap) {
opponentGroupMap[opp.id].forEach(function(groupPos) {
groupPos.group.opponents[groupPos.idx] = opp;
});
}

ReformCopyright
committed
}
if (--outstandingLoads % 16 == 0) {
updateSelectableOpponents();
updateIndividualSelectScreen();

ReformCopyright
committed
updateSelectableGroups(0);
updateSelectableGroups(1);
updateGroupSelectScreen();
/* grab and parse the opponent listing file */
$.ajax({
type: "GET",
url: listingFile,
dataType: "text",
var $xml = $(xml);

ReformCopyright
committed
/* start by parsing the group descriptions to know which
* characters to load in addition to the main roster and any included statuses */

ReformCopyright
committed
$xml.find('groups>group').each(function () {
var opp1 = $(this).attr('opp1');
var opp2 = $(this).attr('opp2');
var opp3 = $(this).attr('opp3');
var opp4 = $(this).attr('opp4');

ReformCopyright
committed
var ids = [opp1, opp2, opp3, opp4];

ReformCopyright
committed
var newGroup = createNewGroup(title);
ids.forEach(function(id, idx) {
if (!(id in opponentGroupMap)) {
opponentGroupMap[id] = [];
}
opponentGroupMap[id].push({ group: newGroup, idx: idx });
});
loadedGroups[$(this).attr('testing') ? 1 : 0].push(newGroup);

ReformCopyright
committed
/* now actually load the characters */
var oppDefaultIndex = 0; // keep track of an opponent's default placement
$individualListings = $xml.find('individuals');
$individualListings.find('opponent').each(function () {
var oppStatus = $(this).attr('status');
var id = $(this).text();
var releaseNumber = $(this).attr('release');

ReformCopyright
committed
var doInclude = (oppStatus === undefined || includedOpponentStatuses[oppStatus]);
if (doInclude || id in opponentGroupMap) {
outstandingLoads++;
if (doInclude) {
opponentMap[id] = oppDefaultIndex++;
}
loadOpponentMeta(id, oppStatus, releaseNumber, onComplete);

ReformCopyright
committed
}
});
}
});
}
/************************************************************
* Loads and parses the meta XML file of an opponent.
************************************************************/
function loadOpponentMeta (id, status, releaseNumber, onComplete) {
/* grab and parse the opponent meta file */

ReformCopyright
committed
console.log("Loading metadata for \""+id+"\"");
url: 'opponents/' + id + '/' + metaFile,
var $xml = $(xml);
var opponent = new Opponent(id, $xml, status, releaseNumber);

ReformCopyright
committed
onComplete(opponent);
},
error: function(err) {
console.log("Failed reading \""+id+"\"");
onComplete();
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
function updateStatusIcon(elem, status) {
var icon_img = 'img/testing-badge.png';
var tooltip = TESTING_STATUS_TOOLTIP;
if(!status) {
elem.removeAttr('title').removeAttr('data-original-title').hide();
return;
}
if (status === 'offline') {
icon_img = 'img/offline-badge.png';
tooltip = OFFLINE_STATUS_TOOLTIP;
} else if (status === 'incomplete') {
icon_img = 'img/incomplete-badge.png';
tooltip = INCOMPLETE_STATUS_TOOLTIP;
}
elem.attr({
'src': icon_img,
'title': tooltip,
'data-original-title': tooltip,
}).show().tooltip({
'placement': 'left'
});
}
/************************************************************
* Loads opponents onto the individual select screen based
* on the currently selected page.
************************************************************/
function updateIndividualSelectScreen () {
/* safety wrap around */
if (individualPage < 0) {
/* wrap to last page */
individualPage = Math.ceil(selectableOpponents.length/4)-1;
}
$individualPageIndicator.val(individualPage+1);
/* keep track of how many opponents were on this screen */
var empty = 0;
/* create and load all of the individual opponents */
for (var i = individualPage*4; i < (individualPage+1)*4; i++) {
var index = i - individualPage*4;
if (i < selectableOpponents.length) {
shownIndividuals[index] = selectableOpponents[i];
$individualNameLabels[index].html(selectableOpponents[i].first + " " + selectableOpponents[i].last);
$individualPrefersLabels[index].html(selectableOpponents[i].label);
$individualSexLabels[index].html(selectableOpponents[i].gender);
$individualSourceLabels[index].html(selectableOpponents[i].source);
$individualWriterLabels[index].html(wordWrapHtml(selectableOpponents[i].writer));
$individualArtistLabels[index].html(wordWrapHtml(selectableOpponents[i].artist));
$individualDescriptionLabels[index].html(selectableOpponents[i].description);
if (EPILOGUE_BADGES_ENABLED && selectableOpponents[i].ending) {
$individualBadges[index].show();
}
else {
$individualBadges[index].hide();
}
updateStatusIcon($individualStatuses[index], selectableOpponents[i].status);

ReformCopyright
committed
$individualLayers[index].attr("src", "img/layers" + selectableOpponents[i].layers + ".png");
$individualImages[index].attr('src', selectableOpponents[i].folder + selectableOpponents[i].image);

ReformCopyright
committed
$individualImages[index].css('height', selectableOpponents[i].scale + '%');

ReformCopyright
committed
$individualImages[index].show();
if (selectableOpponents[i].enabled == "true") {
$individualButtons[index].html('Select Opponent');
$individualButtons[index].attr('disabled', false);
} else {
$individualButtons[index].html('Coming Soon');
$individualButtons[index].attr('disabled', true);
}
} else {

ReformCopyright
committed
delete shownIndividuals[index];
$individualNameLabels[index].html("");
$individualPrefersLabels[index].html("");
$individualSexLabels[index].html("");
$individualSourceLabels[index].html("");
$individualWriterLabels[index].html("");
$individualArtistLabels[index].html("");
$individualCountBoxes[index].css("visibility", "hidden");
$individualDescriptionLabels[index].html("");
$individualStatuses[index].hide();

ReformCopyright
committed
$individualImages[index].hide();
$individualButtons[index].attr('disabled', true);
/* reload if the page is empty */
if (empty == 4 && individualPage != 0) {
individualPage = 0;
updateIndividualSelectScreen();
}
}
/************************************************************
* Loads opponents onto the group select screen based on the
* currently selected page.
************************************************************/
function updateGroupSelectScreen () {
/* safety wrap around */
groupPage[groupSelectScreen] = (selectableGroups[groupSelectScreen].length)-1;
} else if (groupPage[groupSelectScreen] > selectableGroups[groupSelectScreen].length-1) {
$groupPageIndicator.val(groupPage[groupSelectScreen]+1);
$groupMaxPageIndicator.html("of "+selectableGroups[groupSelectScreen].length);
/* create and load all of the individual opponents */
$groupButton.attr('disabled', false);
var opponent = selectableGroups[groupSelectScreen].length > 0 ?
selectableGroups[groupSelectScreen][groupPage[groupSelectScreen]].opponents[i] :
undefined;
if (opponent && typeof opponent == "object") {
$groupNameLabels[i].html(opponent.first + " " + opponent.last);
$groupPrefersLabels[i].html(opponent.label);
$groupSexLabels[i].html(opponent.gender);
$groupSourceLabels[i].html(opponent.source);
$groupWriterLabels[i].html(wordWrapHtml(opponent.writer));
$groupArtistLabels[i].html(wordWrapHtml(opponent.artist));
$groupDescriptionLabels[i].html(opponent.description);
if (EPILOGUE_BADGES_ENABLED && opponent.ending) {
$groupBadges[i].show();
}
else {
$groupBadges[i].hide();
}
updateStatusIcon($groupStatuses[i], opponent.status);
$groupLayers[i].attr("src", "img/layers" + opponent.layers + ".png");
$groupImages[i].attr('src', opponent.folder + opponent.image);

ReformCopyright
committed
$groupImages[i].css('height', opponent.scale + '%');

ReformCopyright
committed
$groupImages[i].show();

ReformCopyright
committed
delete shownGroup[i];
$groupNameLabels[i].html("");
$groupPrefersLabels[i].html("");
$groupSexLabels[i].html("");
$groupSourceLabels[i].html("");
$groupWriterLabels[i].html("");
$groupArtistLabels[i].html("");
$groupDescriptionLabels[i].html("");
$groupStatuses[i].hide();

ReformCopyright
committed
$groupImages[i].hide();
$groupButton.attr('disabled', true);
if (selectableGroups[groupSelectScreen].length == 0) {
$groupNameLabel.html("(No matches)");
} else {
$groupNameLabel.html(selectableGroups[groupSelectScreen][groupPage[groupSelectScreen]].title);
}
/* Sets the suggested opponent to be displayed in a given slot and quadrant.
* Arguments:
* - opponent: the opponent object to display
* - slot: the selection slot to load into
* - quad: the quadrant of said selection slot to load into
*/
function updateSuggestionQuad(slot, quad, opponent) {
var img_elem = $suggestionQuads[slot][quad].children('.opponent-suggestion-image');
var label_elem = $suggestionQuads[slot][quad].children('.opponent-suggestion-label');
var tooltip = null;
if (opponent.status === 'testing') {
tooltip = TESTING_STATUS_TOOLTIP;
} else if (opponent.status === 'offline') {
tooltip = OFFLINE_STATUS_TOOLTIP;
} else if (opponent.status === 'incomplete') {
tooltip = INCOMPLETE_STATUS_TOOLTIP;
}
shownSuggestions[slot][quad] = opponent.id;
img_elem.attr({
'title': tooltip,
'data-original-title': tooltip,
'src': opponent.folder+opponent.image
}).tooltip();
label_elem.text(opponent.label);
}
/* Sets the given selection screen slot to display 4 opponents from an array.
* Arguments:
* - slot: the main select screen slot to update (zero-indexed)
* - suggestionsArray: the array to draw suggestions from
* - startIndex: the index into suggestionsArray to begin drawing suggestions from
*/
function updateSuggestions(slot, suggestionsArray, startIndex) {
for(var i=0;i<4;i++) {
if (suggestionsArray[startIndex+i]) {
updateSuggestionQuad(slot, i, suggestionsArray[startIndex+i]);
}
}
}
/**********************************************************************
***** Interaction Functions *****
**********************************************************************/
Chase Southwood
committed
/************************************************************
* Filters the list of selectable opponents based on those
* already selected and performs search and sort logic.
************************************************************/
function updateSelectableOpponents(autoclear) {
Chase Southwood
committed
var name = $searchName.val().toLowerCase();
var source = $searchSource.val().toLowerCase();
var tag = $searchTag.val().toLowerCase();
// Array.prototype.filter automatically skips empty slots
selectableOpponents = loadedOpponents.filter(function(opp) {
Chase Southwood
committed
// filter by name

ReformCopyright
committed
if (name
&& opp.label.toLowerCase().indexOf(name) < 0
&& opp.first.toLowerCase().indexOf(name) < 0
&& opp.last.toLowerCase().indexOf(name) < 0) {
return false;
Chase Southwood
committed
}
// filter by source
if (source && opp.source.toLowerCase().indexOf(source) < 0) {
return false;
Chase Southwood
committed
}
// filter by tag
if (!opp.tags || !opp.tags.some(function(t) {
return t.toLowerCase().indexOf(tag) >= 0;
})) {
return false;
Chase Southwood
committed
// filter by gender
if ((chosenGender == 2 && opp.gender !== eGender.MALE)
|| (chosenGender == 3 && opp.gender !== eGender.FEMALE)) {
return false;
Chase Southwood
committed
}
/* hide selected opponents */
if (players.some(function(p) { return p && p.id == opp.id; })) {
return false;
Chase Southwood
committed
}
if (loadingOpponents.some(function(p) { return p && p === opp.id; })) {
return false;
}
return true;
});
Chase Southwood
committed
// If a unique match was made, automatically clear the search so
// another opponent can be found more quickly.
if (autoclear && (name != null || source != null) && selectableOpponents.length == 0) {
clearSearch();
return;
}
Chase Southwood
committed
/* sort opponents */
// Since selectableOpponents is always reloaded here with featured order,
// check if a different sorting mode is selected, and if yes, sort it.
if (sortingOptionsMap.hasOwnProperty(sortingMode)) {
selectableOpponents.sort(sortingOptionsMap[sortingMode]);
}
/* update max page indicator */
$individualMaxPageIndicator.html("of "+Math.ceil(selectableOpponents.length/4));
}

FarawayVision
committed
/************************************************************
* The player clicked on a suggested character button.
************************************************************/
function suggestionSelected(slot, quad) {
var selectedID = shownSuggestions[slot-1][quad-1];
if(!selectedID) {
/* This shouldn't happen. */
console.error("Could not find suggested opponent ID for slot " + slot + " and quad " + quad);
return;
}

FarawayVision
committed
/* Find the character they selected. */
for (var i=0; i<loadedOpponents.length; i++) {
if (loadedOpponents[i].id === selectedID) {

FarawayVision
committed
players[slot] = null;
loadingOpponents[slot-1] = selectedID;

FarawayVision
committed
updateSelectionVisuals();
loadedOpponents[i].loadBehaviour(playerLoadedCallback, slot);

FarawayVision
committed
return;
}
}
/* This shouldn't happen, either. */
console.error("Could not find opponent with ID " + selectedID);

FarawayVision
committed
}
/************************************************************
* The player clicked on an opponent slot.
************************************************************/
function selectOpponentSlot (slot) {

ReformCopyright
committed
if (!(slot in players)) {
/* add a new opponent */
selectedSlot = slot;
/* Make sure the user doesn't have target-count sorting set if
* the amount of loaded opponents drops to 0. */
if (sortingMode === "Targeted most by selected") {
var player_count = countLoadedOpponents();
if (player_count <= 1) {
setSortingMode("Featured");
}
}
/* update the list of selectable opponents based on those that are already selected, search, and sort options */
updateSelectableOpponents(true);
/* reload selection screen */
updateIndividualSelectScreen();
updateIndividualCountStats();
/* switch screens */
screenTransition($selectScreen, $individualSelectScreen);
} else {
/* remove the opponent that's there */

ReformCopyright
committed
$selectImages[slot-1].off('load');

ReformCopyright
committed
delete players[slot];
updateSelectionVisuals();
}
}
/************************************************************

ReformCopyright
committed
* The player clicked on the Preset Tables or Testing Tables button.
************************************************************/

ReformCopyright
committed
function clickedSelectGroupButton (screen) {
switchSelectGroupScreen(screen)
/* switch screens */
screenTransition($selectScreen, $groupSelectScreen);
}
/************************************************************

ReformCopyright
committed
* The player clicked on the Preset Tables or Testing Tables
* button from within the table select screen.
************************************************************/
function switchSelectGroupScreen (screen) {
if (screen !== undefined) {
groupSelectScreen = screen;
} else {
groupSelectScreen = 1 - groupSelectScreen;
}
if (groupSelectScreen == 1) {
$groupSwitchTestingButton.html("Preset Tables");
} else {
$groupSwitchTestingButton.html("Testing Tables");
}
updateSelectableGroups(groupSelectScreen);

ReformCopyright
committed
updateGroupSelectScreen();
/************************************************************
* Filters the list of selectable opponents based on those
* already selected and performs search and sort logic.
************************************************************/
function updateSelectableGroups(screen) {
var groupname = $groupSearchGroupName.val().toLowerCase();
var name = $groupSearchName.val().toLowerCase();
var source = $groupSearchSource.val().toLowerCase();
var tag = $groupSearchTag.val().toLowerCase();
// reset filters
selectableGroups[screen] = loadedGroups[screen].filter(function(group) {

ReformCopyright
committed
if (!group.opponents.every(function(opp) { return opp; })) return false;

ReformCopyright
committed
if (groupname && group.title.toLowerCase().indexOf(groupname) < 0) return false;
if (name && !group.opponents.some(function(opp) {
return opp.label.toLowerCase().indexOf(name) >= 0
|| opp.first.toLowerCase().indexOf(name) >= 0
|| opp.last.toLowerCase().indexOf(name) >= 0;
})) return false;
if (source && !group.opponents.some(function(opp) {
return opp.source.toLowerCase().indexOf(source) >= 0;
})) return false;
if ((chosenGroupGender == 2 || chosenGroupGender == 3)
&& !group.opponents.every(function(opp) {
return opp.gender == (chosenGroupGender == 2 ? eGender.MALE : eGender.FEMALE);
})) return false;
if (chosenGroupGender == 4
&& !(group.opponents.some(function(opp) { return opp.gender == eGender.MALE; })
&& group.opponents.some(function(opp) { return opp.gender == eGender.FEMALE; })))
return false;
return true;
})
/************************************************************
* The player clicked on the select random group slot.
************************************************************/
function clickedRandomGroupButton () {
selectedSlot = 1;
for (var i = 1; i < players.length; i++) {
players[i] = null;
}
/* get a random number for the group listings */
var randomGroupNumber = getRandomNumber(0, loadedGroups[0].length);
var chosenGroup = loadedGroups[0][randomGroupNumber];
console.log(chosenGroup.title);
for (var i = 0; i < chosenGroup.opponents.length; i++) {
/* Don't try to load empty character slots */
if (!chosenGroup.opponents[i] || typeof chosenGroup.opponents[i] !== 'object') {
continue;
}
/* character exists? Okay, load it */
chosenGroup.opponents[i].loadBehaviour(playerLoadedCallback, i+1);

ReformCopyright
committed
updateSelectionVisuals();
}
/************************************************************
* The player clicked on the all random button.
************************************************************/
function clickedRandomFillButton (predicate) {
/* compose a copy of the loaded opponents list */

ReformCopyright
committed
var loadedOpponentsCopy = loadedOpponents.filter(function(opp) {
// Filter out already selected characters
return (!players.some(function(p) { return p && p.id == opp.id; })

ReformCopyright
committed
&& (!predicate || predicate(opp)));
});
/* select random opponents */
for (var i = 1; i < players.length; i++) {
/* if slot is empty */

ReformCopyright
committed
if (!(i in players)) {
players[i] = null;
/* select random opponent */
var randomOpponent = getRandomNumber(0, loadedOpponentsCopy.length);
loadedOpponentsCopy[randomOpponent].loadBehaviour(playerLoadedCallback, i);
/* remove random opponent from copy list */
loadedOpponentsCopy.splice(randomOpponent, 1);
}
}

ReformCopyright
committed
updateSelectionVisuals();
/************************************************************
* The player clicked on the remove all button.
************************************************************/

ReformCopyright
committed
delete players[i];
/************************************************************
* The player clicked on a change stats card button on the
* individual select screen.
************************************************************/
function changeIndividualStats (target) {
for (var i = 1; i < 5; i++) {
for (var j = 1; j < 4; j++) {
if (j != target) {
$('#individual-stats-page-'+i+'-'+j).hide();
}
else {
$('#individual-stats-page-'+i+'-'+j).show();
}
}
}
individualCreditsShown = (target == 2); // true when Credits button is clicked
}
/************************************************************
* The player clicked the select opponent button on the
* individual select screen.
************************************************************/
function selectIndividualOpponent (slot) {
/* move the stored player into the selected slot and update visuals */

ReformCopyright
committed
players[selectedSlot] = null;
loadingOpponents[selectedSlot-1] = shownIndividuals[slot-1].id;

ReformCopyright
committed
updateSelectionVisuals();
shownIndividuals[slot-1].loadBehaviour(playerLoadedCallback, selectedSlot);

ReformCopyright
committed
/* switch screens */
screenTransition($individualSelectScreen, $selectScreen);
}
/************************************************************
* This callback is called after an opponent's behaviour file is loaded,
* in all selection cases: indiv. select, group select, random selection, etc.
************************************************************/
function playerLoadedCallback (playerObject, slot) {
console.log(slot+": "+playerObject);
players[slot] = playerObject;
delete loadingOpponents[slot-1];
updateBehaviour(slot, SELECTED);
updateSelectionVisuals();
}
/************************************************************
* The player is changing the page on the individual screen.
************************************************************/
function changeIndividualPage (skip, page) {
console.log("resigtered");
if (skip) {
if (page == -1) {
/* go to first page */
individualPage = 0;
} else if (page == 1) {
/* go to last page */
individualPage = Math.ceil(selectableOpponents.length/4)-1;
} else {
/* go to selected page */
individualPage = Number($individualPageIndicator.val()) - 1;
}
} else {
individualPage += page;
}
updateIndividualSelectScreen();
updateIndividualCountStats();
}
/************************************************************
* The player clicked on a change stats card button on the
* group select screen.
************************************************************/
function changeGroupStats (target) {
for (var i = 1; i < 5; i++) {
for (var j = 1; j < 4; j++) {
if (j != target) {
$('#group-stats-page-'+i+'-'+j).hide();
}
else {
$('#group-stats-page-'+i+'-'+j).show();
}
}
}
groupCreditsShown = (target == 2); // true when Credits button is clicked
}
/************************************************************
* The player clicked the select opponent button on the
* group select screen.
************************************************************/
function selectGroup () {
/* clear the selection screen */
for (var i = 1; i < 5; i++) {
players[i] = null;
}
updateSelectionVisuals();
/* load the group members */
for (var i = 0; i < 4; i++) {
var member = selectableGroups[groupSelectScreen][groupPage[groupSelectScreen]].opponents[i];
if (member) {
member.loadBehaviour(playerLoadedCallback, i+1);
Joseph Kantel
committed
}

ReformCopyright
committed
/* switch screens */
screenTransition($groupSelectScreen, $selectScreen);
}
/************************************************************
* The player is changing the page on the group screen.
************************************************************/
function changeGroupPage (skip, page) {
if (skip) {
if (page == -1) {
/* go to first page */
} else if (page == 1) {
/* go to last page */
groupPage[groupSelectScreen] = selectableGroups[groupSelectScreen].length-1;
} else {
/* go to selected page */
groupPage[groupSelectScreen] = Number($groupPageIndicator.val()) - 1;
updateGroupCountStats();
}
/************************************************************
* The player clicked on the back button on the individual or
* group select screen.
************************************************************/
function backToSelect () {
/* switch screens */
screenTransition($individualSelectScreen, $selectScreen);
screenTransition($groupSelectScreen, $selectScreen);
}
/************************************************************
* The player clicked on the start game button on the main
* select screen.
************************************************************/
function advanceSelectScreen () {

FarawayVision
committed
console.log("Starting game...");
gameID = generateRandomID();
'date': (new Date()).toISOString(),
'userAgent': navigator.userAgent,
'table': {}
};
for (let i=1;i<5;i++) {
if (players[i]) {
usage_tracking_report.table[i] = players[i].id;
}
}
$.ajax({
url: USAGE_TRACKING_ENDPOINT,
method: 'POST',
data: JSON.stringify(usage_tracking_report),
contentType: 'application/json',
error: function (jqXHR, status, err) {
console.error("Could not send usage tracking report - error "+status+": "+err);
},
});
}
advanceToNextScreen($selectScreen);
}
/************************************************************
* The player clicked on the back button on the main select
* screen.
************************************************************/
function backSelectScreen () {
screenTransition($selectScreen, $titleScreen);
}
/**********************************************************************
***** Display Functions *****