// ==UserScript== // @name Formify // @version 4.1 // @description Let AI solve your google forms // @author rohitaryal // @license MIT // @grant GM_addElement // @grant GM_addStyle // @grant unsafeWindow // @namespace https://docs.google.com/ // @match https://docs.google.com/forms/* // @icon https://www.google.com/s2/favicons?sz=64&domain=docs.google.com // @downloadURL https://update.greasyfork.icu/scripts/480209/Formify.user.js // @updateURL https://update.greasyfork.icu/scripts/480209/Formify.meta.js // ==/UserScript== const css = `body.hidden { overflow: hidden; } .dialog-container * { margin: 0; padding: 0; font-family: system-ui; box-sizing: border-box; } .dialog-container { position: fixed; top: 0; left: 0; height: 100vh; width: 100vw; background-color: rgba(0, 0, 0, 0.836); z-index: 999; display: none; } .dialog-container .dialog { z-index: 1000; position: relative; top: 50%; left: 50%; transform: translate(-50%, -50%); height: 80%; width: 30rem; background-color: white; border-radius: .5rem; overflow: hidden; display: flex; align-items: center; flex-direction: column; } .dialog-container.active { display: block; } .dialog-container .dialog .formify-header { display: flex; width: 100%; padding: .5rem 1rem .5rem .5rem; gap: 1rem; align-items: center; font-size: xx-large; font-family: system-ui; border-bottom: 4px solid rgba(0, 0, 255, 0.45); } .dialog-container .dialog .formify-header .formify-text { flex: 1; } .dialog-container .dialog .formify-header .formify-text b { font-weight: 600; font-size: 1.2em; background-clip: text; color: transparent; background-image: linear-gradient(to right, #159957, #155799, rgb(81, 81, 255)); animation: animateheader 2s linear infinite; } @keyframes animateheader { 0%, 100% { filter: hue-rotate(0deg); } 100% { filter: hue-rotate(360deg); } } .dialog-container .dialog .formify-header .close { cursor: pointer; transition-duration: .2s; } .dialog-container .dialog .formify-header .close:hover { opacity: .5; } .dialog-container .dialog .formify-header .close { font-size: 2.5rem; } .dialog-container .dialog .formify-header .close:active { transform: scale(1.1); } .dialog-container .dialog .form-body { flex: 1; width: 100%; overflow-y: auto; } .dialog-container .dialog .form-body ul { list-style-type: none; } .dialog-container .dialog .form-body ul li { font-size: large; padding: 1rem; display: flex; align-items: center; border-bottom: 1px solid rgba(0, 0, 255, 0.192); } .dialog-container .dialog .form-body ul li code { background-color: rgba(128, 128, 128, 0.084); padding: .2em; border-radius: 5px; color: rgb(21, 21, 21); } .dialog-container .dialog .form-body ul label { flex: 1; font-size: 1.2rem; padding: .5rem 0; } .dialog-container .dialog .form-body ul input, .dialog-container .dialog .form-body ul select { outline: none; border: 2px solid rgba(0, 0, 255, 0.192); font-size: 1.2rem; padding: .5rem; width: 60%; border-radius: .5rem; background-color: transparent; } .dialog-container .dialog .form-footer { width: 100%; display: flex; padding: .1rem .5rem; gap: .5rem; align-items: center; border-top: 2px solid rgba(0, 0, 0, 0.107); } .dialog-container .dialog .form-footer img { cursor: pointer; text-decoration: none; } .dialog-container .dialog .form-footer a { text-decoration: none; color: #155799; } .ai-container { overflow: hidden; display: flex; flex-direction: column; height: fit-content; margin: .5rem; border-radius: 1rem; box-shadow: rgba(0, 0, 0, 0.15) 0px 2px 8px; } .ai-container.inactive { display: none; } .ai-container .container-header { display: flex; padding: .5rem 1rem; align-items: center; background-color: rgba(0, 0, 255, 0.192); justify-content: space-between; border-bottom: 1px solid rgba(0, 0, 0, 0.201); } .ai-container .container-header .buttons button { border: none; cursor: pointer; padding: .5rem; border-radius: .3rem; background-color: transparent; } .ai-container .container-header .buttons button:hover { color: white; background-color: rgba(0, 0, 0, 0.322); } .ai-container .container-body { padding: 1rem; color: rgb(20, 20, 34); } .aiChatDialog { position: fixed; /* Stay in place on the screen */ top: 0; /* stick to the top */ right: 0; /* stick to the right */ transform: translate(0,0); /* Reset transform properties, crucial */ background-color: #f9f9f9; border: 1px solid #ccc; box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1); width: 300px; height: 400px; display: none; flex-direction: column; border-radius: 5px; overflow: hidden; z-index: 1000; left: auto; /* Add this to ensure it doesn't try to be on the left */ } .dialogHeader { background-color: #ddd; padding: 10px; cursor: move; display: flex; justify-content: space-between; align-items: center; } .closeButton { cursor: pointer; font-size: 16px; } .chatHistory { flex-grow: 1; padding: 10px; overflow-y: auto; } .chatInputArea { padding: 10px; border-top: 1px solid #ccc; display: flex; outline: none; align-items: center; } .messageInput { flex-grow: 1; padding: 8px; border: 1px solid #ccc; border-radius: 4px; margin-right: 5px; } .sendButton { padding: 8px 12px; background-color: #4CAF50; color: white; border: none; border-radius: 4px; cursor: pointer; } .message { margin-bottom: 5px; padding: 8px; border-radius: 5px; } .userMessage { background-color: #DCF8C6; text-align: right; } .aiMessage { background-color: #ECE5DD; text-align: left; } @media(max-width: 600px) { .dialog-container .dialog { width: 25rem; } } `; const js = `const overlay = document.querySelector(".dialog-container"); const dialog = document.querySelector(".dialog-container .dialog"); const closeButton = document.querySelector(".dialog-container .dialog .formify-header .close"); const apiField = document.querySelector('.dialog-container .dialog input#apikey'); const modelSelect = document.querySelector('.dialog-container .dialog select#ai-model'); const searchEngineSelect = document.querySelector('.dialog-container .dialog select#search-engine'); const customPromptField = document.querySelector('.dialog-container .dialog input#custom-prompt'); const setItem = (key, value) => { const storage = localStorage.getItem('formify'); const parsedStorage = JSON.parse(storage || '{}'); parsedStorage[key] = value; localStorage.setItem('formify', JSON.stringify(parsedStorage)); } const getItem = (key) => { const storage = localStorage.getItem('formify'); const parsedStorage = JSON.parse(storage || '{}'); return parsedStorage[key]; } const apiKey = getItem('apiKey'); const model = getItem('model'); const searchEngine = getItem('searchURL'); const customPrompt = getItem('customPrompt'); if (!model) setItem('model', "gemini-2.0-flash-lite"); if (!searchEngine) setItem('searchURL', "https://www.google.com/search?q="); if (!customPrompt) setItem("customPrompt", "Your answer must include one of the options i provided here and please answer shortly and no markdown like response allowed, plain text and write the full answer, provide short description if possible"); // Re-assigning values to the fields apiField.value = getItem('apiKey'); modelSelect.value = getItem('model'); searchEngineSelect.value = getItem('searchURL'); customPromptField.value = getItem('customPrompt'); overlay.addEventListener('click', (e) => { if ( e.target === overlay || e.target === closeButton ) { overlay.classList.toggle("active"); } }); apiField.addEventListener('input', (e) => { const apiKey = e.target.value; setItem('apiKey', apiKey); }); modelSelect.addEventListener('change', (e) => { const selectedModel = e.target.value; setItem('model', selectedModel); }); searchEngineSelect.addEventListener('change', (e) => { const selectedEngine = e.target.value; setItem('searchURL', selectedEngine); }); customPromptField.addEventListener('input', (e) => { const promptValue = e.target.value; setItem('customPrompt', promptValue); });`; // src/utils/parsers/HTMLFormParser.ts var formHeaderParser = (form) => { const formContentContainer = form.querySelector(".lrKTG"); if (!formContentContainer) throw new Error("[!] Form content not found. Are you sure you are providing correct form?"); const formHeader = formContentContainer.querySelector(".m7w29c.O8VmIc.tIvQIf"); if (!formHeader) console.warn("[W] Form header was not found"); const formTitleContainer = formHeader?.querySelector(".ahS2Le"); const formDescriptionContainer = formHeader?.querySelector(".cBGGJ.OIC90c"); return { formTitle: formTitleContainer?.textContent || document.title, formDescription: formDescriptionContainer?.textContent || "" }; }; var formQuestionParser = (form) => { if (!(form instanceof HTMLFormElement)) throw new Error("[!] I strictly require HTMLFormElement to parse header"); const questionContainer = form.querySelector(".o3Dpx[role='list']"); if (!questionContainer) throw new Error("Question container is missing. Are you sure you are providing correct form?"); const questionList = questionContainer.querySelectorAll(".Qr7Oae[role='listitem']"); if (!questionList.length) console.warn("[W] No questions found."); const parsedQuestions = [...questionList]?.map((questionContainer2) => { const infoContainerDiv = questionContainer2.querySelector("div[jsmodel='CP1oW']"); const dataParams = infoContainerDiv?.getAttribute("data-params"); const betterDataParams = dataParams?.replace("%.@.", "[").replace(/"/g, "'"); const question = JSON.parse(betterDataParams || "[]")[0]; if (!question) { return { title: "", moreInfo: "", type: -1, id: "", required: false, options: [] }; } const questionTitle = question[1]; const extraInformation = question[9] || null; const questionType = question[3]; const submitID = question[4][0][0]; const isRequiredQuestion = question[4][0][2]; const options = question[4][0][1]?.map((option) => { return { value: option[0], moreInfo: option[5] || null }; }); return { title: questionTitle, moreInfo: extraInformation, type: questionType, id: submitID, required: isRequiredQuestion, options }; }); return parsedQuestions; }; var parse = () => { const form = document.querySelector("form#mG61Hd"); if (!form) { throw new Error("Form element not found"); } const { formDescription, formTitle } = formHeaderParser(form); const parsedQuestionList = formQuestionParser(form); return { title: formTitle, description: formDescription, questions: parsedQuestionList }; }; // src/utils/Utils.ts var groupedLog = (title, ...args) => { console.groupCollapsed(title); args.forEach((arg) => console.log(arg)); console.groupEnd(); }; // src/utils/StorageUtils.ts var getItem = (key) => { const storage = localStorage.getItem("formify"); try { const parsedStorage = JSON.parse(storage || "{}"); return parsedStorage[key] || null; } catch (err) { groupedLog("Failed to parse JSON from localStorage", err); return null; } }; var setItem = (key, value) => { const storage = localStorage.getItem("formify"); let parsedStorage = {}; try { parsedStorage = JSON.parse(storage || "{}"); } catch (err) { groupedLog("Failed to parse JSON from localStorage", err); } parsedStorage[key] = value; localStorage.setItem("formify", JSON.stringify(parsedStorage)); }; // src/utils/NetworkUtils.ts var request = async (requestURL, requestOption = { method: "GET" }) => { if (requestOption.method == "GET" && requestOption.body) { groupedLog("Removing body from GET request.", requestURL, requestOption); delete requestOption.body; } try { const response = await fetch(requestURL, requestOption); const responseBody = await response.text(); if (response.status != 200) { groupedLog(`Server responded with status ${response.status}`, requestURL, requestOption, responseBody); } else { groupedLog("Server responded successfully", requestURL, requestOption, responseBody); } return { success: response.status == 200, response: responseBody, statusText: response.statusText }; } catch (err) { groupedLog("Failed to send request.", requestURL, requestOption, err); return { success: false, statusText: "ERROR", response: err instanceof Error ? err.message : String(err) }; } }; // src/utils/AIUtils.ts var getAIResponse = async (prompt2) => { const model = getItem("model") || "gemini-2.0-flash"; if (model == "gemini-2.0-flash" || model == "gemini-2.0-pro-experimental" || model == "gemini-2.0-flash-lite" || model == "gemini-2.0-pro-exp-02-05") { return getGeminiResponse(prompt2); } else { return "Model not supported for now: " + model; } }; var getGeminiResponse = async (prompt2) => { const model = getItem("model") || "gemini-2.0-flash"; const apiKey = getItem("apiKey") || ""; const response = await request(`https://generativelanguage.googleapis.com/v1beta/models/${model}:generateContent?key=${apiKey}`, { headers: new Headers({ "Content-Type": "application/json" }), method: "POST", body: JSON.stringify({ contents: [{ parts: [ { text: prompt2.prompt } ] }] }) }); if (!response.success) { return "Failed to fetch: " + response.statusText; } try { const parsedContent = JSON.parse(response.response); return parsedContent?.candidates?.[0]?.content?.parts?.[0]?.text; } catch (err) { return "Failed to parse response: " + err.message; } }; // src/utils/DOMUtils.ts var answerModal = ({ options, question, answer }) => { const div = document.createElement("div"); div.classList.add("ai-container"); const header = document.createElement("span"); header.classList.add("container-header"); const modelNameSpan = document.createElement("span"); modelNameSpan.classList.add("model-name"); modelNameSpan.textContent = `\uD83E\uDD95 ${getItem("model") || "gemini-2.0-flash"}`; const buttons = document.createElement("span"); buttons.classList.add("buttons"); const body = document.createElement("span"); body.classList.add("container-body"); body.textContent = answer; const option = options?.map((option2) => option2.value)?.join(" ") || ""; const buttonConfigs = [ { id: "copy", title: "Copy answer to clipboard", text: "Copy", onclick: (e) => { e.preventDefault(); navigator.clipboard.writeText(answer); e.target.textContent = "Copied"; setTimeout(() => { e.target.textContent = "Copy"; }, 2000); } }, { id: "regenerate", title: "Re-generate answer", text: "Re-generate", onclick: async (e) => { e.preventDefault(); body.textContent = "Re-generating answer... \uD83E\uDD95"; const response = await getAIResponse({ prompt: getItem("customPrompt") + ` ` + question + option }); body.textContent = response; } }, { id: "open-chat", title: "Open this question in chat", text: "Open in Chat", onclick: (e) => { e.preventDefault(); openAIChat(); sendMessage(getItem("customPrompt") + ` ` + question + option); } }, { id: "search", title: "Search this question", text: "Search", onclick: (e) => { const anchor = document.createElement("a"); anchor.href = (getItem("searchURL") || "https://www.google.com/search?q=") + question + option; anchor.target = "_blank"; anchor.click(); } } ]; buttonConfigs.forEach(({ id, title, text, onclick }) => { const button = document.createElement("button"); button.setAttribute("type", "button"); button.id = id; button.title = title; button.textContent = text; buttons.appendChild(button); button.addEventListener("click", onclick); }); header.appendChild(modelNameSpan); header.appendChild(buttons); div.appendChild(header); div.appendChild(body); return div; }; var toggleDialog = (force) => { const dialog = document.querySelector(".dialog-container"); if (!dialog) { groupedLog("Dialog container not found"); return; } if (force === true) { dialog.classList.remove("hidden"); } else if (force === false) { dialog.classList.add("active"); } else { dialog.classList.toggle("active"); } }; var toggleAnswers = (force) => { const aiResponse = document.querySelectorAll(".ai-container"); aiResponse.forEach((response) => { if (force === true) { response.classList.remove("inactive"); } else if (force === false) { response.classList.add("inactive"); } else { response.classList.toggle("inactive"); } }); }; var dialogWidth = "300px"; var dialogHeight = "400px"; var aiChatDialog = null; var chatHistory; var messageInput; var isDragging = false; var offsetX; var offsetY; function addMessage(text, isUser) { const messageElement = document.createElement("div"); messageElement.classList.add("message"); messageElement.classList.add(isUser ? "userMessage" : "aiMessage"); messageElement.textContent = text; chatHistory.appendChild(messageElement); chatHistory.scrollTop = chatHistory.scrollHeight; } async function sendMessage(msg = "") { const message = messageInput.value.trim() || msg; if (message) { addMessage(message, true); messageInput.value = ""; try { const aiResponse = await getAIResponse({ prompt: message }); addMessage(aiResponse, false); } catch (error) { addMessage("Error: Could not get AI response.", false); console.error("Error fetching AI response:", error); } } } function createAIChatDialog() { aiChatDialog = document.createElement("div"); aiChatDialog.id = "aiChatDialog"; aiChatDialog.style.width = dialogWidth; aiChatDialog.style.height = dialogHeight; const dialogHeader = document.createElement("div"); dialogHeader.id = "dialogHeader"; const headerText = document.createTextNode("Formify"); dialogHeader.appendChild(headerText); const closeButton = document.createElement("span"); closeButton.id = "closeButton"; const closeText = document.createTextNode("×"); closeButton.appendChild(closeText); chatHistory = document.createElement("div"); chatHistory.id = "chatHistory"; const chatInputArea = document.createElement("div"); chatInputArea.id = "chatInputArea"; messageInput = document.createElement("input"); messageInput.type = "text"; messageInput.id = "messageInput"; messageInput.placeholder = "Type your message..."; const sendButton = document.createElement("button"); sendButton.id = "sendButton"; const sendText = document.createTextNode("Send"); sendButton.appendChild(sendText); dialogHeader.appendChild(closeButton); chatInputArea.appendChild(messageInput); chatInputArea.appendChild(sendButton); aiChatDialog.appendChild(dialogHeader); aiChatDialog.appendChild(chatHistory); aiChatDialog.appendChild(chatInputArea); closeButton.addEventListener("click", closeAIChat); sendButton.addEventListener("click", () => sendMessage()); messageInput.addEventListener("keydown", (event) => { if (event.key === "Enter") { sendMessage(); } }); dialogHeader.addEventListener("mousedown", (e) => { isDragging = true; offsetX = e.clientX - aiChatDialog.offsetLeft; offsetY = e.clientY - aiChatDialog.offsetTop; }); document.addEventListener("mouseup", () => { isDragging = false; }); document.addEventListener("mousemove", (e) => { if (!isDragging) return; let newX = e.clientX - offsetX; let newY = e.clientY - offsetY; const maxX = window.innerWidth - aiChatDialog.offsetWidth; const maxY = window.innerHeight - aiChatDialog.offsetHeight; newX = Math.max(0, Math.min(newX, maxX)); newY = Math.max(0, Math.min(newY, maxY)); aiChatDialog.style.left = newX + "px"; aiChatDialog.style.top = newY + "px"; aiChatDialog.style.right = "auto"; }); aiChatDialog.classList.add("aiChatDialog"); dialogHeader.classList.add("dialogHeader"); closeButton.classList.add("closeButton"); chatHistory.classList.add("chatHistory"); chatInputArea.classList.add("chatInputArea"); messageInput.classList.add("messageInput"); sendButton.classList.add("sendButton"); document.body.appendChild(aiChatDialog); addMessage("Starting conversation...", false); return aiChatDialog; } function closeAIChat() { if (aiChatDialog) { aiChatDialog.style.display = "none"; } } function openAIChat() { if (!aiChatDialog) { createAIChatDialog(); } aiChatDialog.style.display = "flex"; } var ready = () => { const dialogContainer = document.createElement("div"); dialogContainer.className = "dialog-container"; const dialog = document.createElement("div"); dialog.className = "dialog"; const header = document.createElement("span"); header.className = "formify-header"; const logo = document.createElement("img"); logo.alt = "F is for Formify"; logo.height = 40; logo.src = ""; const headerText = document.createElement("span"); headerText.className = "formify-text"; const boldText = document.createElement("b"); boldText.innerText = "Formify"; headerText.appendChild(boldText); const closeBtn = document.createElement("span"); closeBtn.className = "close"; closeBtn.innerText = "×"; header.appendChild(logo); header.appendChild(headerText); header.appendChild(closeBtn); const formBody = document.createElement("span"); formBody.className = "form-body"; const ul = document.createElement("ul"); const listItems = [ { label: "API Key", type: "text", id: "apikey", placeholder: "Paste API key here" }, { label: "AI Model", type: "select", id: "ai-model", options: [ { value: "gemini-2.0-pro-exp-02-05", text: "Gemini 2.0 Pro Experimental" }, { value: "gemini-2.0-flash", text: "Gemini 2.0 Flash" }, { value: "gemini-2.0-flash-lite", text: "Gemini 2.0 Flash-Lite" }, { value: "gemini-1.5-pro", text: "Gemini 1.5 Pro" }, { value: "gpt-4.5-preview", text: "gpt-4.5-preview" }, { value: "gpt-4o-mini", text: "gpt-4o-mini" } ] }, { label: "Search Engine", type: "select", id: "search-engine", options: [ { value: "https://www.google.com/search?q=", text: "Google" }, { value: "https://chatgpt.com/?q=", text: "ChatGPT Search" }, { value: "https://www.bing.com/search?q=", text: "Bing" }, { value: "https://search.yahoo.com/search?p=", text: "Yahoo" }, { value: "https://duckduckgo.com/?q=", text: "DuckDuckGo" }, { value: "https://www.baidu.com/s?wd=", text: "Baidu" }, { value: "https://www.yandex.com/search/?text=", text: "Yandex" }, { value: "https://www.ecosia.org/search?q=", text: "Ecosia" }, { value: "https://www.ask.com/web?q=", text: "Ask" }, { value: "https://www.startpage.com/do/search?q=", text: "Startpage" }, { value: "https://search.brave.com/search?q=", text: "Brave Search" } ] }, { label: "Custom Prompt", type: "text", id: "custom-prompt", placeholder: "Custom prompt to feed the model" }, { label: "Dialog Shortcuts", type: "code", value: "ALT + K" }, { label: "AI Response hide/unhide", type: "code", value: "ALT + M" } ]; listItems.forEach((item) => { const li = document.createElement("li"); const label = document.createElement("label"); label.innerText = item.label; if (item.type === "text") { const input = document.createElement("input"); input.type = "text"; input.setAttribute("name", item.id || ""); input.setAttribute("id", item.id || ""); input.setAttribute("placeholder", item.placeholder || ""); li.appendChild(label); li.appendChild(input); } else if (item.type === "select") { const select = document.createElement("select"); select.setAttribute("name", item.id || ""); select.setAttribute("id", item.id || ""); item.options.forEach((optionData) => { const option = document.createElement("option"); option.setAttribute("value", optionData.value); option.textContent = optionData.text; select.appendChild(option); }); li.appendChild(label); li.appendChild(select); } else if (item.type === "code") { const code = document.createElement("code"); code.textContent = item.value || ""; li.appendChild(label); li.appendChild(code); } ul.appendChild(li); }); formBody.appendChild(ul); const formFooter = document.createElement("span"); formFooter.className = "form-footer"; const footerLinks = [ { href: "https://github.com/rohitaryal/formify", text: "Submit a bug ↗" }, { href: "https://www.youtube.com/watch?v=dQw4w9WgXcQ", text: "Hmm ↗" } ]; footerLinks.forEach((linkData) => { const link = document.createElement("a"); link.target = "_blank"; link.href = linkData.href; link.innerText = linkData.text; formFooter.appendChild(link); }); dialog.appendChild(header); dialog.appendChild(formBody); dialog.appendChild(formFooter); dialogContainer.appendChild(dialog); document.body.appendChild(dialogContainer); }; // src/index.ts (async function() { GM_addStyle(css); ready(); GM_addElement("script", { textContent: js }); let apiKey = getItem("apiKey"); if (!apiKey) { apiKey = prompt("Please paste your API key, you can generate free api key from: https://aistudio.google.com/apikey"); if (apiKey) { setItem("apiKey", apiKey); } else { alert("API key is required to interact with the AI model."); } } document.addEventListener("keydown", (e) => { if (e.altKey && e.key == "k" || e.key == "Escape") { toggleDialog(); } if (e.key == "m" && e.altKey) { toggleAnswers(); } }); let scrapedContent = parse(); const questionContainers = document.querySelectorAll(".Qr7Oae[role='listitem']"); for (let i = 0;i < questionContainers.length; i++) { const questionContainer = questionContainers[i]; const question = scrapedContent.questions[i]; const prompt2 = getItem("customPrompt") + ` ` + question.title + (question.options?.map((option) => option.value + ", ") || ""); const aiAnswer = await getAIResponse({ prompt: prompt2 }); const options = questionContainer.querySelectorAll("label"); for (const option of options || []) { const betterOptionText = option.textContent?.trim(); const betterAiAnswer = aiAnswer.trim(); if (betterAiAnswer.includes(betterOptionText)) { if (question.type == 2 || question.type == 4) { option.click(); if (question.type == 2) { break; } } } } const answer = answerModal({ question: question.title, options: question.options, answer: aiAnswer }); questionContainer.appendChild(answer); } })();