// ==UserScript==
// @name Cottonee's VR Clipboard
// @namespace http://tampermonkey.net/
// @version 2.1
// @description Adds a discrete icon to the YouTube header to automatically copy video links for VR. History toggles on click.
// @author Cottonee
// @match *://www.youtube.com/*
// @grant GM_setValue
// @grant GM_getValue
// @grant GM_addStyle
// @icon 
// ==/UserScript==
(function () {
"use strict";
// --- CONFIGURATION ---
const MAX_HISTORY_SIZE = 10;
const UI_TARGET_SELECTOR = "ytd-masthead #end"; // A stable target for UI injection
// --- STATE MANAGEMENT ---
let isEnabled = GM_getValue("isEnabled", true);
// History now stores objects: { url: string, timestamp: number }
let linkHistory = GM_getValue("linkHistory", []);
let lastCopiedUrl = "";
let isDropdownOpen = false;
// --- UI ELEMENTS ---
let clipboardContainer, toggleSwitch, historyList, statusMessage, iconWrapper, historyDropdown;
/**
* Helper function to format a timestamp into a human-readable string.
*/
function formatTimestamp(timestamp) {
const date = new Date(timestamp);
const options = {
month: "short",
day: "numeric",
hour: "2-digit",
minute: "2-digit",
hour12: true,
};
return date.toLocaleDateString(undefined, options);
}
/**
* Transforms old history (array of strings) to new format (array of objects).
* This handles existing users who update the script.
*/
function migrateHistoryIfNecessary(history) {
if (history.length === 0) return [];
if (typeof history[0] === "string") {
// Old format detected (array of URLs)
console.log("Cottonee's VR Clipboard: Migrating old history format...");
return history.map((url) => ({
url: url,
timestamp: Date.now(), // Use current time as a reasonable default
}));
}
return history; // Already in new format
}
/**
* Injects the CSS for the GUI.
*/
function addStyles() {
GM_addStyle(`
#cottonees-clipboard {
position: relative;
display: flex;
align-items: center;
margin-right: 8px; /* Standard YouTube button spacing */
font-family: 'Roboto', Arial, sans-serif;
user-select: none;
}
.cc-icon-wrapper {
position: relative;
display: flex;
align-items: center;
justify-content: center;
width: 40px;
height: 40px;
background-color: transparent;
border-radius: 50%;
cursor: pointer;
transition: background-color 0.2s ease;
}
.cc-icon-wrapper:hover {
background-color: var(--yt-spec-badge-chip-background, #3f3f3f);
}
/* Ensure the image fits within its wrapper */
.cc-icon-wrapper img {
width: 24px;
height: 24px;
/* Optional: Add some padding if the icon appears too large */
/* padding: 2px; */
}
.cc-history-dropdown {
display: none; /* Hidden by default, toggled by JS */
position: absolute;
top: 50px; /* Position below the icon */
right: 0;
width: 300px;
background-color: #282828;
border: 1px solid #4a4a4a; /* Slightly lighter border */
border-radius: 12px;
padding: 12px 15px; /* More generous padding */
z-index: 9999;
box-shadow: 0 6px 16px rgba(0,0,0,0.6); /* Enhanced shadow */
}
.cc-history-dropdown.open {
display: block;
}
.cc-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 10px; /* More space */
padding-bottom: 8px; /* Separator for header */
border-bottom: 1px solid #4a4a4a; /* Subtle line */
}
.cc-title {
font-weight: 500;
font-size: 15px; /* Slightly larger title */
color: #e0e0e0; /* Slightly brighter text */
}
.cc-status {
font-size: 13px; /* Slightly larger status message */
color: #d1b4ff; /* Purple for copied status! */
height: 18px; /* Reserve space to prevent layout shifts */
text-align: right;
min-width: 60px; /* Ensure space for longer messages */
font-weight: 500;
}
.cc-history-list {
max-height: 220px;
overflow-y: auto;
padding-top: 5px; /* Space from separator */
}
.cc-history-list::-webkit-scrollbar {
width: 8px;
}
.cc-history-list::-webkit-scrollbar-track {
background: #383838;
border-radius: 4px;
}
.cc-history-list::-webkit-scrollbar-thumb {
background: #555;
border-radius: 4px;
}
.cc-history-list::-webkit-scrollbar-thumb:hover {
background: #666;
}
.cc-history-item {
padding: 6px 6px; /* Adjusted padding for timestamp */
font-size: 12px;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
border-radius: 6px; /* Smoother corners */
cursor: pointer;
transition: background-color 0.2s ease, color 0.2s ease;
color: #eeeeee; /* Brighter default text */
display: flex; /* Use flexbox for timestamp and link */
flex-direction: column;
line-height: 1.3;
}
.cc-history-item:hover {
background-color: #4a3070; /* Purple hover background */
color: #ffffff; /* White text on hover for contrast */
}
.cc-history-list > div:not(:last-child).cc-history-item {
margin-bottom: 4px; /* Spacing between items */
}
.cc-history-timestamp {
font-size: 10px; /* Smaller timestamp */
color: #999; /* Grey timestamp */
margin-bottom: 2px; /* Space between timestamp and link */
}
.cc-history-url-text {
flex-grow: 1;
overflow: hidden;
text-overflow: ellipsis;
}
/* Toggle Switch CSS (with purple accent) */
.cc-switch {
position: relative;
display: inline-block;
width: 38px; /* Slightly wider */
height: 22px; /* Slightly taller */
}
.cc-switch input { display: none; }
.cc-slider {
position: absolute;
cursor: pointer;
top: 0; left: 0; right: 0; bottom: 0;
background-color: #666; /* Slightly darker grey when off */
transition: .4s;
border-radius: 22px; /* Matches height */
}
.cc-slider:before {
position: absolute;
content: "";
height: 16px; width: 16px; /* Larger handle */
left: 3px; bottom: 3px;
background-color: white;
transition: .4s;
border-radius: 50%;
}
input:checked + .cc-slider { background-color: #8a2be2; } /* Vibrant purple when on */
input:checked + .cc-slider:before { transform: translateX(16px); } /* Adjust translation for new width */
`);
}
/**
* Creates and injects the GUI into the page.
*/
function createGUI(parentElement) {
if (document.getElementById("cottonees-clipboard")) return; // Prevent duplicates
clipboardContainer = document.createElement("div");
clipboardContainer.id = "cottonees-clipboard";
// Use an <img> tag for the ICO icon with the correct MIME type
const iconBase64 = ""
clipboardContainer.innerHTML = `
<div class="cc-icon-wrapper" title="Cottonee's VR Clipboard - Click to Toggle History">
<img src="${iconBase64}" alt="Cottonee's VR Clipboard Icon"/>
</div>
<div class="cc-history-dropdown">
<div class="cc-header">
<span class="cc-title">Cottonee's VR Clipboard</span>
<label class="cc-switch">
<input type="checkbox" id="cc-toggle">
<span class="cc-slider"></span>
</label>
</div>
<div class="cc-status" id="cc-status-message"></div>
<div class="cc-history-list" id="cc-history-list"></div>
</div>
`;
parentElement.insertBefore(clipboardContainer, parentElement.firstChild);
// Assign element references
iconWrapper = clipboardContainer.querySelector(".cc-icon-wrapper");
historyDropdown = clipboardContainer.querySelector(".cc-history-dropdown");
toggleSwitch = document.getElementById("cc-toggle");
historyList = document.getElementById("cc-history-list");
statusMessage = document.getElementById("cc-status-message");
// Set initial state
toggleSwitch.checked = isEnabled;
updateHistoryUI();
// Event Listeners
iconWrapper.addEventListener("click", (e) => {
e.stopPropagation(); // Prevent document click from closing immediately
toggleDropdown();
});
toggleSwitch.addEventListener("change", () => {
isEnabled = toggleSwitch.checked;
GM_setValue("isEnabled", isEnabled);
showStatusMessage(isEnabled ? "Enabled" : "Disabled");
});
historyList.addEventListener("click", (e) => {
// Find the closest history item to the clicked element
const historyItem = e.target.closest(".cc-history-item");
if (historyItem && historyItem.dataset.url) { // Ensure dataset.url exists
// Ensure we copy the URL from the dataset
navigator.clipboard.writeText(historyItem.dataset.url).then(() => {
showStatusMessage("Copied!");
});
}
});
// Close dropdown if clicking outside
document.addEventListener("click", (e) => {
if (isDropdownOpen && !clipboardContainer.contains(e.target)) {
closeDropdown();
}
});
}
/**
* Toggles the visibility of the history dropdown.
*/
function toggleDropdown() {
isDropdownOpen = !isDropdownOpen;
historyDropdown.classList.toggle("open", isDropdownOpen);
}
/**
* Closes the history dropdown.
*/
function closeDropdown() {
isDropdownOpen = false;
historyDropdown.classList.remove("open");
}
/**
* Updates the history list in the UI.
*/
function updateHistoryUI() {
if (!historyList) return; // Ensure element exists
historyList.innerHTML = "";
if (linkHistory.length === 0) {
historyList.innerHTML =
'<div style="color:#888; font-size:12px; padding: 4px;">History is empty.</div>';
return;
}
// Display newest links first
[...linkHistory].reverse().forEach((item) => {
// item is now { url: string, timestamp: number }
const div = document.createElement("div");
div.className = "cc-history-item";
div.title = `Click to copy: ${item.url}`; // Full URL on hover tooltip
div.dataset.url = item.url; // Store full URL for copying
const timestampSpan = document.createElement("span");
timestampSpan.className = "cc-history-timestamp";
timestampSpan.textContent = formatTimestamp(item.timestamp);
div.appendChild(timestampSpan);
const urlSpan = document.createElement("span");
urlSpan.className = "cc-history-url-text";
urlSpan.textContent = item.url.replace("https://www.", ""); // Display a cleaner version
div.appendChild(urlSpan);
historyList.appendChild(div);
});
}
/**
* Shows a temporary message in the status area.
*/
function showStatusMessage(message) {
if (!statusMessage) return; // Ensure element exists
statusMessage.textContent = message;
setTimeout(() => {
// Only clear if the message hasn't been replaced by a new one
if (statusMessage.textContent === message) {
statusMessage.textContent = "";
}
}, 1500);
}
/**
* The main logic to check, clean, and copy the URL.
*/
function processUrl() {
if (!isEnabled) return;
const currentUrl = window.location.href;
// Only process video pages (watch?v=)
if (!currentUrl.includes("watch?v=")) return;
const urlObj = new URL(currentUrl);
const videoId = urlObj.searchParams.get("v");
if (!videoId) return;
// Construct the clean, canonical YouTube video URL
const cleanUrl = `https://www.youtube.com/watch?v=${videoId}`;
// Only copy and update history if the URL is different from the last one
if (cleanUrl !== lastCopiedUrl) {
lastCopiedUrl = cleanUrl;
navigator.clipboard.writeText(cleanUrl).then(() => {
showStatusMessage("Copied!");
const newHistoryItem = {
url: cleanUrl,
timestamp: Date.now(), // Store current timestamp
};
// Update history: remove existing entry if its URL is present, then add to end
// Need to check by URL property now
const existingIndex = linkHistory.findIndex((item) => item.url === cleanUrl);
if (existingIndex > -1) {
linkHistory.splice(existingIndex, 1);
}
linkHistory.push(newHistoryItem);
// Enforce maximum history size
if (linkHistory.length > MAX_HISTORY_SIZE) {
linkHistory.shift(); // Remove the oldest item
}
GM_setValue("linkHistory", linkHistory); // Persist history
updateHistoryUI(); // Refresh the UI display
});
}
}
/**
* Waits for a specific element to appear in the DOM and then executes a callback.
* Uses MutationObserver for efficiency.
*/
function waitForElement(selector, callback) {
// Check immediately in case the element is already there
let targetElement = document.querySelector(selector);
if (targetElement) {
callback(targetElement);
return;
}
const observer = new MutationObserver((mutations, obs) => {
targetElement = document.querySelector(selector);
if (targetElement) {
obs.disconnect(); // Stop observing once found
callback(targetElement);
}
});
// Start observing the body for childList changes and subtree for deep changes
observer.observe(document.body, { childList: true, subtree: true });
}
// --- INITIALIZATION ---
// Migrate history format immediately after retrieving it
linkHistory = migrateHistoryIfNecessary(linkHistory);
GM_setValue("linkHistory", linkHistory); // Save migrated history
addStyles();
waitForElement(UI_TARGET_SELECTOR, (targetElement) => {
createGUI(targetElement);
// Initial check for the URL on page load, with a slight delay
// to ensure YouTube's JS has processed the initial video.
setTimeout(processUrl, 1500);
});
// Listen for YouTube's custom navigation event (for SPA updates)
window.addEventListener("yt-navigate-finish", processUrl);
})();