- // ==UserScript==
- // @name Gemini Keyboard Shortcuts
- // @namespace http://tampermonkey.net/
- // @version 1.2.5
- // @description This userscript enhances your Gemini experience by adding a wide range of keyboard shortcuts for streamlined navigation and interaction, as well as cleaning up Gemini's UI.
- // @license MIT
- // @author Henry Getz
- // @match https://gemini.google.com/u/*
- // @match https://gemini.google.com/app*
- // @icon 
- // @supportURL https://github.com/HenryGetz/GeminiPilot/issues
- // @grant none
- // @run-at document-start
- // ==/UserScript==
- /*
-
- #New Feature: URL Parameters!
-
- Empower your automation workflows! Directly open Gemini with pre-populated prompts by using query parameters in the URL (e.g., `https://gemini.google.com/app?q=YOURTESTPROMPT`).
-
-
- # Included Keyboard Shortcuts:
-
-
- ## Chat Management
-
- | Shortcut (Mac/Windows) | Action |
- |:--------------------------:|:--------------:|
- | ⌘/Ctrl + Shift + O | Open new chat |
- | ⌘/Ctrl + Shift + Backspace | Delete chat |
- | ⌘/Ctrl + Shift + F | Toggle sidebar |
- | ⌥/Alt + 1-9 | Go to nth chat |
- | ⌘/Ctrl + Shift + = | Next chat |
- | ⌘/Ctrl + Shift + – | Previous chat |
-
-
- ## Text Input and Editing
-
- | Shortcut (Mac/Windows) | Action |
- |:----------------------:|:-----------------------------:|
- | Shift + Esc | Focus chat input |
- | ⌘/Ctrl + Shift + E | Edit text |
- | ⌘/Ctrl + Shift + ; | Copy last code block |
- | ⌘/Ctrl + Shift + ' |Copy second-to-last code block |
- | ⌘/Ctrl + Shift + C | Copy response |
- | ⌘/Ctrl + Shift + K | Stop/start generation |
-
-
- ## Draft Navigation
-
- | Shortcut (Mac/Windows) | Action |
- |:----------------------:|:--------------------:|
- | ⌘/Ctrl + Shift + D | Generate more drafts |
- | ⌘/Ctrl + Shift + , | Next draft |
- | ⌘/Ctrl + Shift + . | Previous draft |
-
-
- ## Sharing and Linking
-
- | Shortcut (Mac/Windows) | Action |
- |:----------------------:|:-------------------------:|
- | ⌘/Ctrl + Shift + L | Copy prompt/response link |
- | ⌘/Ctrl + Shift + M | Copy chat link |
-
-
- ## Audio and File Shortcuts
-
- | Shortcut (Mac/Windows) | Action |
- |:----------------------:|:---------------------:|
- | ⌘/Ctrl + Shift + K | Stop/start generation |
- | ⌘/Ctrl + Shift + Y | Play/pause audio |
- | ⌘/Ctrl + Shift + S | Voice to text |
- | ⌘/Ctrl + O | Open file |
-
-
-
- */
-
- //With this false, it will copy from the response in the viewport.
-
- const assumeLastResponse = false;
-
- //This setting allows you to delete chats in succession, like browser tabs, instead of beign forced to go to a new one. Perfect if doing Gemini housekeeping
-
- const goToNextChatOnDelete = true;
-
-
-
- const hasQuery = window.location.href.includes("?q=");
- let url = new URL(window.location.href);
- let params = new URLSearchParams(url.search);
- let query = unescape(params.get('q'));
-
- window.onload = onLoad;
-
- function onLoad(){
- //This code makes the prompt take up the full width of the screen, and moves the heading
- let s = document.createElement("style");
- document.head.append(s);
- s.textContent = `
-
- .conversation-container, .input-area-container, .bottom-container {
- max-width: -webkit-fill-available !important;
- }
-
- .capabilities-disclaimer, #gbwa, .cdk-overlay-backdrop {
- display: none !important;
- }
-
- .code-block-decoration.footer, .code-block-decoration.header {
- user-select: none; /* Standard syntax */
- -webkit-user-select: none; /* WebKit (Safari, Chrome) browsers */
- -moz-user-select: none; /* Firefox */
- -ms-user-select: none; /* Internet Explorer/Edge */
-
- }
-
- .bottom-container {
- padding-bottom: 20px;
- }
-
- bard-mode-switcher {
- position: fixed;
- top: 0px;
- right: 64px;
- z-index: 1000;
- background: var(--bard-color-surface-container);
- border: solid var(--bard-color-surface-container) 4px;
- border-right: solid var(--bard-color-surface-container) 100px;
- transform: translate(100px, -4px);
- border-radius: 100px;
- box-shadow: 0 0 20px 12px rgba(var(--bard-color-main-container-background-rgb), 77%)
- }
-
- .mat-mdc-focus-indicator::before {
- border: none !important;
- }
-
- * > .conversation-container:first-child {
- border-top: solid transparent 60px !important;
- }
-
- `;
-
- const nums = ["first", "second", "third", "fourth", "fifth", "sixth", "seventh", "eighth", "ninth", "tenth"];
- const rapidClickDelayMS = 100;
- const capitalize = word => word.charAt(0).toUpperCase() + word.slice(1);
-
-
- //This code makes sure that the 'more chats' feature is selected without user interaction (so that you can select chats 6-9 with alt as well.)
-
- //This code also allows for query parameters in the URL.
-
- let showMoreClicked = false;
- let inputBarClicked = false;
- const observer = new MutationObserver((_, observer) => {
- const showMore = document.querySelector('[data-test-id="show-more-button"]');
- const inputBar = document.querySelector('.text-input-field');
- const textInput = document.querySelector('[aria-label="Enter a prompt here"]');
-
- if (showMore && !showMoreClicked) {
- showMoreClicked = true;
- simulateClick(showMore);
- }
- if (hasQuery && inputBar && !inputBarClicked) {
- if (textInput && !inputBarClicked) {
-
-
- inputBarClicked = true;
- console.log(query);
- params.delete('q');
- window.history.pushState(null,"",url.origin + url.pathname);
-
- setTimeout(function(){
- inputBar.click();
-
- setTimeout(function(){
- textInput.firstChild.remove();
- query = query.split("\n");
- for (let line of query) {
- let p = document.createElement("p");
- p.innerText = line;
- textInput.append(p);
- }
-
- //This waits to also change the url when the drafts generate. Google is weird and changes it back
- const observer = new MutationObserver((_, observer) => {
- let showDrafts = document.querySelector('[data-test-id="generate-more-drafts-button"]');
- if (showDrafts) {
- observer.disconnect();
-
- setTimeout(function(){
- url = new URL(window.location.href);
- params = new URLSearchParams(url.search);
- window.history.pushState(null,"",url.origin + url.pathname);
- }, 2000)
- }
- });
- observer.observe(document.body, {childList: true, subtree: true});
-
- setTimeout(function(){
- document.querySelector('[aria-label="Send message"]').click();
- }, rapidClickDelayMS)
- } ,rapidClickDelayMS)
- }, rapidClickDelayMS)
-
- }
- } else if (inputBar && !inputBarClicked) {
- console.log(hasQuery)
- inputBarClicked = true;
- setTimeout(() => inputBar.click(), rapidClickDelayMS)
- }
-
- if (showMoreClicked && inputBarClicked) {
- observer.disconnect();
- }
- });
- observer.observe(document.body, {childList: true, subtree: true});
-
- let c = null;
- function getLastElement(querySelector) {
- const containers = document.querySelectorAll('.conversation-container');
- c = containers[containers.length - 1];
- if (!assumeLastResponse) {
- let mostVisibleElement = null;
- let maxVisibleArea = 0;
-
- containers.forEach(container => {
- const rect = container.getBoundingClientRect();
- const viewportHeight = window.innerHeight;
-
- // Calculate visible area (only consider area within the viewport)
- const visibleArea = Math.max(0, Math.min(rect.bottom, viewportHeight) - Math.max(rect.top, 0));
-
- if (visibleArea > maxVisibleArea && visibleArea !== 0) {
- maxVisibleArea = visibleArea;
- mostVisibleElement = container;
- }
- });
- c = mostVisibleElement;
- }
- return c.querySelectorAll(querySelector)[c.querySelectorAll(querySelector).length - 1];
- }
-
- function copy(text) {
- const textarea = document.createElement('textarea');
- textarea.value = text;
- document.body.appendChild(textarea);
- textarea.select();
- document.execCommand('copy');
- document.body.removeChild(textarea);
- }
-
- function copyRichTextFromDiv(element) {
- const div = element;
-
- if (!div) {
- console.error("Div not found.");
- return;
- }
-
- document.querySelectorAll('.code-block-decoration.footer, .code-block-decoration.header, .table-footer').forEach(el => el.style.display = 'none');
-
- const selection = window.getSelection();
- const range = document.createRange();
- range.selectNodeContents(div);
- selection.removeAllRanges();
- selection.addRange(range);
-
- try {
- const successful = document.execCommand('copy');
- } catch (err) {
- console.error('Failed to copy rich text: ', err);
- }
-
- selection.removeAllRanges();
- setTimeout(function(){
- document.querySelectorAll('.code-block-decoration.footer, .code-block-decoration.header').forEach(el => el.style.display = '');
- }, rapidClickDelayMS)
-
- }
-
-
-
- function clearNotifications() {
- for (let ele of document.querySelectorAll(".gemini-key-notification")) {
- ele.remove();
- }
- }
-
- function notify(text) {
- clearNotifications();
- for (let ele of document.querySelectorAll(".gmat-mdc-dialog")) {
- ele.remove();
- }
-
- var div = document.createElement('div');
- div.classList.add("gemini-key-notification");
- div.innerText = text;
- let tDuration = 125;
- let nDuration = 3000;
- let tLeft = nDuration - tDuration;
- div.style.cssText = `position: absolute;bottom: 26px;left: 26px;font-family: var(--mdc-snackbar-supporting-text-font);font-size: var(--mdc-snackbar-supporting-text-size);font-weight: var(--mdc-snackbar-supporting-text-weight);line-height: var(--mdc-snackbar-supporting-text-line-height);color: var(--mdc-snackbar-supporting-text-color);border-radius: var(--mdc-snackbar-container-shape);background-color: var(--mdc-snackbar-container-color);z-index: 2147483647;padding: 16px;line-height: 20px;transition-property: opacity, scale;transition-duration: ${tDuration}ms;transform-origin: center;scale: 0.6;opacity: 0;`;
- document.body.append(div);
- setTimeout(function(){div.style.opacity = 1; div.style.scale = 1;}, rapidClickDelayMS)
- setTimeout(function(){
- div.style.opacity = 0;
- setTimeout(function(){div.remove()}, tDuration)
- }, tLeft);
- }
-
-
- function simulateClick(element) {
- element.click();
- }
-
- let draftIndex = 0;
- let googleDraftCount = 3;
- let waitOnGeneration = false;
-
- function changeDraft(direction) {
- let draftButtons = document.querySelectorAll(".draft-preview-button");
- if (!waitOnGeneration) {
- draftIndex = (draftIndex + direction + googleDraftCount) % googleDraftCount; // Ensure index stays within 0-2
- }
-
- if (!waitOnGeneration && draftButtons[draftIndex]) {
- simulateClick(draftButtons[draftIndex]);
- //notify(`${capitalize(nums[draftIndex])} draft`)
- } else if (!waitOnGeneration) {
- draftIndex = 0;
- draftIndex = (draftIndex + direction + googleDraftCount) % googleDraftCount;
- simulateClick(getLastElement('[data-test-id="generate-more-drafts-button"]'));
- notify(`Generating ${nums[draftIndex]} draft`)
- waitOnGeneration = true;
-
- const observer = new MutationObserver((_, observer) => {
- draftButtons = document.querySelectorAll(".draft-preview-button");
- if (draftButtons[draftIndex]) {
- observer.disconnect();
- setTimeout(function(){
- waitOnGeneration = false;
- simulateClick(draftButtons[draftIndex]);
- //notify(`${capitalize(nums[draftIndex])} draft`)
- },rapidClickDelayMS * 2)
- }
- });
- observer.observe(document.body, {childList: true, subtree: true});
- } else {
- notify("Waiting on generation");
- }
- }
-
- const nextDraft = () => changeDraft(1);
- const previousDraft = () => changeDraft(-1);
-
- let chatIndex = 0;
- let waitOnLoadingMore = false;
-
- function changeChat(direction) {
- chatIndex = Array.from(document.querySelectorAll('[data-test-id="conversation"]')).indexOf(document.querySelector('.selected[data-test-id="conversation"]'));
- let chatButtons = document.querySelectorAll('[data-test-id="conversation"]');
-
- if (!waitOnLoadingMore) {
- chatIndex = Math.max(0, chatIndex + direction);
- }
-
- if (!waitOnLoadingMore && chatButtons[chatIndex]) {
- simulateClick(chatButtons[chatIndex]);
- notify(`"${chatButtons[chatIndex].querySelector(".conversation-title").innerHTML.trim()}"`)
- } else if (!waitOnLoadingMore) {
- simulateClick(document.querySelector('[data-test-id="load-more-button"]'));
- notify(`Loading chats`)
- waitOnLoadingMore = true;
-
- const observer = new MutationObserver((_, observer) => {
- chatButtons = document.querySelectorAll('[data-test-id="conversation"]');
- if (chatButtons[chatIndex]) {
- observer.disconnect();
- setTimeout(function(){
- waitOnLoadingMore = false;
- simulateClick(chatButtons[chatIndex]);
- //notify(`${capitalize(nums[draftIndex])} draft`)
- notify(`"${chatButtons[chatIndex].querySelector(".conversation-title").innerHTML.trim()}"`)
- },rapidClickDelayMS * 2)
- }
- });
- observer.observe(document.body, {childList: true, subtree: true});
- } else {
- notify("Chats loading");
- }
- }
-
- const nextChat = () => changeChat(1);
- const previousChat = () => changeChat(-1);
-
-
-
- var isMac = /(Mac|iPhone|iPod|iPad)/i.test(navigator.platform);
-
- document.addEventListener('keydown', function(event) {
- // Check for Command or Control key
-
- if (event.shiftKey && event.key === "Escape") {
- simulateClick(document.querySelector('.text-input-field'));
- event.preventDefault();
- }
-
- let keyNumber = parseInt(event.code.replace("Digit",""));
- keyNumber = keyNumber === 0 ? 10 : keyNumber;
-
- if (event.altKey && keyNumber) {
- document.querySelectorAll('[data-test-id="conversation"]')[keyNumber - 1].click();
- chatIndex = Array.from(document.querySelectorAll('[data-test-id="conversation"]')).indexOf(document.querySelector('.selected[data-test-id="conversation"]'));
- notify(`"${document.querySelectorAll('[data-test-id="conversation"]')[chatIndex].querySelector(".conversation-title").innerHTML.trim()}"`)
- //notify(`${capitalize(nums[keyNumber-1])} conversation`)
- event.preventDefault();
- }
-
-
- if (event.key === "Escape" && document.activeElement.getAttribute("aria-label").includes("Edit prompt")) {
- simulateClick(getLastElement('[aria-label*="Cancel"]'));
- event.preventDefault();
- }
-
- const isCmdOrCtrl = (isMac && event.metaKey) || (!isMac && event.ctrlKey);
-
- if (!isCmdOrCtrl) return;
-
- if (isCmdOrCtrl && event.key === 'o' && !event.shiftKey) {
- event.preventDefault();
- simulateClick(document.querySelector('.upload-button button'));
- simulateClick(document.querySelector('[aria-label*="Upload files"]'))
- }
-
- switch (event.key) {
- case 'o':
- if (event.shiftKey) {
- simulateClick(document.querySelector('[aria-label*="New chat"] button'));
- simulateClick(document.querySelector('.text-input-field'));
- //notify("New chat created");
- event.preventDefault();
- } else {
- document.querySelector('[aria-label*="upload file"]').click(); setTimeout(function(){document.body.querySelector('[aria-label*="Upload files"]').click()}, rapidClickDelayMS);
- }
- break;
- //BELOW NEEDS MORE TIME
- case 'c':
- if (event.shiftKey) {
- event.preventDefault();
- getLastElement();
- copyRichTextFromDiv(c.querySelector(".model-response-text"));
- notify("Copied response")
-
-
- /* All of the below code was me desperately trying to do it through Google's menus, and failing for 2+ hours. Good riddance
-
- simulateClick(getLastElement('[aria-label*="options"]'));
- setTimeout(function(){simulateClick(document.querySelector('[aria-label*="Copy"]'))},rapidClickDelayMS*2)
- simulateClick(getLastElement('[aria-label*="options"]'));
- simulateClick(document.querySelector('#overflow-container'))
- setTimeout(function(){document.querySelector('.cdk-overlay-pane').style.top = "99999999px"; c.focus()},rapidClickDelayMS)
- clearNotifications();
- */
- }
- break;
- case 'i':
- if (event.shiftKey) {
- // Implement custom instructions if Gemini supports them
- event.preventDefault();
- }
- break;
- case 'f':
- if (event.shiftKey) {
- simulateClick(document.querySelector('[aria-label*="Main menu"]'));
- event.preventDefault();
- }
- break;
- case 'Backspace':
- if (event.shiftKey) {
- event.preventDefault();
- chatIndex = Array.from(document.querySelectorAll('[data-test-id="conversation"]')).indexOf(document.querySelector('.selected[data-test-id="conversation"]'));
- document.querySelector('.conversation.selected').parentElement.querySelector('[data-test-id="actions-menu-button"]').click(); setTimeout(function(){document.body.querySelector('[data-test-id="delete-button"]').click()}, rapidClickDelayMS); setTimeout(function(){document.body.querySelector('[data-test-id="confirm-button"]').click(); setTimeout(function(){if(goToNextChatOnDelete){simulateClick(document.querySelectorAll('[data-test-id="conversation"]')[chatIndex])}}, rapidClickDelayMS)}, rapidClickDelayMS)
- }
- break;
- case 'd':
- if (event.shiftKey) {
- let element = getLastElement('[data-test-id="generate-more-drafts-button"]');
- if (!element) {
- element = getLastElement('[mattooltip="Regenerate drafts"]');
- }
- simulateClick(element);
- event.preventDefault();
- }
- break;
- case 'e':
- if (event.shiftKey) {
- simulateClick(getLastElement('[mattooltip="Edit text"]'));
- event.preventDefault();
- }
- break;
- case ';':
- if (event.shiftKey) {
- event.preventDefault();
- // simulateClick(getLastElement('[mattooltip="Copy code"]'));
- getLastElement();
- copyRichTextFromDiv(c.querySelectorAll("code-block")[c.querySelectorAll("code-block").length - 1]);
- notify("Copied last code block to clipboard");
- }
- break;
- case '\'':
- if (event.shiftKey) {
- event.preventDefault();
- // simulateClick(getLastElement('[mattooltip="Copy code"]'));
- getLastElement();
- copyRichTextFromDiv(c.querySelectorAll("code-block")[c.querySelectorAll("code-block").length - 2]);
- notify("Copied second-last code block to clipboard");
- }
- break;
- case 'm':
- if (event.shiftKey) {
- simulateClick(getLastElement('[aria-label*="Share"]'));
- setTimeout(function(){simulateClick(document.querySelector('[aria-label*="Share response"]'))},rapidClickDelayMS)
- setTimeout(function(){simulateClick(document.querySelector('[data-test-id="share-mode-radio-button-full"] label'))},rapidClickDelayMS*2)
- setTimeout(function(){simulateClick(document.querySelector('[data-test-id="create-button"]'))},rapidClickDelayMS*3)
-
- //below waits until the link menu loads, then copies it and closes the menu
- const observer = new MutationObserver((_, observer) => {
- const element = document.querySelector('[aria-label="Copy public link"]');
- if (element) {
- observer.disconnect();
- simulateClick(element);
- setTimeout(function(){
- simulateClick(document.querySelector('[aria-label="Close"]'))
- notify("Chat link copied");
- },rapidClickDelayMS)
- }
- });
- observer.observe(document.body, {childList: true, subtree: true});
-
-
- clearNotifications();
- //notify("Last response copied to clipboard");
- event.preventDefault();
- }
- break;
- case 'l':
- if (event.shiftKey) {
- simulateClick(getLastElement('[aria-label*="Share"]'));
- setTimeout(function(){simulateClick(document.querySelector('[aria-label*="Share response"]'))},rapidClickDelayMS)
- setTimeout(function(){simulateClick(document.querySelector('[data-test-id="create-button"]'))},rapidClickDelayMS*2)
-
- //below waits until the link menu loads, then copies it and closes the menu
- const observer = new MutationObserver((_, observer) => {
- const element = document.querySelector('[aria-label="Copy public link"]');
- if (element) {
- observer.disconnect();
- simulateClick(element);
- setTimeout(function(){
- simulateClick(document.querySelector('[aria-label="Close"]'));
- notify("Prompt/response link copied");
- },rapidClickDelayMS)
- }
- });
- observer.observe(document.body, {childList: true, subtree: true});
- //notify("Last response copied to clipboard");
- event.preventDefault();
- }
- break;
- case ',':
- if (event.shiftKey) {
- previousDraft();
- }
- break;
- case '.':
- if (event.shiftKey) {
- nextDraft();
- }
- break;
- case '-':
- if (event.shiftKey) {
- event.preventDefault();
- previousChat();
- }
- break;
- case '=':
- if (event.shiftKey) {
- event.preventDefault();
- nextChat();
- }
- break;
- case 'k':
- event.preventDefault();
- if (event.shiftKey) {
- simulateClick(document.querySelector('[aria-label="Send message"]'));
- //notify("Last response copied to clipboard");
- }
- break;
- case 'y':
- if (event.shiftKey) {
- simulateClick(getLastElement('.response-tts-container button'));
- event.preventDefault();
- }
- break;
- case 's':
- if (event.shiftKey) {
- simulateClick(document.querySelector('[mattooltip="Use microphone"]'));
- event.preventDefault();
- }
- break;
- }
- });
- }