You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

254 lines
7.4 KiB
JavaScript

3 years ago
let page = "Padliography";
let endpoint = `https://pzwiki.wdka.nl/mw-mediadesign/api.php?action=parse&format=json&origin=*&section=1&page=${page}&prop=text`;
3 years ago
const container = document.getElementById("wiki-contents");
const filtersContainer = document.getElementById("filters-container");
3 years ago
const activeAllButton = document.getElementById("active-all");
3 years ago
// We use a set and not an array in order to avoid repetition for the categories
3 years ago
let filters = new Set();
3 years ago
// Variables for the filter system
let allActive = true;
let tagList;
// Global variable for the generated table
let table;
3 years ago
// Request the content to the Wiki API and parse it as a JSON file
fetch(endpoint)
.then((response) => response.json())
.then((data) => {
// Create a new element to parse the response from the wiki and get an HTML table element out of it
3 years ago
let section = document.createElement("div");
section.innerHTML = data.parse.text["*"];
let wikiTable = section.getElementsByTagName("table")[0];
// Create a new table using the data in the Wiki table
createTable(wikiTable);
// Fill the filter section with all the tags gathered in the filters Set
// We don't need any argument because the filters set is defined as global
createFilters();
3 years ago
// Initially show all the categories
3 years ago
showAllTags();
3 years ago
});
// ---
// Contents Generation
// ---
3 years ago
function createTable(data) {
3 years ago
table = document.createElement("table");
3 years ago
// Traverse the rows collection as an array
[...data.rows].forEach((row, index) => {
// Avoid first row that contains column titles
if (index > 0) {
// Create a row element
let tableRow = document.createElement("tr");
3 years ago
// Create a list of tags from the categories cell
let categoryTags = row.cells[3].innerText.split(",").map((category) => category.trim());
// Set the categories as classes for future filtering
categoryTags.forEach((tag) => tableRow.classList.add(tag));
// --- Beginning of the row
3 years ago
// Columns order could be modified simply by reordering the next 4 sections
// Create a cell with date
let date = document.createElement("td");
date.innerHTML = row.cells[4].innerText;
tableRow.appendChild(date);
3 years ago
// Create a cell with title + link to the pad
let title = document.createElement("td");
title.appendChild(createTitleLink(row));
tableRow.appendChild(title);
// Create a cell with categories
let categories = document.createElement("td");
3 years ago
categories.appendChild(createTags(categoryTags));
3 years ago
tableRow.appendChild(categories);
// Create a cell with the overview
let overview = document.createElement("td");
overview.innerHTML = row.cells[2].innerText;
tableRow.appendChild(overview);
// --- End of the row
3 years ago
// Insert the row in the table
table.appendChild(tableRow);
}
});
// Insert the table in the container
container.appendChild(table);
}
function createTitleLink(row) {
// Take the first cell of the wiki (link)
let url = row.cells[0].innerText;
// Take the second cell of the wiki (title)
let title = row.cells[1].innerText;
// Create a link element
let link = document.createElement("a");
link.setAttribute("target", "_blank");
// Use the link as href
link.href = url;
// Use the title as text
link.innerHTML = title;
return link;
}
// Take a list of categories and create an unordered list with them
// + store every category to the filters Set
3 years ago
function createTags(categories) {
// Create a list to contain all the tags
let tagsContainer = document.createElement("ul");
// For each category create a li element
3 years ago
categories.forEach((item) => {
let tag = document.createElement("li");
tag.classList.add("tag");
tag.innerHTML = item;
3 years ago
tagsContainer.appendChild(tag);
// add the category to the filters Set to use it later
filters.add(item);
});
return tagsContainer;
}
// ---
// Filtering
// ---
// Parse the filters Set to create a list of interactive list elements
3 years ago
function createFilters() {
filters.forEach((item) => {
3 years ago
let tag = document.createElement("li");
// Store the filter value in the markup and use it for filtering later
3 years ago
tag.setAttribute("data-tag", item);
// Tab index is for accessibility via keyboard
3 years ago
tag.setAttribute("tabindex", 0);
tag.setAttribute("role", "button");
3 years ago
tag.classList.add("tag");
tag.innerHTML = item;
3 years ago
filtersContainer.appendChild(tag);
});
// Build a list of the tag for managing the filter
3 years ago
tagList = document.querySelectorAll("[data-tag]");
}
// Manage all the combinations when a filter is clicked or selected
3 years ago
function conditionalToggle(element) {
// If all the filters are active disable them all except the one selected
3 years ago
if (allActive) {
tagList.forEach((tag) => {
filters.delete(tag.getAttribute("data-tag"));
tag.classList.remove("active");
tag.setAttribute("aria-expanded", "false");
});
activeAllButton.classList.remove("active");
allActive = false;
}
// If the filter is active turn it off, else the contrary
3 years ago
if (element.classList.contains("active")) {
element.classList.remove("active");
element.setAttribute("aria-expanded", "false");
filters.delete(element.getAttribute("data-tag"));
} else {
element.classList.add("active");
element.setAttribute("aria-expanded", "true");
filters.add(element.getAttribute("data-tag"));
}
// Check if every filter is active or not
// use it to apply the correct style when all the filters are active
3 years ago
allActive = tagList.length == filters.size;
if (allActive) {
activeAllButton.classList.add("active");
activeAllButton.setAttribute("aria-expanded", "true");
}
// If there are no filters active enable them all
3 years ago
if (filters.size === 0) {
showAllTags();
}
3 years ago
}
3 years ago
// Display the rows that have their categories in the filters Set
3 years ago
function showSelectedRows() {
[...table.rows].forEach((row) => {
row.classList.remove("active");
});
filters.forEach((filter) => {
let selectedRows = [...table.rows].filter((el) => {
filter = filter.replaceAll(" ", "-");
return el.classList.contains(filter);
});
selectedRows.forEach((row) => row.classList.add("active"));
});
allActiveCheck();
}
function showAllTags() {
tagList.forEach((tag) => {
filters.add(tag.getAttribute("data-tag"));
tag.classList.add("active");
tag.setAttribute("aria-expanded", "true");
});
allActive = true;
showSelectedRows();
}
function allActiveCheck() {
if (tagList.length === filters.size) {
allActive = true;
tagList.forEach((tag) => tag.classList.add("all"));
filtersContainer.firstElementChild.classList.add("active");
} else {
allActive = false;
tagList.forEach((tag) => tag.classList.remove("all"));
filtersContainer.firstElementChild.classList.remove("active");
}
}
// ---
3 years ago
// Event Listener
// ---
3 years ago
filtersContainer.addEventListener("click", (e) => {
if (e.target.tagName === "LI") {
conditionalToggle(e.target);
showSelectedRows();
}
});
activeAllButton.addEventListener("click", (e) => {
activeAllButton.classList.add("active");
activeAllButton.setAttribute("aria-expanded", "true");
showAllTags();
});
document.addEventListener("keydown", (event) => {
if (event.isComposing || event.keyCode === 229) {
return;
}
if (event.keyCode === 32) {
if (document.activeElement.tagName === "LI") {
conditionalToggle(document.activeElement);
showSelectedRows();
}
}
// do something
});