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