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