// ==UserScript==
// @name Simplistic Caleb - Light
// @namespace http://tampermonkey.net/
// @version 1.6.2
// @description The Only Caleb Theme
// @author Crjase
// @match https://caleb.btac.nsw.edu.au/*
// @grant none
// ==/UserScript==
let near_due_date = 0;
function myLinks()
{
// Button CSS
const buttonStyle = `
.button-30 {
align-items: center;
appearance: none;
background-color: #FCFCFD;
border-radius: 4px;
border-width: 0;
box-shadow: rgba(45, 35, 66, 0.4) 0 2px 4px,rgba(45, 35, 66, 0.3) 0 7px 13px -3px,#D6D6E7 0 -3px 0 inset;
box-sizing: border-box;
color: #36395A;
cursor: pointer;
display: inline-flex;
font-family: "JetBrains Mono",monospace;
height: 48px;
justify-content: center;
line-height: 1;
list-style: none;
overflow: hidden;
padding-left: 16px;
padding-right: 16px;
position: relative;
text-align: left;
text-decoration: none;
transition: box-shadow .15s,transform .15s;
user-select: none;
-webkit-user-select: none;
touch-action: manipulation;
white-space: nowrap;
will-change: box-shadow,transform;
font-size: 18px;
}
.button-30:focus {
box-shadow: #D6D6E7 0 0 0 1.5px inset, rgba(45, 35, 66, 0.4) 0 2px 4px, rgba(45, 35, 66, 0.3) 0 7px 13px -3px, #D6D6E7 0 -3px 0 inset;
}
.button-30:hover {
box-shadow: rgba(45, 35, 66, 0.4) 0 4px 8px, rgba(45, 35, 66, 0.3) 0 7px 13px -3px, #D6D6E7 0 -3px 0 inset;
transform: translateY(-2px);
}
.button-30:active {
box-shadow: #D6D6E7 0 3px 7px inset;
transform: translateY(2px);
}
`
// Creating the button stylesheet
let stylesheet = document.createElement("style");
stylesheet.innerHTML = buttonStyle;
document.head.appendChild(stylesheet);
const island = document.getElementsByClassName("small-12 island");
// Iterate through all specific class
for (let i=0; i<island.length; i++)
{
// Find the container needed
if (island[i].innerHTML.includes("Daily Prayer"))
{
var linkBoard = island[i];
// Get rid of Daily Prayer
island[i].innerHTML = "";
// -------- island style --------
island[i].style.textAlign = "center";
island[i].style.backgroundColor = "white";
island[i].style.padding = "20px";
// island[i].style.outline = "1px solid black";
island[i].style.position = "relative";
// --------> SubHeader <--------
let subHeader = document.createElement("h2");
subHeader.setAttribute("class", "subheader");
subHeader.innerHTML = "My Link Board";
subHeader.style.textDecoration = "underline";
subHeader.style.fontSize = "1.5em";
subHeader.style.fontWeight = "bold";
linkBoard.appendChild(subHeader);
linkBoard.appendChild(document.createElement("br"));
linkBoard.appendChild(document.createElement("br"));
// --------> Add-Button <--------
let addButton = document.createElement("button");
addButton.innerHTML = "+";
addButton.style.fontWeight = "bold";
addButton.style.fontSize = "2em";
addButton.style.background = "none";
addButton.style.borderRadius = "50%";
addButton.style.width = "4vw";
addButton.style.height = "9vh";
addButton.style.position = "absolute";
addButton.style.top = "0px";
addButton.style.left = "0px";
addButton.onmouseover = function() {
addButton.style.backgroundColor = "white";
}
addButton.onmouseout = function() {
addButton.style.backgroundColor = "transparent";
}
// Create a whole iframe on-top, because I like putting my self through pain
addButton.onclick = function() {
let container = document.createElement("div");
container.style.position = "fixed";
container.style.top = "0";
container.style.left = "0";
container.style.width = "100vw";
container.style.height = "100vh";
container.style.backgroundColor = "rgba(0,0,0,0.5)";
container.style.zIndex = "10000";
container.style.cursor = "pointer";
container.onclick = function()
{
container.remove();
window.location.reload();
}
let iframeContainer = document.createElement("div");
iframeContainer.style.border = "1px solid whitesmoke";
iframeContainer.style.backgroundColor = "rgb(255, 255, 255)";
iframeContainer.style.width = "80%";
iframeContainer.style.height = "80%";
iframeContainer.style.padding = "10px";
iframeContainer.style.position = "fixed";
iframeContainer.style.top = "50%";
iframeContainer.style.left = "50%";
iframeContainer.style.transform = "translate(-50%, -50%)";
container.appendChild(iframeContainer);
let iframe = document.createElement("iframe");
iframe.src = "https://caleb.btac.nsw.edu.au/cms/myLinks";
iframe.style.width = "100%";
iframe.style.height = "100%";
iframeContainer.appendChild(iframe);
// ---- Append Container to DOM ----
document.body.appendChild(container);
}
linkBoard.appendChild(addButton);
// ---- Replace it with something better ----
let linkContainer = document.getElementById("side-menu-mylinks");
let offCanvasList = linkContainer.querySelector(".my-links").querySelector(".off-canvas-list");
let hyperLinkList = offCanvasList.querySelectorAll("li")
links = [];
// Get all links and append to links[]
for (let i = 0; i < hyperLinkList.length; i++)
{
let link = hyperLinkList[i].querySelector("a");
links.push(link);
}
// Remove MyLinks from the sidebar
linkContainer.remove();
// Create button on the DOM for each link
for (const a of links)
{
let button = document.createElement("button");
button.onclick = function() {
window.open(a.href);
}
button.innerHTML = a.innerHTML;
button.setAttribute("class", "button-30");
button.style = "margin-left: 10px; margin-right: 10px;";
linkBoard.appendChild(button);
}
}
}
}
function leftMenu()
{
// Local Variables
const container = document.getElementById("left-menu");
const subMenu = document.querySelectorAll(".left-submenu");
const logo = document.querySelector("img[src='/images/logo.php?logo=skin_logo_large&size=normal']");
// Sleep Method
function sleep(ms) {
return new Promise(resolve => setTimeout(resolve, ms));
}
async function menuLoop() {
// Each individual link on the container
const elements = container.getElementsByTagName("a");
const menuTitle = container.getElementsByTagName("h3")[0];
// Cold Colours
const coldColours = [
"magenta", "blue", "purple", "navy",
"royal", "space", "iceberg", "sapphire",
"denim", "vivid cerulean", "lapis lazuli",
"maya blue"]
// Void Colours
const voidColours = ["#000000", "#696969", "#808080", "#A9A9A9"]
// One-Time-Applied text CSS
for (let i=0; i<elements.length ; i++)
{
// This makes a calm fade effect
elements[i].style.transition = "1s"
// Set light-theme colour for the menu
container.style.backgroundColor = "white";
// Make sure the menu has no image overlay
container.style.backgroundImage = "none";
// Set logo image to inverted colours
logo.style = "filter: invert(1);";
// Set H3 font weight
if (elements[i].parentNode.tagName == "H3")
{elements[i].style.fontWeight = "700";}
// Set menu title css
menuTitle.style.color = "black";
menuTitle.style.fontSize = "large";
menuTitle.style.letterSpacing = "2px";
menuTitle.innerHTML = "Caleb";
}
// One-Time-Applied subMenu CSS
for (let i=0; i<subMenu.length ; i++)
{
// Make sure the subMenu has no image overlay
subMenu[i].style.backgroundImage = "none";
// Set light-theme for the subMenu
subMenu[i].style.backgroundColor = "white";
}
while (true)
{
await sleep(100);
for (var i=0; i<elements.length; i++)
{
// Select a random cold colour
let coldColour = coldColours[Math.floor(Math.random()*coldColours.length)];
// Select a random hot colour
let voidColour = voidColours[Math.floor(Math.random()*voidColours.length)];
if (! (elements[i].parentNode.tagName == "H3"))
{
elements[i].style.color = coldColour
} else
{
elements[i].style.color = voidColour
}
}
}
}
let worker = new Worker(menuLoop());
}
function dueWorkContainer()
{
const island = document.getElementsByClassName("small-12 island");
// Iterate through all specific class
for (let i=0; i<island.length; i++)
{
const elem = island[i];
// Find the correct container
if (elem.innerHTML.includes("Upcoming Due Work"))
{
// Parent node of elem
// const elemP = elem.parentNode;
// -------- Apply Custom Style --------
// elem.style.outline = "1px solid black";
elem.style.padding = "10px";
elem.style.backgroundColor = "white";
// ---- Make the due work highlighted ----
const infoList = elem.querySelector("section>ul.information-list").querySelectorAll("li");
for (let i = 0; i < infoList.length; i++) {
let info = infoList[i];
// Get the link for later
const link = info.querySelector("div>h3>a.title");
//info.style.backgroundColor = "whitesmoke";
info.style.margin = "10px";
info.style.cursor = "pointer";
// ---- Determine if something is near the due date ----
const due = info.querySelector("div").querySelectorAll("p");
// Array of tasks near due date
let dueTasks = [];
for (let i = 0; i < due.length; i++) {
if (due[i].outerText.includes("Due")) {
let date = due[i].outerText;
let split_date = date.split(" ")
/* For Debugging */
//console.log(split_date);
let index;
var day = "Failed to Get";
var hours = "Failed to Get";
// Get the due date using the more complex method.
if (split_date[2] != "at" && !(split_date[2].includes("("))) {
index = `${split_date[1]} ${split_date[2]} ${split_date[3]}`;
date = split_date.join(" ");
let date_1 = new Date(index);
let date_2 = new Date();
const days = (date_1, date_2) =>{
let difference = date_1.getTime() - date_2.getTime();
let TotalDays = Math.ceil(difference / (1000 * 3600 * 24));
return TotalDays;
};
day = days(date_1, date_2);
}
// If there is less than 6 days left, 'at' will be present.
if (split_date.includes("at") && split_date[5] && !(split_date[5].includes("hours"))) {
index = split_date[2];
day = index.replace("(", "");
};
// There is no 'at', but [2] includes '(<int>'. This is around a few days left.
if (split_date[2] && split_date[2].includes("(")) {
index = split_date[2];
day = index.replace("(", "");
};
if (split_date[5] && split_date[5].includes("hours")) {
index = split_date[4];
hours = index.replace("(", "");
};
// I'm not sure why the position is relative? Forgot???
info.style.position = "relative";
// Globalize (p), so it can be accessed by infoTheme
var p = document.createElement("p");
if (day != "Failed to Get") {
let height = info.height;
p.innerHTML = `${day} Days`;
p.style.fontSize = "1em";
p.style.position = "absolute";
p.style.right = "0";
p.style.top = "50%";
p.style.transform = "translate(-50%, -50%)";
info.appendChild(p);
}
else if (hours != "Failed to Get") {
let height = info.height;
p.innerHTML = `${hours} Hours`;
p.style.fontSize = "1em";
p.style.position = "absolute";
p.style.right = "0";
p.style.top = "50%";
p.style.transform = "translate(-50%, -50%)";
info.appendChild(p);
}
else {
let height = info.height;
p.innerHTML = "Failed to Get Time";
p.style.fontSize = "1em";
p.style.position = "absolute";
p.style.right = "0";
p.style.top = "50%";
p.style.transform = "translate(-50%, -50%)";
info.appendChild(p);
};
};
function infoTheme(info, { bg_over, bg_leave, img_over, img_leave, fg_over, fg_leave, fg_weight, img_transition, bg_transition, audio }) {
// Make the div also a link to the task
info.onclick = () => {
location.href = link;
};
if (fg_weight)
p.style.fontWeight = fg_weight;
// Replace background with image on mouse leave
if (img_leave)
info.style.backgroundImage = img_leave;
// Set the foreground colour
if (fg_leave)
info.style.color = fg_leave;
// Replace background with colour on mouse leave
if (bg_leave)
info.style.backgroundColor = bg_leave;
// Highlight it
info.onmouseover = () => {
if (audio) {
let aud = new Audio(audio);
aud.play();
};
if (fg_over)
info.style.transition = "0s";
info.style.color = fg_over;
if (img_over && bg_over) {
console.warn("You cannot have both over events, Priortising img_over...");
// Transition Time
if (img_transition)
info.style.transition = img_transition;
info.style.backgroundImage = `url("${img_leave}")`;
info.style.backgroundSize = "100%";
info.style.backgroundRepeat = "no-repeat";
info.style.backgroundPosition = "center";
if (bg_leave)
info.style.removeProperty("background-color");
return;
};
if (img_over) {
// Transition Time
if (img_transition)
info.style.transition = img_transition;
info.style.backgroundImage = `url("${img_over}")`;
info.style.backgroundSize = "100%";
info.style.backgroundRepeat = "no-repeat";
info.style.backgroundPosition = "center";
if (bg_leave)
info.style.removeProperty("background-color");
};
if (bg_over) {
// Transition Time
if (bg_transition)
info.style.transition = bg_transition;
info.style.backgroundColor = bg_over;
if (img_leave)
info.style.removeProperty("background-image");
info.style.removeProperty("background-size");
info.style.removeProperty("background-position");
info.style.removeProperty("background-repeat");
};
};
// Don't Highlight it
info.onmouseleave = () => {
if (fg_leave)
info.style.transition = "0s";
info.style.color = fg_leave;
if (img_leave && bg_leave) {
console.warn("You cannot have both leave events, Priortising img_leave...");
// Transition Time
if (img_transition)
info.style.transition = img_transition;
info.style.backgroundImage = `url("${img_leave}")`;
info.style.backgroundSize = "100%";
info.style.backgroundRepeat = "no-repeat";
info.style.backgroundPosition = "center";
// Remove background-color even though the image covers it
if (bg_over)
info.style.removeProperty("background-color");
return;
};
if (img_leave) {
// Transition Time
if (img_transition)
info.style.transition = img_transition;
info.style.backgroundImage = `url("${img_leave}")`;
info.style.backgroundSize = "100%";
info.style.backgroundRepeat = "no-repeat";
info.style.backgroundPosition = "center";
// Remove background-color even though the image covers it
if (bg_over)
info.style.removeProperty("background-color");
};
if (bg_leave) {
// Transition Time
if (bg_transition)
info.style.transition = bg_transition;
if (img_over)
info.style.removeProperty("background-image");
info.style.removeProperty("background-size");
info.style.removeProperty("background-position");
info.style.removeProperty("background-repeat");
info.style.backgroundColor = bg_leave;
};
};
};
// Find how close something is to be submitted and apply themes
if (due[i].innerHTML.includes("Due")) {
// Error Occured
if (day == "Failed to Get") {
const url = "https://rare-gallery.com/uploads/posts/336764-Anime-Scenery-Horizon-Shooting-Star-Sunset-4K-3840x2160.jpg";
infoTheme(info, { img_over : url, bg_leave : "#01031A", fg_leave : "red", img_transition: "2.4s", bg_transition: "0s" });
}
// No Warning
if (day > 20) {
infoTheme(info, { bg_over : "whitesmoke", bg_leave : "white"} );
}
// Warning
if (day <= 20) {
infoTheme(info, { bg_over : "orange", bg_leave : "#FFDFBF" });
}
// Critical Warning
if (day <= 10) {
// Append this task to the dueTasks array
const url = "";
infoTheme(info, {
img_over : url,
bg_leave : "#ffcccb",
img_transition : "0.4s",
bg_transition : "0s",
fg_over : "whitesmoke",
fg_leave: "black"
});
// Getting near due date
near_due_date += 1;
};
// Galactically-Critical Warning
if (hours != "Failed to Get") {
const url = "";
infoTheme(info, { img_over : url, bg_leave : "#0B135E", fg_leave : "red", fg_over : "white", img_transition: "2.4s", bg_transition: "0s", fg_weight : "600" });
}
};
}
}
}
}
}
function specialNameTag()
{
// ahhh, old code.
// Find container easily, as it's at the top of the page.
const welcoming = document.querySelector("div.small-12.columns>h1")
const nameLabel = document.querySelector("div.small-12.columns>h1>strong")
nameLabel.style.color = "cornflowerblue";
nameLabel.style.fontWeight = "600";
// ahhh, new code.
welcoming.style.marginBottom = "0";
let due = document.createElement("p");
due.innerHTML = `You have ${near_due_date} near due tasks.`;
due.style.marginBottom = "2rem";
if (near_due_date > 0) {
due.style.color = "red";
}
const aaaaa = welcoming.parentNode;
aaaaa.appendChild(due);
}
function importFont(url)
{
// Open Sans
let font = document.createElement("link")
font.href = url;
font.rel = "stylesheet";
font.type = "text/css";
document.head.appendChild(font);
}
function notificationBox()
{
const boxWide = document.getElementsByClassName("right-menu-dock");
const boxClass = document.getElementsByClassName("icon-notifications");
for (let i = 0; i < boxWide.length; i++)
{
boxWide[i].style.backgroundColor = "green";
boxWide[i].style.outline = "1px solid black";
}
for (let i = 0; i < boxClass.length; i++)
{
boxClass[i].style.backgroundColor = "green";
boxClass[i].style.outline = "1px solid black";
}
}
// This function replaces the "student printing" text with
// what subjects I should focus for, to learn mechatronics.
function subjectFocus() {
const SUBJECTS = `
Education & Training for a Mechatronic Engineer
Prerequisite subjects,
or assumed knowledge,
in one or more of <span style="font-weight: bold;">English</span>,
<span style="font-weight: bold;">mathematics</span>,
<span style="font-weight: bold;">chemistry</span> and <span style="font-weight: bold;">physics</span> are normally required.
`
// Iterate through small-12 island
for (let i=0; i < document.getElementsByClassName("small-12 island").length; i++)
{
let island = document.getElementsByClassName("small-12 island")[i];
// Find the correct container
if (island.innerHTML.includes("STUDENT PRINTING IS NOW AVAILABLE"))
{
// ---------- Replace all text with subjectFocus ----------
//
const textDict = island.querySelector("section>article>p").querySelectorAll("span");
const container = island.querySelector("section");
// Remove all text
for (let i = 0; i < textDict.length; i++) {
let text = textDict[i];
text.remove();
}
// Add my own text
const sp = document.createElement("span");
sp.innerHTML = SUBJECTS;
// Append text
island.querySelector("section>article>p").appendChild(sp);
// -------- Container Style --------
container.style.backgroundColor = "#FDFD96";
}
}
}
// Main Method
(function() {
'use strict';
// Call Methods
leftMenu();
myLinks();
dueWorkContainer();
specialNameTag();
// notificationBox();
subjectFocus();
})();