// ==UserScript== // @name MENDO.MK Enhancement // @version 50.2 // @namespace mendo-mk-enhancement // @description Adds dark mode, search in tasks and other stuff to MENDO.MK // @author EntityPlantt // @match *://mendo.mk/* // @exclude *://mendo.mk/jforum/* // @require https://cdn.jsdelivr.net/npm/chart.js@4.4.8 // @noframes // @icon https://mendo.mk/img/favicon.ico // @grant none // @license CC-BY-ND // @downloadURL https://update.greasyfork.icu/scripts/450985/MENDOMK%20Enhancement.user.js // @updateURL https://update.greasyfork.icu/scripts/450985/MENDOMK%20Enhancement.meta.js // ==/UserScript== const VERSION = 50.2, AprilFools = new Date().getMonth() == 3 && new Date().getDate() < 3, EventDeadline = new Date("apr 15 25").getTime(); console.log("%cMENDO.MK Enhancement", "color:magenta;text-decoration:underline;font-size:20px"); function localize(english, macedonian) { return document.cookie.includes("mkjudge_language=en") ? english : macedonian; } const achlink = { task: "/Training.do?cid=1", readlec: "/Training.do?cid=6" }; const achname = { task0: localize("Apprentice (10 tasks)", "Чирак (10 задачи)"), task1: localize("Guru (25 tasks)", "Гуру (25 задачи)"), task2: localize("Pupil (50 tasks)", "Ученик (50 задачи)"), task3: localize("Specialist (100 tasks)", "Специјалист (100 задачи)"), task4: localize("Sage (150 tasks)", "Мудрец (150 задачи)"), task5: localize("Expert (200 tasks)", "Експерта (200 задачи)"), task6: localize("Master (250 tasks)", "Мајстор (250 задачи)"), task7: localize("Champion (300 tasks)", "Шампион (300 задачи)"), task8: localize("Titan (350 tasks)", "Титан (350 задачи)"), task9: localize("The Architect (400 tasks)", "Архитектот (400 задачи)"), task10: localize("Baba (450 tasks)", "Баба (450 задачи)"), task11: localize("Zenith (500 tasks)", "Зенит (500 задачи)"), readlec0: localize("Student I (all Learn C++ tasks solved)", "Студент I (сите Научи C++ лекции прочитани)"), readlec1: localize("Student II (all Algorithms & Learn C++ tasks solved)", "Студент II (сите Алгоритми и Научи C++ лекции прочитани)"), colorful0: localize("Colorful! (Get 1 testcase with every verdict on a single submission)", "Шарено! (Добиј барем 1 тест случај од секој verdict на едно исто решение)") }; async function MendoMkEnhancement() { try { function logFinish(taskName) { (console.debug ?? console.log)("%cFinished setting up:%c " + taskName, "color:#0f0", ""); } function collapseNavigation() { if (!document.querySelector(".main-navigation")) return; localStorage.setItem("nav collapsed", document.querySelector(".main-navigation").classList.toggle("collapsed")); } if (localStorage.getItem("nav collapsed") == "true") collapseNavigation(); logFinish("collapse navigation if collapsed"); var style = document.createElement("style"); if (!(parseFloat(localStorage.getItem("enhancement last version")) >= VERSION)) { Changelog(); localStorage.setItem("enhancement last version", VERSION); } if (!localStorage.getItem("mendo-mk-enhancement-theme")) { localStorage.setItem("mendo-mk-enhancement-theme", window.matchMedia && matchMedia("(prefers-color-scheme: dark)").matches ? "dark" : "light"); logFinish("detect color scheme"); } style.innerHTML = ` @import url("https://cdn.jsdelivr.net/npm/bootstrap-icons@1.11.3/font/bootstrap-icons.min.css"); ${ // Dark mode localStorage.getItem("mendo-mk-enhancement-theme") == "dark" ? ` .page-container, .page-container img, .page-container svg, #cboxWrapper, .copy-io-btn span, .precinematicscreen, .sitelogo, #container, #container img { filter: invert(1) hue-rotate(180deg); } body, img, svg { background-color: black; } ::-webkit-scrollbar { width: initial; } ::-webkit-scrollbar-track { background-color: #eee; } body::-webkit-scrollbar-track { background-color: #111; } ::-webkit-scrollbar-thumb { background-color: #ddd; } #infoinfo { color: white; } ` : ""} body::-webkit-scrollbar-thumb { background-color: #222; } ${AprilFools ? ` td.solved, td.wrong { background-color: #bfb !important; } td.correct { background-color: #fbb !important; }` : ` td.solved, td.correct { background-color: #bfb !important; } td.wrong { background-color: #fbb !important; } `} .copy-io-btn span:before { content: "📃"; } #search { font-family: consolas; } #search, #search-submit { border: solid 2px black; transition: box-shadow .5s; margin-bottom: 20px; } #search-submit:hover { cursor: pointer; } #search:focus { background-color: #eee; } #search-submit:hover, #search:hover { background-color: #ddd; } .copy-io-btn { float: right; background-color: #ddd; padding: 5px; cursor: pointer; border-radius: 5px; user-select: none; } .copy-io-btn:hover { background-color: #e8e8e8; } #search:active, #search:focus, #search-submit:active, #search-submit:focus { box-shadow: 0 0 2.5px 2.5px black; } @keyframes gta-cinematic-image { 0%, 50% {filter: blur(500px); opacity: 0;} 10%, 40% {filter: blur(0px); opacity: 1;} } td.share-solved { background-color: #ac0 !important; } .update-available { animation: update-available .5s infinite linear; } @keyframes update-available { from, to {color: red} 50% {color: white} } .ojtxt { animation: ojtxt 3s 1 linear; color: white; display: block; } @keyframes ojtxt { from {color: #4f4} to {color: white} } .progbar { background-color: #0004; margin-top: .5em; border-radius: 5px; padding: 2.5px; width: 100%; } .progbar > div { background-color: #bfb; border: solid 1px gray; border-radius: 2.5px; height: 100%; position: relative; padding: 5px; box-sizing: border-box; } .hidden { display: none !important; } .sorttask { font: inherit; background-color: transparent; border: none; display: inline-block; float: right; width: 1rem; } td.wrong.verdict-re { background-color: #fda !important; } td.wrong.verdict-tle { background-color: #bbf !important; } .precinematicscreen { top: 0px; left: 0px; position: fixed; width: 100vw; height: 100vh; background-color: white; font-size: 20px; cursor: pointer; z-index: 99999; } .page-container.deathscreen { rotate: 4deg; scale: .95; animation: deathscreen 10s linear 1; } #achievement-toast { position: fixed; bottom: 0; right: 0; font-family: Times New Roman, serif; text-align: right; color: black; } #achievement-toast > div { background-color: gold; border-top-left-radius: 20px; border-bottom-left-radius: 20px; padding: 10px 20px; flex-direction: column; margin-bottom: 10px; animation: achievement-toast 6s ease-in 1; opacity: 0; translate: 100%; scale: 0; } #achievement-toast > div > :first-child { font-size: 20px; } #achievement-toast > div > :last-child { font-size: 30px; } @keyframes achievement-toast { 0%, 100% { opacity: 0; translate: 100%; scale: 0; } 20%, 80% { opacity: 1; translate: 0%; scale: 1; } } @keyframes deathscreen { 0% { rotate: 0deg; scale: 1 } to { rotate: 4deg; scale: .95 } } .main { display: flex; flex-direction: row; justify-content: center; } .main-navigation { transition: width .5s; overflow: hidden !important; } .main-navigation.collapsed { width: 35px; } .main-navigation>* { transition: opacity .5s; opacity: 1; } .main-navigation.collapsed>* { opacity: 0; } .collapse-navigation-container { opacity: 1 !important; float: right; } .collapse-navigation { cursor: pointer; border: none; padding: 0; font-size: 25px; overflow: visible; height: 0; margin-right: 10px; color: white; transition: color .5s; z-index: 99; position: relative; top: -5px; } .main-navigation.collapsed .collapse-navigation { color: black; } .main-content { padding-left: 30px; margin: 0; } .event { position: fixed; left: 0; bottom: 0; display: flex; flex-direction: column; width: 200px; height: 100vh; background-color: #606; font-family: system-ui, Verdana, sans-serif; font-size: 20px; font-weight: bold; text-align: center; color: white; opacity: .75; } .event-text { flex-shrink: 0; padding: 10px 0; } .event-pbar { height: 100%; width: 100%; display: flex; flex-direction: column-reverse; overflow: hidden; } .event-prog { background-color: #808; width: 100%; padding-top: 10px; overflow: hidden; } .event select { color: inherit; font: inherit; background-color: transparent; border: none; border-top: solid 3px #808; } .event-hot, td.event-hot { background-color: #fbf !important; } .event-hot a:hover { background-color: #fdf !important; } #olympsearch { color: #808; text-decoration: underline; } .nameplate { display: flex; align-items: center; flex-direction: row; gap: .5em; } /* April Fools'! */ html.mirrored { transform: rotateY(1620deg) rotateX(-10deg); } html { transition: transform 3s cubic-bezier(0.45, 0, 0.55, 1); } `; document.head.appendChild(style); logFinish("inject style sheet"); if (document.querySelector(".sitename h1 a")) { document.querySelector(".sitename h1").innerHTML += " Enhanced"; } logFinish("complete site logo"); fetch("https://raw.githubusercontent.com/EntityPlantt/EntityPlantt.github.io/refs/heads/main/mendo-enhancement/version.txt").then(async x => { x = await x.text(); let offv = parseFloat(x); if (offv > VERSION) { if (document.getElementById("enhancement-logo")) { document.getElementById("enhancement-logo").classList.add("update-available"); document.querySelector("#enhancement-logo b").innerText = "Update"; } } else if (offv < VERSION) { if (document.getElementById("enhancement-logo")) { document.querySelector("#enhancement-logo b").innerText = "Development"; } } logFinish("check for updates"); }); /* if (document.querySelector(".main-navigation > ul") && !document.URL.includes("Help.do")) { let elm = document.createElement("li"); elm.innerHTML = `${true ? "Пријави Грешка" : "Report Bug"}`; document.querySelector(".main-navigation > ul").appendChild(elm); logFinish("add report bug form"); } */ if (document.URL.includes("/Training.do") || document.URL.includes("/User_Competition.do")) { var search = document.createElement("form"); search.className = "content-search"; search.action = "#"; search.innerHTML = ` `; search.onsubmit = e => { e.preventDefault(); location.hash = "#" + encodeURIComponent(search.querySelector("#search").value); search.querySelector("#search").blur(); hashChange(); } function hashChange() { let kw = decodeURIComponent(location.hash.substring(1)); search.querySelector("#search").value = kw; kw = kw.toLowerCase(); if (kw.includes("mirror")) document.body.parentElement.classList.add("mirrored"); else document.body.parentElement.classList.remove("mirrored"); document.querySelectorAll("body > div.page-container > div.main > div.main-content > div > div > table > tbody > tr").forEach(elm => { if (!elm.querySelector("td:nth-child(2) > a")) { return; } if ((kw == "event.olymp2025" && /(олимпијада|мои|ibuoi)/i.test(elm.querySelector("td:nth-child(3)").innerText)) || elm.innerText.toLowerCase().includes(kw) || elm.querySelector("td:nth-child(2) > a").href.toLowerCase().includes(kw)) { elm.style.display = ""; } else { elm.style.display = "none"; } }); if (AprilFools) { document.querySelectorAll("body > div.page-container > div.main > div.main-content > div > div > table > tbody").forEach(elm => { let afa = Array.from(elm.querySelectorAll("tr")).slice(1); function shuffle(array) { let currentIndex = array.length; while (currentIndex != 0) { let randomIndex = Math.floor(Math.random() * currentIndex); currentIndex--; [array[currentIndex], array[randomIndex]] = [ array[randomIndex], array[currentIndex]]; } } shuffle(afa); afa.forEach(x => elm.appendChild(x)); }); } } window.onhashchange = hashChange; hashChange(); document.querySelector(".main-content").prepend(search); logFinish("add task search bar"); document.querySelector("body > div.page-container > div.main > div.main-content > div:nth-child(3)").innerHTML += `[ ${localize("Other tasks", "Други задачи")} ] `; document.querySelector("body > div.page-container > div.main > div.main-content > div:last-child").innerHTML = document.querySelector("body > div.page-container > div.main > div.main-content > div:nth-child(3)").innerHTML; logFinish("add secret tasks"); if (/(Статистика|Success rate)/.test(document.querySelector(".main-content > .column1-unit > .training-content table tr th:last-child")?.textContent)) { let elm = document.querySelector(".main-content > .column1-unit > .training-content table tr th:last-child"); let select = document.createElement("select"); select.innerHTML = [["normal", "indexmin"], ["latest", "indexmax"], ["min %", "percmin"], ["max %", "percmax"], ["least tried", "submin"], ["most tried", "submax"], ["least solved", "solvmin"], ["most solved", "solvmax"]].map(x => ``).join(""); select.className = "sorttask"; let sortcriteria = { index: tr => parseInt(tr.querySelector("td").textContent), perc: tr => parseInt(/\((\d+)%\)/.exec(tr.querySelector("td:last-child").textContent)[1]), sub: tr => parseInt(/\/(\d+)\b/.exec(tr.querySelector("td:last-child").textContent)[1]), solv: tr => parseInt(/^(\d+)\//.exec(tr.querySelector("td:last-child").textContent)[1]) }; select.oninput = event => { let [, criteria, rev] = /^([a-z]+)(min|max)$/.exec(select.value); rev = rev == "max" ? -1 : 1; let f = sortcriteria[criteria]; let rows = Array.from(document.querySelectorAll(".main-content > .column1-unit > .training-content table tr")).slice(1).sort((a, b) => (f(a) - f(b)) * rev); rows.forEach(x => document.querySelector(".main-content > .column1-unit > .training-content tbody").appendChild(x)); }; elm.appendChild(select); logFinish("add statistics sorting"); } } if (document.querySelector("body > div.page-container > div.header > div.header-bottom > div")) { document.querySelector("body > div.page-container > div.header > div.header-bottom > div").innerHTML += `
`; logFinish("add ii algorithms button"); document.querySelector("body > div.page-container > div.header > div.header-bottom > div > ul:nth-child(1) > li > a").href = "/"; document.querySelector("body > div.page-container > div.header > div.header-bottom > div > ul:nth-child(2) > li > a").href = "/Training.do"; document.querySelector("body > div.page-container > div.header > div.header-bottom > div > ul:nth-child(2) > li > a").className = ""; document.querySelectorAll("div.main-content > div > div > table > tbody > tr > td:nth-child(2) > a").forEach(e => void (e.target = "_blank")); logFinish("make task links open in another window"); if (document.URL.includes("/Training.do")) { var solved = 0, total = 0, progbar = document.createElement("div"); document.querySelectorAll("body > div.page-container > div.main > div.main-content > div > div > table > tbody > tr").forEach(elm => { if (elm.children.length < 2 || elm.children[0].nodeName.toUpperCase() == "TH") return; if (elm.querySelector("td.solved")) solved++; total++; }); progbar.className = "progbar"; progbar.innerHTML = `ONLINE_JUDGE
`;
setTimeout(() => document.querySelector(".ojtxt:last-child").remove(), 3000);
}
}, 500);
logFinish("#define ONLINE_JUDGE");
}
}
(document.querySelector(".footer") ?? {}).innerHTML += `MENDO.MK Enhancement 🎨 Changelog
`; window.toggleTheme = () => { localStorage.setItem("mendo-mk-enhancement-theme", localStorage.getItem("mendo-mk-enhancement-theme") == "dark" ? "light" : "dark"); location.reload(); }; let dmodebtn = document.createElement("div"); dmodebtn.className = `bi bi-${localStorage.getItem("mendo-mk-enhancement-theme") == "dark" ? "moon" : "sun"}-fill`; document.body.appendChild(dmodebtn); dmodebtn.style = "width:30px;height:30px;font-size:30px;position:fixed;bottom:0;right:0;margin:10px;cursor:pointer;color:#89AAD6"; dmodebtn.setAttribute("onclick", "toggleTheme()"); logFinish("dark mode button"); if (/^https?:\/\/mendo\.mk\/.*?User_Submission.do\?/.test(document.URL) && !document.URL.includes("ctest=true")) { let ok = false; for (let elm of document.querySelectorAll("img")) { if (elm.src.includes("loadingAnimation")) { ok = true; break; } } function checkForCinematic() { let usubTBody = document.querySelector("div.main-content > div > div > table:nth-child(6) > tbody"); if (!ok) { taskSolveCinematic(0, true); return; } if (!usubTBody) { requestAnimationFrame(checkForCinematic); } else if (usubTBody.querySelectorAll("tr td.correct:first-child").length + 1 >= usubTBody.querySelectorAll("tr").length) taskSolveCinematic(1, true); else taskSolveCinematic(2, true); } checkForCinematic(); logFinish("task solve cinematic setup"); } if (!isNaN(parseInt(document.querySelector("#LoginForm>fieldset>p:last-child>a")?.innerText))) { let elm = document.createElement("p"); let tasks = parseInt(document.querySelector("#LoginForm>fieldset>p:last-child>a").innerText); let ach = getAchievements(); let allread = true, lecturepage = false; for (let e of document.querySelectorAll(".training-content td:last-child")) { if (e.innerText.includes(localize("lecture", "предавање"))) { lecturepage = true; if (!e.classList.contains("solved")) { allread = false; break; } } } if (tasks >= 10) addAchievement("task", 0); if (tasks >= 25) addAchievement("task", 1); if (tasks >= 50) addAchievement("task", 2); if (tasks >= 100) addAchievement("task", 3); if (tasks >= 150) addAchievement("task", 4); if (tasks >= 200) addAchievement("task", 5); if (tasks >= 250) addAchievement("task", 6); if (tasks >= 300) addAchievement("task", 7); if (tasks >= 350) addAchievement("task", 8); if (tasks >= 400) addAchievement("task", 9); if (tasks >= 450) addAchievement("task", 10); if (tasks >= 500) addAchievement("task", 11); if (allread && lecturepage && (document.URL.endsWith("/Training.do") || document.URL.endsWith("/Training.do?cid=0") || document.URL.endsWith("/Training.do?cid=4"))) addAchievement("readlec", 0); if (allread && lecturepage && document.URL.endsWith("/Training.do?cid=6") && ach.readlec === 0) addAchievement("readlec", 1); ach = getAchievements(); elm.innerHTML = `${localize("Achievements", "Постигнувања")}: