How does SuperDog work?



//A custom array of events to populate data within a table
let events = [{
event: "ComicCon",
city: "New York",
state: "New York",
attendance: 240000,
date: "06/01/2017"
},
{
event: "ComicCon",
city: "New York",
state: "New York",
attendance: 250000,
date: "06/01/2018"
},
{
event: "ComicCon",
city: "New York",
state: "New York",
attendance: 257000,
date: "06/01/2019"
},
{
event: "ComicCon",
city: "San Diego",
state: "California",
attendance: 130000,
date: "06/01/2017"
},
{
event: "ComicCon",
city: "San Diego",
state: "California",
attendance: 140000,
date: "06/01/2018"
},
{
event: "ComicCon",
city: "San Diego",
state: "California",
attendance: 150000,
date: "06/01/2019"
},
{
event: "HeroesCon",
city: "Charlotte",
state: "North Carolina",
attendance: 40000,
date: "06/01/2017"
},
{
event: "HeroesCon",
city: "Charlotte",
state: "North Carolina",
attendance: 45000,
date: "06/01/2018"
},
{
event: "HeroesCon",
city: "Charlotte",
state: "North Carolina",
attendance: 50000,
date: "06/01/2019"
}
]

//build a dropdown of distinct cities
function buildDropdown() {

let eventDD = document.getElementById("eventDropdown");
//clears the dropdown
eventDD.innerHTML = "";

//get the template
let ddTemplate = document.getElementById("cityDD-template");

let curEvents = JSON.parse(localStorage.getItem("eventData")) || events;

//get unique city values from the array
let distinctEvents = [...new Set(curEvents.map((event) => event.city))];

let ddItemNode = document.importNode(ddTemplate.content, true);
let ddItem = ddItemNode.querySelector("a");
ddItem.setAttribute("data-string", "All");
ddItem.textContent = "All";

//append the node item to the page
eventDD.appendChild(ddItemNode);

//add the cities to the dropdown
for (let index = 0; index < distinctEvents.length; index++) {
ddItemNode = document.importNode(ddTemplate.content, true);
ddItem = ddItemNode.querySelector("a");
ddItem.setAttribute("data-string", distinctEvents[index]);
ddItem.textContent = distinctEvents[index];
eventDD.appendChild(ddItemNode);
}

//display stats for all events
displayStats(curEvents);
displayData();

}

//display data for the current events
function displayData() {
let template = document.getElementById("eventData-template");
let eventBody = document.getElementById("eventBody");

eventBody.innerHTML = "";

let curEvents = JSON.parse(localStorage.getItem("eventData")) || events;

for (let index = 0; index < curEvents.length; index++) {
let eventRow = document.importNode(template.content, true);
let eventCols = eventRow.querySelectorAll("td");

eventCols[0].textContent = curEvents[index].event;
eventCols[1].textContent = curEvents[index].city;
eventCols[2].textContent = curEvents[index].state;
eventCols[3].textContent = curEvents[index].attendance;
eventCols[4].textContent = new Date(
curEvents[index].date).toLocaleDateString();

eventBody.appendChild(eventRow);
}
}

function displayStats(filteredEvents) {
let total = 0;
let average = 0;
let most = 0;
let least = -1;

filteredEvents.forEach((e) => {
let currentAttendance = e.attendance;
total += currentAttendance;

if (most < currentAttendance) {
most = currentAttendance;
}
if (least > currentAttendance || least < 0) {
least = currentAttendance;
}
average = total / filteredEvents.length;


})

document.getElementById("total").innerHTML = total.toLocaleString();
document.getElementById("most").innerHTML = most.toLocaleString();
document.getElementById("average").innerHTML = average.toLocaleString(
undefined, {
minimumFractionDigits: 0,
maximumFractionDigits: 0,
});

document.getElementById("least").innerHTML = least.toLocaleString();
}

//get the events for the selected city
function getEvents(ddElement) {
let cityName = ddElement.getAttribute("data-string");

let curEvents = JSON.parse(localStorage.getItem("eventData")) || events;
let filteredEvents = curEvents;

document.getElementById("statsHeader").innerHTML = `Stats for ${cityName}`

if (cityName != "All") {
//filter the array using filter array method
filteredEvents = curEvents.filter(function (event) {
if (event.city == cityName) {
return event;
}
})
}
displayStats(filteredEvents);
}

//Save event data to local storage
function saveData() {
let curEvents = JSON.parse(localStorage.getItem("eventData")) || events;
let stateSelect = document.getElementById("addEventState");
let eventDate = document.getElementById("addDate").value;
let eventDate2 = `${eventDate} 00:00`;

let newEvent = {
event: document.getElementById("addEventName").value,
city: document.getElementById("addCity").value,
state: stateSelect.options[stateSelect.selectedIndex].text,
attendance: parseInt(document.getElementById("addAttendance").value, 10),
date: new Date(eventDate2).toLocaleDateString(),
};

curEvents.push(newEvent);

localStorage.setItem("eventData", JSON.stringify(curEvents));

buildDropdown();
displayData();


}

The first few lines of code are dedicated to the data that will be fed to a table for display inside the SuperDog application. This data contains information regarding locations, dates and attendance figures for ComicCon events.


Next, a function named buildDropdown performs the task implied by its name, which is to build a dropdown menu on page load. This will give the user an intuitive button to click so that the events within the pre-filled table on the lower portion of the page can be filtered by city.


The next function is called displayData, and is responsible for retrieving any data that exists within the table and displaying it to the user. This function accounts for the possibility that the data could originate from more than one potential source: either the pre-existing data that was mentioned in the previous paragraph OR any data that has been manually added to the table by the user. Local storage is utilized to simulate the presence of a database, and is where any new user-created data is held. This function parses local storage via the JavaScript Object Notation file (commonly referred to as JSON) and then combines both potential sources into one seamless view.


The next function (displayStats) is responsible for the calculations that allow for filtering of the events within the table by city. This gives users of the application the ability to compare attendance statistics by market in order to evaluate which cities draw the largest or fewest number of attendees. Any new event data added to the table by the user is retained in the browser's local cache upon being saved. The function then runs again and includes this new data in its calculations.