KIIT-Moodle

Plugins support for moodle to optimize your performance.

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==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();
})();