於靜宜大學課程清單網頁顯示目前選課人數與人數上限
// ==UserScript== // @name 靜宜大學課程清單選課人數顯示 // @namespace https://www.moontai0724.tw // @version 1.1.1 // @description 於靜宜大學課程清單網頁顯示目前選課人數與人數上限 // @author moontai0724 // @match https://alcat.pu.edu.tw/2011courseAbstract/main.php* // @supportURL https://github.com/moontai0724/pu-courses-quota-filter-script // @grant none // @license MIT // ==/UserScript== (function () { "use strict"; document.head.innerHTML += `<style type="text/css">#menu>li { padding: 0px 0px 0px 0px !important; }</style>`; document.querySelector("tr>th:nth-child(2)").innerHTML += "(目前人數/人數上限)"; const li = document.createElement("li"); const button = document.createElement("a"); button.href = "#"; button.innerHTML = "獲取目前修課人數"; button.addEventListener("click", showQuota); li.appendChild(button); document.querySelector("#menu").appendChild(li); function showQuota() { if (Number(sessionStorage.getItem("delayTime")) > new Date().getTime()) { window.alert("請稍後再嘗試!冷卻時間十秒!"); return; } sessionStorage.setItem("delayTime", new Date().getTime() + 10000); removeExisting(); const courses = Array.from(document.getElementsByTagName("tr")) .map((element) => { const items = element.getElementsByTagName("td"); if (items.length === 0) return NaN; const id = items[0]; return { id: parseInt(id.textContent), target: items[1], element }; }) .filter((item) => !isNaN(item.id)); courses.forEach(async (course) => { const quota = await fetchQuota(course.id); const display = document.createElement("span"); display.classList.add("classQuota"); if (quota.current >= quota.quota) { display.style.color = "red"; display.style.fontWeight = "bold"; } display.innerHTML = `(${quota.current}/${quota.quota})`; course.target.appendChild(display); }); } function removeExisting() { const existing = document.getElementsByClassName("classQuota"); Array.from(existing).forEach((e) => e.remove()); } async function fetchQuota(classNumber) { const data = new URLSearchParams({ selectno: classNumber.toString().padStart(4, "0"), }); const response = await fetch( "https://alcat.pu.edu.tw/choice/q_person.html", { method: "POST", headers: { "Content-Type": "application/x-www-form-urlencoded", }, body: data, cache: "no-cache", } ).then((response) => response.text()); const rawTable = response.match(/\<table[\s\S]*?\<\/table\>/).shift(); let table = document.createElement("table"); table.innerHTML = rawTable; const list = Array.from(table.querySelectorAll("tr")); const quota = Number(list.shift().querySelectorAll("td")[1].textContent); const current = list .map((e) => Number(e.querySelectorAll("td")[1].textContent)) .reduce((a, b) => a + b, 0); return { quota, current }; } })();