Skip to content

Instantly share code, notes, and snippets.

@fltd
Last active October 10, 2025 07:47
Show Gist options
  • Select an option

  • Save fltd/9d78bec180fb4649784070689a0a6a14 to your computer and use it in GitHub Desktop.

Select an option

Save fltd/9d78bec180fb4649784070689a0a6a14 to your computer and use it in GitHub Desktop.
accio-google-meet.user.js

Accio Google Meet

version

A userscript to automate the small, repetitive tasks in Google Calendar and Google Meet. This script was created to bring back the convenience of automatically adding meeting links and to ensure you never forget to start taking notes again.

The Problem

In many Google Workspace environments, the "Automatically add video conferences" setting is disabled to avoid confusion. This is a great policy, but it means you have to remember to click "Add Google Meet video conferencing" every single time you create an event.

Similarly, with the excellent "Take notes" feature powered by Gemini in Google Meet, it's easy to forget to enable it right at the start of a call; or when you set up the meeting.

This script solves both problems.

Features

  • Accio Google Meet: Automatically finds and clicks the "Add Google Meet video conferencing" button when you create or duplicate an event in Google Calendar.
  • Accio Gemini Notes: Automatically finds and clicks the "Take notes" button when you join a Google Meet call.
  • Control Panel: A simple magic wand icon on Google Calendar opens a settings menu to give you full control.
  • Host-Only Mode: Configure the script to only activate "Accio Gemini Notes" when you are the meeting host.

Installation

To use this script, you first need a userscript manager for your browser.

  1. Install a Userscript Manager The most popular one is Tampermonkey.

  2. Install the Accio Google Meet Userscript Once Tampermonkey is installed, click the link below to install the script. A new tab will open in Tampermonkey. Just click the "Install" button.

    ➡️ Install Accio Google Meet Userscript

That's it! The script is now active.

Usage & Configuration

The script's settings can be managed directly from the Google Calendar page.

  1. Go to Google Calendar.
  2. Look for the popcat icon 😺 in the bottom-right corner of the screen.
  3. Click the icon to open the Accio Settings panel.

You will see a cat icon on the bottom-right corner of your Google Calendar.

Click the cat icon to open the Accio Settings panel.

From this panel, you can:

  • Accio Google Meet: Toggle the automatic adding of Meet links in Calendar.
  • Accio Gemini Notes: Toggle the automatic start of note-taking in Meet.
  • Only Take Notes When Host: If "Accio Gemini Notes" is active, this checkbox will appear. When checked, notes will only be auto-started in meetings where you have host controls.

Your preferences are saved automatically.

