// ==UserScript==
// @name Back to Top
// @namespace http://tampermonkey.net/
// @version 0.7
// @description 为网页添加平滑返回顶部按钮
// @author sept
// @match *://*/*
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_registerMenuCommand
// @license MIT
// @icon 
// ==/UserScript==
(function () {
'use strict';
if (window.self !== window.top || window.myBackToTopScriptHasRun) {
return;
}
window.myBackToTopScriptHasRun = true;
const HOSTNAME = window.location.hostname;
const ENABLED_SITES_KEY = 'back-to-top-enabled-sites';
const BUTTON_POSITION_KEY = 'back-to-top-button-position';
// 获取已启用的网站列表,默认为 a list of 'github.com'
let enabledSites = GM_getValue(ENABLED_SITES_KEY, ['github.com']);
// 检查当前网站是否已启用
function isSiteEnabled() {
return enabledSites.includes(HOSTNAME);
}
// 切换当前网站的启用状态
function toggleSite() {
if (isSiteEnabled()) {
enabledSites = enabledSites.filter(site => site !== HOSTNAME);
} else {
enabledSites.push(HOSTNAME);
}
GM_setValue(ENABLED_SITES_KEY, enabledSites);
window.location.reload();
}
// 注册菜单命令
GM_registerMenuCommand(`${isSiteEnabled() ? '关闭' : '开启'}返回顶部`, toggleSite);
// 如果当前网站未启用,则不执行任何操作
if (!isSiteEnabled()) {
return;
}
// 创建按钮元素
const backToTopBtn = document.createElement('div');
backToTopBtn.id = 'gemini-back-to-top-button';
backToTopBtn.innerHTML = '↑';
// 创建样式
const style = document.createElement('style');
style.textContent = `
#gemini-back-to-top-button {
position: fixed;
width: 40px;
height: 40px;
border-radius: 50%;
background: linear-gradient(135deg, #238636, #2ea043);
color: white;
display: flex;
justify-content: center;
align-items: center;
cursor: grab;
box-shadow: 0 4px 12px rgba(0,0,0,0.2);
opacity: 0;
transition: opacity 0.3s, transform 0.3s;
font-size: 20px;
font-weight: bold;
z-index: 9999;
}
#gemini-back-to-top-button.visible {
opacity: 1;
}
#gemini-back-to-top-button:hover {
transform: scale(1.1);
box-shadow: 0 6px 16px rgba(0,0,0,0.2);
}
#gemini-back-to-top-button:active {
cursor: grabbing;
transform: scale(0.9);
}
`;
// 添加样式和按钮到页面
document.head.appendChild(style);
document.body.appendChild(backToTopBtn);
function applyPosition(position) {
let top, left;
const buttonWidth = 40;
const buttonHeight = 40;
if (position.pctTop !== undefined && position.pctLeft !== undefined) {
top = position.pctTop * (window.innerHeight - buttonHeight);
left = position.pctLeft * (window.innerWidth - buttonWidth);
} else if (position.top !== undefined && position.left !== undefined) {
top = position.top;
left = position.left;
} else {
return;
}
top = Math.max(0, Math.min(top, window.innerHeight - buttonHeight));
left = Math.max(0, Math.min(left, window.innerWidth - buttonWidth));
backToTopBtn.style.top = `${top}px`;
backToTopBtn.style.left = `${left}px`;
}
const defaultPosition = { pctTop: 0.9, pctLeft: 0.9 };
const savedPosition = GM_getValue(BUTTON_POSITION_KEY, defaultPosition);
applyPosition(savedPosition);
function smoothScrollTo(targetY, duration) {
const startY = window.pageYOffset;
const distance = targetY - startY;
let startTime = null;
function animation(currentTime) {
if (startTime === null) startTime = currentTime;
const timeElapsed = currentTime - startTime;
const run = ease(timeElapsed, startY, distance, duration);
window.scrollTo(0, run);
if (timeElapsed < duration) requestAnimationFrame(animation);
}
// easeInOutQuad easing function
function ease(t, b, c, d) {
t /= d / 2;
if (t < 1) return c / 2 * t * t + b;
t--;
return -c / 2 * (t * (t - 2) - 1) + b;
}
requestAnimationFrame(animation);
}
// 滚动事件监听
window.addEventListener('scroll', function () {
if (window.pageYOffset > 300) {
backToTopBtn.classList.add('visible');
} else {
backToTopBtn.classList.remove('visible');
}
});
// 窗口大小调整事件监听
window.addEventListener('resize', () => {
const pos = GM_getValue(BUTTON_POSITION_KEY, defaultPosition);
applyPosition(pos);
});
let isDragging = false;
let hasDragged = false;
let offsetX, offsetY, startX, startY;
backToTopBtn.addEventListener('mousedown', (e) => {
isDragging = true;
hasDragged = false;
startX = e.clientX;
startY = e.clientY;
offsetX = e.clientX - backToTopBtn.getBoundingClientRect().left;
offsetY = e.clientY - backToTopBtn.getBoundingClientRect().top;
backToTopBtn.style.transition = 'none'; // Disable transition during drag
});
document.addEventListener('mousemove', (e) => {
if (isDragging) {
const dx = e.clientX - startX;
const dy = e.clientY - startY;
if (Math.sqrt(dx * dx + dy * dy) > 5) {
hasDragged = true;
}
if (hasDragged) {
let newTop = e.clientY - offsetY;
let newLeft = e.clientX - offsetX;
// Constrain to viewport
newTop = Math.max(0, Math.min(newTop, window.innerHeight - backToTopBtn.offsetHeight));
newLeft = Math.max(0, Math.min(newLeft, window.innerWidth - backToTopBtn.offsetWidth));
backToTopBtn.style.top = `${newTop}px`;
backToTopBtn.style.left = `${newLeft}px`;
}
}
});
document.addEventListener('mouseup', () => {
if (isDragging) {
isDragging = false;
backToTopBtn.style.transition = 'opacity 0.3s, transform 0.3s'; // Re-enable transition
if (hasDragged) {
const pctTop = backToTopBtn.offsetTop / (window.innerHeight - backToTopBtn.offsetHeight);
const pctLeft = backToTopBtn.offsetLeft / (window.innerWidth - backToTopBtn.offsetWidth);
GM_setValue(BUTTON_POSITION_KEY, {
pctTop: Math.max(0, Math.min(1, pctTop)),
pctLeft: Math.max(0, Math.min(1, pctLeft)),
});
}
}
});
// 点击事件 - 平滑滚动
backToTopBtn.addEventListener('click', function () {
if (!hasDragged) {
smoothScrollTo(0, 300);
}
});
})();