Refresh the page
// ==UserScript==
// @name CDC Booking Script v0
// @namespace http://tampermonkey.net/
// @version 0.12
// @description Refresh the page
// @author afjw
// @match https://bookingportal.cdc.com.sg/*
// @include https://bookingportal.cdc.com.sg/NewPortal/Booking/BookingPL.aspx
// @grant none
// @license MIT
// ==/UserScript==
const isTesting = false;
let stop = false;
const getPreferredTimeSlots = () => {
return JSON.parse(localStorage.getItem("preferredTimeSlots") || "[]");
};
const isLoading = () => {
const loadingSpinner = document.querySelector(
"#ctl00_ContentPlaceHolder1_UpdateProgress1"
);
return loadingSpinner && loadingSpinner.style.display !== "none";
};
const toggleSelector = async () => {
const s = document.querySelector("#ctl00_ContentPlaceHolder1_ddlCourse");
s.value = s.children[1].value;
s.dispatchEvent(new Event("change"));
await sleep(1);
while (isLoading()) {
sendMessage("reloading");
await sleep(1000);
window.location.reload()
}
return Promise.resolve();
};
const isTimeSlotPreferred = ({ date = "", day = "", slot = "" }) => {
if (!date) {
return false;
}
const preferredTimeSlots = getPreferredTimeSlots();
const matched = preferredTimeSlots.find(
(timeSlot) => timeSlot.day === day && timeSlot.slot === slot
);
return !!matched;
};
const getAvailableSlots = () => {
const table = document.querySelector("#ctl00_ContentPlaceHolder1_gvLatestav");
if (!table) {
return [];
}
const rows = table.querySelectorAll("tr");
const timeSlotRow = rows[0];
let allSlots = [];
rows.forEach((row, index) => {
if (index !== 0) {
const slots = [];
const cells = row.querySelectorAll("td");
const timeSlotCells = timeSlotRow.querySelectorAll("th");
cells.forEach((cell, index) => {
const slotInput = cell.querySelector("input");
if (slotInput) {
const src = slotInput.src || "";
if (src.includes("Images1")) {
const date = cells[0].textContent;
const day = cells[1].textContent;
const slot = timeSlotCells[index].textContent.slice(1);
slots.push({
slotInput,
date,
day,
slot,
});
}
}
});
allSlots = allSlots.concat(slots);
}
});
return allSlots;
};
const reserve = (availableSlots) => {
const timeAvailableSlot = availableSlots.find((slot) =>
isTimeSlotPreferred(slot)
);
if (timeAvailableSlot) {
const slotInput = timeAvailableSlot.slotInput;
if (slotInput) {
slotInput.click();
return timeAvailableSlot;
}
}
return false;
};
const run = async () => {
try {
await toggleSelector();
console.log(`Refreshed at`, new Date().toLocaleTimeString());
const availableSlots = getAvailableSlots();
const hasResult = !!availableSlots.length;
if (hasResult) {
const reservedSlot = reserve(availableSlots);
let msg;
if (reservedSlot) {
msg = `Slot is reserved\n${reservedSlot.date} ${reservedSlot.day} ${reservedSlot.slot}`;
} else {
const slotsMsg = `${availableSlots
.map(({ date, day, slot }) => `${date} ${day} ${slot}`)
.join("\n")}`;
msg = `Slots are available\n${slotsMsg}`;
}
sendMessage(`[${new Date().toLocaleString("en-ca")}] ${msg}`);
}
return Promise.resolve(hasResult);
} catch (e) {
return Promise.reject(`error: ${e}`);
}
};
const sleep = (interval) => {
return new Promise((resolve) => {
const jitter = Math.random() * 3 * 1000; // ±10s
window.setTimeout(resolve, interval + jitter);
});
};
// https://simpleguics2pygame.readthedocs.io/en/latest/_static/links/snd_links.html
const audio = new Audio(
"http://codeskulptor-demos.commondatastorage.googleapis.com/descent/gotitem.mp3"
);
const sendMessage = (msg) => {
console.log(msg);
audio.play();
};
let startButton;
let settingButton;
let settingPopup;
const start = async () => {
const hasResult = await run();
if (hasResult) {
await sleep(1);
} else {
await sleep(30 * 1000);
}
if (!stop) {
start();
} else {
console.log('stopped');
}
};
const insertStartButton = () => {
startButton = document.createElement("button");
startButton.className = "start-button";
startButton.textContent = "running...";
const startButtonStyle = document.createElement("style");
startButtonStyle.innerHTML = `
.start-button {
position: fixed;
bottom: 1670px;
right: 150px;
width: 100px;
padding: 5px;
line-height: 1;
background-color: #ff9933;
border: none;
border-radius: 10px;
font-size: 18px;
cursor: pointer;
}
`;
document.head.appendChild(startButtonStyle);
document.body.appendChild(startButton);
startButton.addEventListener("click", async () => {
try {
stop = true;
startButton.textContent = "stopped";
} catch (e) {
startButton.textContent = "error";
console.log("error", e);
}
});
};
const insertSettingButton = () => {
settingButton = document.createElement("button");
settingButton.className = "setting-button";
settingButton.textContent = "setting";
const settingButtonStyle = document.createElement("style");
settingButtonStyle.innerHTML = `
.setting-button {
position: fixed;
bottom: 1030px;
right: 150px;
width: 100px;
padding: 5px;
line-height: 1;
background-color: #ff9933;
border: none;
border-radius: 10px;
font-size: 18px;
cursor: pointer;
}
`;
document.head.appendChild(settingButtonStyle);
document.body.appendChild(settingButton);
settingButton.addEventListener("click", async () => {
toggleSettingPopup();
});
};
const toggleSettingPopup = () => {
if (window.getComputedStyle(settingPopup).display === "none") {
settingPopup.style.display = "initial";
} else {
settingPopup.style.display = "none";
}
};
const insertSettingPopup = () => {
settingPopup = document.createElement("div");
settingPopup.className = "setting-popup";
const settingPopupStyle = document.createElement("style");
settingPopupStyle.innerHTML = `
.setting-popup {
display: none;
position: fixed;
bottom: 865px;
right: 50px;
width: 900px;
padding: 10px;
background-color: #ff9933;
border: none;
border-radius: 10px;
font-size: 22px;
}
.setting-popup label {
margin-bottom: 10px;
display: block;
}
.setting-popup table {
width: 100%;
table-layout: auto;
border-collapse: collapse;
}
.setting-popup table th,
.setting-popup table td {
text-align: center;
border: 1px solid black;
}
.setting-popup table th {
padding: 5px;
}
.setting-popup table td {
height: 35px;
}
.setting-popup table img {
width: 30px;
}
.setting-popup table .selectable {
cursor: pointer;
}
.setting-buttons {
display: flex;
flex-wrap: wrap;
}
.setting-buttons .toggle-button {
border: none;
padding: 5px;
border-radius: 5px;
margin-right: 5px;
margin-bottom: 5px;
cursor: pointer;
}
.setting-buttons .toggle-button.active {
background-color: red;
}
`;
document.head.appendChild(settingPopupStyle);
document.body.appendChild(settingPopup);
renderSettingTable();
};
const toggleTimeSlot = (day, slot) => {
const savedPreferredTimeSlots = getPreferredTimeSlots();
const matched = savedPreferredTimeSlots.find(
(timeSlot) => timeSlot.day === day && timeSlot.slot === slot
);
if (matched) {
savedPreferredTimeSlots.splice(savedPreferredTimeSlots.indexOf(matched), 1);
} else {
savedPreferredTimeSlots.push({
day,
slot,
});
}
localStorage.setItem(
"preferredTimeSlots",
JSON.stringify(savedPreferredTimeSlots)
);
renderSettingTable();
};
const renderSettingTable = () => {
const slots = [
"08:30 - 10:10",
"10:20 - 12:00",
"12:45 - 14:25",
"14:35 - 16:15",
"16:25 - 18:05",
"18:50 - 20:30",
"20:40 - 22:20",
];
const days = ["MON", "TUE", "WED", "THU", "FRI", "SAT", "SUN"];
window.toggleTimeSlot = toggleTimeSlot;
const savedPreferredTimeSlots = getPreferredTimeSlots();
const table = `
<label>Preferred Time Slots</label>
<table>
<thead>
<tr>
<th>Day</th>
${slots.map((slot) => `<th>${slot}</th>`).join("\n")}
</tr>
</thead>
${days
.map((day) => {
return `
<tr>
<td>${day}</td>
${slots
.map((slot) => {
const isSelected = !!savedPreferredTimeSlots.find(
(timeSlot) => timeSlot.day === day && timeSlot.slot === slot
);
return `
<td class="selectable" onclick="toggleTimeSlot('${day}', '${slot}', '${isSelected}')">
${
isSelected
? '<img src="" />'
: ""
}
</td>
`;
})
.join("\n")}
</tr>`;
})
.join("\n")}
</table>
`;
settingPopup.innerHTML = table;
};
const main = async () => {
insertStartButton();
insertSettingButton();
insertSettingPopup();
start();
};
function isTargetPage() {
return location.href === 'https://bookingportal.cdc.com.sg/NewPortal/Booking/BookingPL.aspx';
}
if (isTargetPage()) {
(function () {
"use strict";
main();
(function () {
setInterval(() => {
sendMessage("reloading");
sleep(1000);
window.location.reload();
}, 27 * 60 * 1000);
})();
})();
}