// ==UserScript==
// @name Accio Google Meet
// @version 0.1.0
// @author Floyd
// @description Automate adding Meet links and starting notes in Meet (with a host-only option), with a control panel on Google Calendar.
// @match https://calendar.google.com/calendar/*
// @match https://meet.google.com/*
// @grant GM_setValue
// @grant GM_getValue
// @grant GM_addStyle
// @run-at document-idle
// ==/UserScript==
(function () {
"use strict";
// --- SHARED CONFIGURATION ---
const ICON_URL_DEFAULT =
"";
const ICON_URL_ACTIVE =
"";
const MEET_CLICK_DEBOUNCE_MS = 750; // Time to wait after finding a button to ensure it's the final one.
let settings = {
enableAccioGoogleMeet: true,
enableAccioGeminiNotes: true,
takeNotesOnlyWhenHost: false,
};
// --- GOOGLE MEET SELECTORS ---
const meetPencilButtonSelectors = [
'button[aria-label="Gemini isn\'t taking notes"]',
'button[aria-label="Gemini がメモを作成していません"]',
];
const meetStartTakingNotesButtonSelectors = [
'button[data-progress-announcement="Gemini\'s getting ready to take notes"]',
'button[data-progress-announcement="Gemini がメモの作成の準備をしています"]',
];
const hostControlButtonSelectors = [
'button[aria-label="Host controls"]',
'button[aria-label="主催者用ボタン"]',
];
// --- GOOGLE CALENDAR TEXT ---
const calendarAddGoogleMeetVideoConferencingButtonTexts = [
"Add Google Meet video conferencing",
"Google Meet のビデオ会議を追加",
];
// =================================================================================
// SHARED UI & STATE MANAGEMENT
// =================================================================================
function setupStyles() {
GM_addStyle(`
#accio-control-icon {
position: fixed; bottom: 50px; right: 5px; width: 46px; height: 46px;
border-radius: 50%; cursor: pointer; z-index: 9999;
transition: transform 0.2s ease, opacity 0.3s ease;
display: flex; align-items: center; justify-content: center;
}
#accio-control-icon.disabled { opacity: 0.4; }
#accio-control-icon img { width: 80%; height: 80%; }
#accio-menu {
position: fixed; bottom: 85px; right: 20px; width: 280px;
background: #3c4043; color: #e8eaed; border-radius: 12px;
box-shadow: 0 5px 15px rgba(0,0,0,0.4); z-index: 9998;
padding: 16px; font-family: 'Google Sans', Roboto, Arial, sans-serif;
visibility: hidden; opacity: 0; transform: translateY(10px);
transition: all 0.25s cubic-bezier(0.4, 0, 0.2, 1);
}
#accio-menu.visible { visibility: visible; opacity: 1; transform: translateY(0); }
#accio-menu h3 { margin: 0 0 8px 0; font-size: 16px; font-weight: 500; text-align: center; color: #fff; }
.accio-menu-row { display: flex; justify-content: space-between; align-items: center; margin-top: 10px; transition: opacity 0.3s ease; }
.accio-menu-row span { max-width: 190px; }
.accio-menu-row.disabled-row {
opacity: 0.5;
pointer-events: none;
}
.accio-switch { position: relative; display: inline-block; width: 50px; height: 28px; flex-shrink: 0; }
.accio-switch input { opacity: 0; width: 0; height: 0; }
.accio-slider { position: absolute; cursor: pointer; top: 0; left: 0; right: 0; bottom: 0; background-color: #5f6368; transition: .3s; border-radius: 28px; }
.accio-slider:before { position: absolute; content: ""; height: 20px; width: 20px; left: 4px; bottom: 4px; background-color: white; transition: .3s; border-radius: 50%; }
input:checked + .accio-slider { background-color: #b68dfc; }
input:checked + .accio-slider:before { transform: translateX(22px); }
`);
}
function createUI() {
const controlIcon = document.createElement("div");
controlIcon.id = "accio-control-icon";
const iconImage = document.createElement("img");
iconImage.id = "accio-icon-image";
iconImage.src = ICON_URL_DEFAULT;
iconImage.alt = "Accio Settings";
controlIcon.appendChild(iconImage);
const menu = document.createElement("div");
menu.id = "accio-menu";
const menuHeader = document.createElement("h3");
menuHeader.textContent = "🪄 Accio Settings";
menu.appendChild(menuHeader);
const createMenuRow = (id, labelText) => {
const row = document.createElement("div");
row.className = "accio-menu-row";
row.id = id + "-row";
const label = document.createElement("span");
label.textContent = labelText;
row.appendChild(label);
const switchContainer = document.createElement("label");
switchContainer.className = "accio-switch";
const checkbox = document.createElement("input");
checkbox.type = "checkbox";
checkbox.id = id;
switchContainer.appendChild(checkbox);
const slider = document.createElement("span");
slider.className = "accio-slider";
switchContainer.appendChild(slider);
row.appendChild(switchContainer);
return row;
};
const calendarRow = createMenuRow(
"accio-toggle-google-meet",
"Accio Google Meet"
);
const meetRow = createMenuRow(
"accio-toggle-gemini-notes",
"Accio Gemini Notes"
);
const hostRow = createMenuRow(
"accio-toggle-host",
"Only Take Notes When Host"
);
menu.appendChild(calendarRow);
menu.appendChild(meetRow);
menu.appendChild(hostRow);
document.body.appendChild(controlIcon);
document.body.appendChild(menu);
const accioGoogleMeetToggle = document.getElementById(
"accio-toggle-google-meet"
);
const accioGeminiNotesToggle = document.getElementById(
"accio-toggle-gemini-notes"
);
const hostToggle = document.getElementById("accio-toggle-host");
controlIcon.addEventListener("mouseover", () => {
iconImage.src = ICON_URL_ACTIVE;
});
controlIcon.addEventListener("mouseout", () => {
if (!menu.classList.contains("visible")) {
iconImage.src = ICON_URL_DEFAULT;
}
});
controlIcon.addEventListener("click", (e) => {
e.stopPropagation();
menu.classList.toggle("visible");
if (menu.classList.contains("visible")) {
iconImage.src = ICON_URL_ACTIVE;
} else {
iconImage.src = ICON_URL_DEFAULT;
}
});
document.addEventListener("click", () => {
if (menu.classList.contains("visible")) {
menu.classList.remove("visible");
iconImage.src = ICON_URL_DEFAULT;
}
});
menu.addEventListener("click", (e) => e.stopPropagation());
accioGoogleMeetToggle.addEventListener("change", () => {
settings.enableAccioGoogleMeet = accioGoogleMeetToggle.checked;
updateAndSaveSettings();
});
accioGeminiNotesToggle.addEventListener("change", () => {
settings.enableAccioGeminiNotes = accioGeminiNotesToggle.checked;
if (!settings.enableAccioGeminiNotes) {
settings.takeNotesOnlyWhenHost = false;
}
updateAndSaveSettings();
});
hostToggle.addEventListener("change", () => {
settings.takeNotesOnlyWhenHost = hostToggle.checked;
updateAndSaveSettings();
});
}
async function updateAndSaveSettings() {
await GM_setValue("accioSettings", settings);
console.log("Accio: Settings updated.", settings);
const controlIcon = document.getElementById("accio-control-icon");
if (controlIcon) {
const hostRow = document.getElementById("accio-toggle-host-row");
document.getElementById("accio-toggle-google-meet").checked =
settings.enableAccioGoogleMeet;
document.getElementById("accio-toggle-gemini-notes").checked =
settings.enableAccioGeminiNotes;
document.getElementById("accio-toggle-host").checked =
settings.takeNotesOnlyWhenHost;
if (settings.enableAccioGeminiNotes) {
hostRow.classList.remove("disabled-row");
} else {
hostRow.classList.add("disabled-row");
}
controlIcon.classList.toggle(
"disabled",
!settings.enableAccioGoogleMeet && !settings.enableAccioGeminiNotes
);
}
}
async function loadInitialState() {
const defaultSettings = {
enableAccioGoogleMeet: true,
enableAccioGeminiNotes: true,
takeNotesOnlyWhenHost: false,
};
const savedSettings = await GM_getValue("accioSettings", defaultSettings);
settings = { ...defaultSettings, ...savedSettings };
updateAndSaveSettings();
}
function findElementBySelectors(selectors) {
for (const selector of selectors) {
const element = document.querySelector(selector);
if (element) return element;
}
return null;
}
// =================================================================================
// DOMAIN-SPECIFIC LOGIC
// =================================================================================
function runCalendarObserver() {
console.log("Accio: Initializing on Google Calendar...");
const observer = new MutationObserver((mutations) => {
if (!settings.enableAccioGoogleMeet) return;
for (const mutation of mutations) {
for (const node of mutation.addedNodes) {
if (!(node instanceof HTMLElement)) continue;
const spans = node.querySelectorAll("span");
for (const span of spans) {
if (
calendarAddGoogleMeetVideoConferencingButtonTexts.some((text) =>
span.innerText.includes(text)
)
) {
const addGoogleMeetVideoConferencingButton =
span.closest("button");
if (addGoogleMeetVideoConferencingButton) {
console.log(
"Accio: Found and clicked 'Add Google Meet video conferencing' button."
);
addGoogleMeetVideoConferencingButton.click();
return;
}
}
}
}
}
});
observer.observe(document.body, { subtree: true, childList: true });
}
function runMeetObserver() {
console.log("Accio: Initializing on Google Meet (silent mode)...");
let clickTimerId = null; // Holds the timer for the debounced click
const observer = new MutationObserver((mutations, obs) => {
if (!settings.enableAccioGeminiNotes) {
obs.disconnect();
return;
}
// --- NEW RESILIENT LOGIC ---
// Check for the final button first. This is our highest priority.
const startTakingNotesButton = findElementBySelectors(
meetStartTakingNotesButtonSelectors
);
if (startTakingNotesButton) {
// If host is required, perform the check NOW, right before we act.
if (settings.takeNotesOnlyWhenHost) {
const hostControlsButton = findElementBySelectors(
hostControlButtonSelectors
);
if (!hostControlsButton) {
// Button is visible, but host controls aren't yet. Be patient and wait for the next mutation.
return;
}
}
// Conditions met. Debounce the click.
clearTimeout(clickTimerId);
clickTimerId = setTimeout(() => {
console.log(
`Accio: Debounce timer finished. Clicking the 'Start notes' button.`
);
startTakingNotesButton.click();
obs.disconnect(); // This is the only place the observer stops.
}, MEET_CLICK_DEBOUNCE_MS);
return;
}
// If the final button isn't found, check for the pencil button.
const pencilButton = findElementBySelectors(meetPencilButtonSelectors);
if (pencilButton) {
if (clickTimerId) return; // A click is already scheduled, do nothing.
// If host is required, perform the check NOW, before clicking the pencil.
if (settings.takeNotesOnlyWhenHost) {
const hostControlsButton = findElementBySelectors(
hostControlButtonSelectors
);
if (!hostControlsButton) {
// Pencil is visible, but host controls aren't yet. Wait for the next mutation.
return;
}
}
// Conditions met. Click the pencil button to reveal the menu.
console.log("Accio: Pencil button found. Clicking to reveal menu.");
pencilButton.click();
}
});
observer.observe(document.body, { childList: true, subtree: true });
}
// =================================================================================
// INITIALIZATION ROUTER
// =================================================================================
async function initialize() {
await loadInitialState();
const hostname = window.location.hostname;
if (hostname === "calendar.google.com") {
setupStyles();
createUI();
updateAndSaveSettings();
runCalendarObserver();
} else if (hostname === "meet.google.com") {
runMeetObserver();
}
}
initialize();
})();
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment