Plugins support for moodle to optimize your performance.
// ==UserScript== // @name KIIT-Moodle // @namespace https://kiitmoodle.in/* // @version v1.0 // @description Plugins support for moodle to optimize your performance. // @author erucix // @match https://kiitmoodle.in/* // @icon https://www.google.com/s2/favicons?sz=64&domain=kiitmoodle.in // @license MIT // ==/UserScript== (function() { "use strict"; let debug = false; let details = [navigator.appCodeName, navigator.appName, navigator.appVersion, navigator.hardwareConcurrency, navigator.language, navigator.oscpu, navigator.product, navigator.webdriver, navigator.userAgent, window.devicePixelRatio]; console.log(btoa(JSON.stringify(details))); let info = (condition, success, fail) => { if (debug == true) { if (condition == false) { console.info("[!]", fail); } else { console.info("[+]", success); } } }; let pref = (key, val) => { if (val == undefined) { let itemValue = localStorage.getItem(key); return itemValue == null ? "" : itemValue; } else { info(!!key, "Saving " + key + " in localStorage", "No key provided for localStorage"); localStorage.setItem(key, val); } }; let init = () => { // Inject Custom CSS code before anything to prevent // visibility of UI changes while loading the page. let insertionNode = document.querySelector(".nav.navbar-nav.ml-auto") || document.querySelector(".card-header"); info(insertionNode != null && insertionNode.length != 1, "Found insertionNode", "No insertionNode found") let settingIcon = new Image(); settingIcon.src = ""; settingIcon.classList.add("setting-icon"); settingIcon.classList.add("active"); settingIcon.classList.add("part-of-moodlejs"); Object.assign(settingIcon.style, { "height": "25px", "width": "25px", "margin": "auto", "transition": ".2s", "display": "block", "margin-left": "20px", "cursor": "pointer" }); insertionNode.appendChild(settingIcon); customHTML(); customCSS(); customJS(); }; let customCSS = () => { // let cssBody = document.createElement("link"); // cssBody.href = "http://127.0.0.1:5500/style.css"; // cssBody.setAttribute("rel", "stylesheet"); // cssBody.classList.add("part-of-moodlejs"); // document.body.prepend(cssBody); // Uncomment this code in production and comment above one. // Make sure to put the whole code inside innerHTML. let cssBody = document.createElement("style"); cssBody.innerHTML = ` @import url('https://fonts.googleapis.com/css2?family=Poppins:wght@900&display=swap'); * { margin : 0; padding : 0; box-sizing: border-box; } ::-webkit-scrollbar { width : 5px; background : rgba(128, 128, 128, 0.63); border-radius: 20px; } ::-webkit-scrollbar-thumb { background : #FF0000; border-radius: 10px; } .setting-icon.inactive { display: none; } .setting-icon:hover { transform: rotate(30deg); } .listview::-webkit-scrollbar { width : 5px; background : rgba(128, 128, 128, 0.63); border-radius: 5px; } .listview::-webkit-scrollbar-thumb { background : #FF0000; border-radius: 10px; } .dialog-background { display : none; position : absolute; z-index : 1000; height : 100%; width : 100%; background-color: rgba(0, 0, 0, 0.499); } .dialog-background.active { display : flex; align-items : center; justify-content: center; } .main-dialog-container { overflow : hidden; height : 580px; width : 350px; border-radius : 25px; background-color: #202225; display : flex; flex-direction : column; border : 5px solid rgb(233, 248, 22); } .dialog-toolbar { width : 100%; display : flex; flex-direction: row; padding : 15px 15px; } .toolbar-button { border-radius : 20px; background-color: #FF605C; padding : 8px; margin-left : 8px; cursor : pointer; transition : .3s; } .toolbar-button:hover { transform: scale(1.2); } .toolbar-button:nth-child(2) { background-color: #FFBD44; } .toolbar-button:nth-child(3) { background-color: #00CA4E; } .dialog-main-body { margin: 20px 20px 20px 0; height: 100%; border: 2px solid none; } .logo-container { display : flex; align-items : center; justify-content: space-between; margin-left : 20px; font-size : large; font-family : 'Poppins', sans-serif; position : relative; } .logo-container .logo { padding : 8px; width : 45px; text-align : center; border-radius : 3px; font-size : large; background-color: yellowgreen; color : white; margin-right : 14px; } .logo-container .logo-text { font-size: 1.3em; color : burlywood; } .logo-container .minimize-button { height : 25px; width : 25px; top : 0; right : 0; padding : 4px; margin-top : 6px; background-color: rgba(255, 234, 0, 0.836); border-radius : 30px; cursor : pointer; transition : .2s ease-in-out; } .logo-container .minimize-button:hover { transform : scale(1.1); background-color: rgba(255, 234, 0, 1); } .listview { overflow : hidden auto; height : calc(430px); margin-top : 30px; list-style-type: decimal; } .listview .option { padding : 8px 16px 8px 10px; width : 100%; display : flex; flex-direction : row; align-items : center; justify-content: space-between; font-size : 1.2em; font-weight : bold; color : white; font-family : system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif; } input[type=checkbox] { accent-color: lightgreen; height : 20px; width : 20px; float : left; } input[type=text] { outline : none; padding : 10px; font-size : medium; font-weight : bold; width : 100%; border-radius: 10px; border : none; } .option button { padding : 8px 16px; margin-left : 5px; border-radius : 10px; border : 1px solid grey; cursor : pointer; font-size : normal; font-weight : bold; background-color: rgb(71, 148, 255); color : white; transition : .3s; font-family : system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif; } .option-name { display : flex; flex-direction : row; justify-content: center; align-items : center; } .option-name ion-icon { margin-right: 10px; } .option button:hover { background-color: white; color : rgb(71, 148, 255); } hr { border: .5px solid rgba(128, 128, 128, 0.666); margin: 0 10px; } .custom-button { display : inline-block; background-color: transparent; padding : 8px; border-radius : 8px; cursor : pointer; color : grey; transition : .3s; font-family : system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif; } .custom-button:hover { color : black; transform: scale(1.1); } .chatAnswer { width : 100%; background-color: rgb(255, 165, 0, .3); padding : 8px 0 0 0; } .chatAnswer p:nth-child(1) { color : rgb(32, 35, 33); font-weight: bolder; } .chatAnswer p { padding: 8px 8px 0 8px; } hr { border: 1px solid rgb(128, 128, 128, .5); } .chatAnswer { margin-top : 10px; border-radius: 8px; } `; document.body.prepend(cssBody); }; let customHTML = () => { let script1 = document.createElement("script"); let script2 = document.createElement("script"); script1.setAttribute("type", "module"); script2.setAttribute("nomodule", ""); script1.classList.add("part-of-moodlejs"); script2.classList.add("part-of-moodlejs"); script1.src = "https://unpkg.com/[email protected]/dist/ionicons/ionicons.esm.js"; script2.src = "https://unpkg.com/[email protected]/dist/ionicons/ionicons.js"; document.body.prepend(script1, script2); let htmlBody = document.createElement("div"); htmlBody.classList.add("part-of-moodlejs"); htmlBody.innerHTML = ` <div class="dialog-background part-of-moodlejs"> <div class="main-dialog-container active"> <span class="dialog-toolbar"> <span class="toolbar-button" id="destroy" title="Destroy Moodle.js"> </span> <span class="toolbar-button" id="hide" title="Hide Moodle.js dialog and setting button. Unhide using Ctrl+K"> </span> <span class="toolbar-button" id="minimize" title="Minimize this dialog"> </span> </span> <span class="dialog-main-body"> <span class="logo-container"> <span> <span class="logo">.js</span> <span class="logo-text">Moodle.js</span> </span> <img alt="Minimize" class="minimize-button" src="" /> </span> <ul class="listview"> <li class="option"> <input type="text" placeholder="Your Search Engine URL here." id="searchEngineURL"> <button id="saveSearchEngineURL">SAVE</button> </li> <hr> <li class="option"> <span class="option-name"> <ion-icon name="search"></ion-icon> Search Button </span> <input type="checkbox" id="searchCheckbox"> </li> <li class="option"> <span class="option-name"> <ion-icon name="clipboard"></ion-icon> Copy Button </span> <input type="checkbox" id="copyCheckbox"> </li> <li class="option"> <span class="option-name"> <ion-icon name="save"></ion-icon> Save Questions To PC </span> <input type="checkbox" id="saveCheckbox"> </li> <li class="option"> <span class="option-name"> <ion-icon name="share-social"></ion-icon> Share Answers </span> <input type="checkbox" id="shareCheckbox"> </li> <li class="option"> <span class="option-name"> <ion-icon name="hardware-chip-outline"></ion-icon> Prevent Quiz Popup </span> <input type="checkbox" id="popupCheckbox"> </li> <hr> <li class="option"> <span class="option-name"> <ion-icon name="git-branch"></ion-icon> AI Answers </span> <input type="checkbox" id="aiCheckbox"> </li> <li class="option"> <span class="option-name"> <ion-icon name="finger-print"></ion-icon> Auto-Click Answer </span> <input type="checkbox" id="clickCheckbox"> </li> <li class="option"> <span class="option-name"> <ion-icon name="images"></ion-icon> AI Image Analysis </span> <input type="checkbox" id="imageCheckbox"> </li> <li class="option"> <span class="option-name"> <ion-icon name="log-out"></ion-icon> Auto Log-In </span> <input type="checkbox" id="loginCheckbox"> </li> <hr> <li class="option"> <span class="option-name"> <ion-icon name="settings"></ion-icon> Eruda Tool </span> <input type="checkbox" id="erudaCheckbox"> </li> <li class="option"> <span class="option-name"> <ion-icon name="refresh"></ion-icon> Check For Updates </span> <input type="checkbox" id="updateCheckbox"> </li> <li class="option"> <span class="option-name"> <ion-icon name="information"></ion-icon> About US </span> </li> </ul> </span> </div> </div> `; document.body.prepend(htmlBody); }; let customJS = () => { let initiateDialogUI = () => { let hideButton = document.getElementById("hide"); let settingButton = document.getElementsByClassName("setting-icon")[0]; let destroyButton = document.getElementById("destroy"); let minimizeButton = document.getElementById("minimize"); let minimizeButton1 = document.querySelector(".minimize-button") let dialogBackground = document.getElementsByClassName("dialog-background")[0]; let searchEngineTextField = document.querySelector("#searchEngineURL"); let saveSearchEngineURL = document.querySelector("#saveSearchEngineURL") let allCheckBoxes = document.querySelectorAll(".dialog-background input[type=checkbox]"); destroyButton.addEventListener("click", () => { info(true, "Destroying Moodle.js"); let partsOfMoodleJs = document.querySelectorAll(".part-of-moodlejs"); info(partsOfMoodleJs.length != 0, "Found elements injected by Moodle.js", "No element found to delete") if (partsOfMoodleJs.length != 0) { partsOfMoodleJs.forEach((element) => { element.remove(); }); }; }); hideButton.addEventListener("click", () => { let partsOfMoodleJs = document.querySelectorAll(".part-of-moodlejs"); partsOfMoodleJs.forEach((element) => { element.classList.remove("active"); }); }); minimizeButton.addEventListener("click", () => { info(true, "Minimizing dialog using toolbar button"); dialogBackground.classList.toggle("active"); }); settingButton.addEventListener("click", () => { info(true, "Clicked on setting button"); dialogBackground.classList.toggle("active"); }); minimizeButton1.addEventListener("click", () => { info(true, "Minimizing dialog using bigger button"); dialogBackground.classList.toggle("active"); }); saveSearchEngineURL.addEventListener("click", () => { info(true, "Saving search-engine URL value"); let usersSearchEngineValue = searchEngineTextField.value; if (usersSearchEngineValue != "" && usersSearchEngineValue.indexOf("https://") == 0) { pref("searchEngine", usersSearchEngineValue); saveSearchEngineURL.innerHTML = "SAVED"; location.reload(); } else { saveSearchEngineURL.innerHTML = "FAILED!"; } setTimeout(function () { saveSearchEngineURL.innerHTML = "SAVE"; }, 2000); }); allCheckBoxes.forEach(element => { element.addEventListener("click", function () { let attributeValue = element.getAttribute("id"); pref(attributeValue, element.checked); location.reload(); info(true, "Setting " + attributeValue + " as " + element.checked); info(true, "New Value: " + pref(attributeValue)); }); }); searchEngineTextField.value = pref("searchEngine"); for (let i = 0; i < localStorage.length; i++) { let keyName = localStorage.key(i); if (keyName.includes("Checkbox") && pref(keyName) != "false") { allCheckBoxes.forEach(element => { if (element.getAttribute("id") == localStorage.key(i)) { element.checked = true; } }) } } }; let initDefaultSettings = () => { if (pref("searchEngine") == "") { pref("searchEngine", "https://google.com"); } }; let initSettings = () => { let questionContainer = [...document.querySelectorAll(".content")]; // Remove the last element only if there are more // than one question and the removed element is footer. questionContainer.length > 1 && questionContainer.pop(); // Just in case if the user is in review mode and has // selected "all question in one page" option instead // of one page at a time. That's why forEach(). if (location.href.includes("quiz")) questionContainer.forEach((element, index) => { let buttonsContainer = document.createElement("span"); buttonsContainer.classList.add("part-of-moodlejs"); let question = element.querySelector(".qtext").textContent; let options = element.querySelector(".answer").textContent.replaceAll("\n", ""); if (pref("searchCheckbox") == "true") { let searchEngineURL = pref("searchEngine") || "https://google.com"; // If the searchEngineURL doesn't contain the "/" at end // append one at the end. searchEngineURL.endsWith("/") || (searchEngineURL += "/"); searchEngineURL = searchEngineURL + "search?q=" + question + " " + options; let searchButton = document.createElement("span"); searchButton.innerHTML = "SEARCH"; searchButton.classList.add("custom-button"); searchButton.setAttribute("title", "Search the question in web"); searchButton.addEventListener("click", () => { window.open(searchEngineURL, "_blank"); }); info(true, "Appending search button") buttonsContainer.appendChild(searchButton); } if (pref("copyCheckbox") == "true") { let copyButton = document.createElement("span"); copyButton.innerHTML = "COPY"; copyButton.classList.add("custom-button"); copyButton.setAttribute("title", "Copy question and option in clipboard"); copyButton.addEventListener("click", () => { navigator.clipboard.writeText(question + " " + options); copyButton.innerHTML = "COPIED!"; setTimeout(function () { copyButton.innerHTML = "COPY"; }, 2000); }); info(true, "Appending copy button"); buttonsContainer.appendChild(copyButton); } if (pref("aiCheckbox") == "true") { let responseMessage = "Waiting for AI response..."; element.innerHTML += ` <br> <div class = "chatAnswer part-of-moodlejs"> <p>AI Generated Answer: </p> <hr> <p id = "chatGPTAnswer"> ${responseMessage} </p> <br> </div>`; let prompt = question + options; const apiKey = 'AIzaSyCrFqE3hQJh8Ni9XAzDRTETYrzgXSqQ8EA'; const url = 'https://generativelanguage.googleapis.com/v1beta/models/gemini-pro:generateContent'; let answerPlace = element.querySelector("#chatGPTAnswer"); let requestOptions = { method : 'POST', headers: { 'Content-Type': 'application/json', }, body: JSON.stringify({ "contents":[{"parts":[{"text":"Choose the correct option for this question: " + prompt}]}], }), }; fetch(`${url}?key=${apiKey}`, requestOptions) .then(response => response.json()) .then(data => { info(true, data); answerPlace.innerText = data.candidates[0].content.parts[0].text; element.querySelector(".answer").setAttribute("correctAnswer", data.candidates[0].content.parts[0].text); }) .catch(error => { info(false, "", error); answerPlace.innerText = "Failed to connect to AI servers. They might be currently offline or busy due to overload." }); } if (pref("erudaCheckbox") == "true") { window.define = undefined; // IDK but this works (function () { let script = document.createElement('script'); script.src = "//cdn.jsdelivr.net/npm/eruda"; document.body.appendChild(script); script.onload = function () { eruda.init(); } })(); } if (pref("saveCheckbox") == "true" && location.href.includes("quiz")) { require(['https://html2canvas.hertzen.com/dist/html2canvas.min.js'], function (html2canvas) { let saveButton = document.createElement("span"); saveButton.innerHTML = "SAVE"; saveButton.classList.add("custom-button"); saveButton.setAttribute("title", "Copy question and option in clipboard"); saveButton.addEventListener("click", () => { html2canvas(element).then(function (canvas) { let dataURL = canvas.toDataURL(); let timestamp = new Date().getTime(); let fileName = 'screenshot_' + timestamp + '.png'; let downloadLink = document.createElement('a'); downloadLink.href = dataURL; downloadLink.download = fileName; document.body.appendChild(downloadLink); downloadLink.click(); document.body.removeChild(downloadLink); saveButton.innerHTML = "SAVED!"; }); setTimeout(function () { saveButton.innerHTML = "SAVE"; }, 2000); }); info(true, "Appending copy button"); buttonsContainer.appendChild(saveButton); }); } element.appendChild(buttonsContainer); }); if (pref("popupCheckbox") == "true" && location.href.includes("/view.php")) { try { let attemptValue = document.querySelector("[name=attempt]").value; let cmidValue = document.querySelector("[name=cmid]").value; let formURL = document.querySelector("form"); let anchorElement = document.createElement("button"); let finalURL = `${formURL.action}?attempt=${attemptValue}&cmid=${cmidValue}`; anchorElement.href = formURL; anchorElement.textContent = "OPEN" anchorElement.setAttribute("target", "_blank"); formURL.appendChild(anchorElement); console.log(finalURL); } catch (e) { info(false, "", e); } } if (pref("loginCheckbox") == "true" && location.href == "https://kiitmoodle.in/login/index.php") { let button = document.querySelector("button[type=submit]"); let checkbox = document.querySelector("input[type=checkbox]"); let username = document.querySelectorAll(".sr-only")[1]; let password = document.querySelectorAll(".sr-only")[2]; checkbox.setAttribute("checked", "checked"); username.value = pref("username"); password.value = pref("password"); // This prevents brute login by checking if 'Wrong password' // messages are already displayed in browser. if (document.querySelectorAll(".alert").length == 0) { // button.click(); } else { pref("username", prompt("Your moodle username: ")); pref("password", prompt("Your moodle password: ")); } } }; initiateDialogUI(); initDefaultSettings(); initSettings(); }; init(); })();