// ==UserScript==
// @name website-time-tracker
// @namespace https://github.com/sansan0/useful-userscripts
// @version 3.0
// @description 用于追踪和统计用户在每个二级域名下的在线时长,并提供友好的可视化统计界面
// @author sansan
// @match http://*/*
// @match https://*/*
// @require https://cdn.jsdelivr.net/npm/[email protected]/dist/echarts.min.js
// @grant GM_getValue
// @grant GM_setValue
// @license GPL-3.0 License
// @icon 
// ==/UserScript==
(function () {
"use strict";
const DEFAULT_STYLES = {
position: "fixed",
zIndex: 999999,
backgroundColor: "rgba(255, 255, 255, 0.95)",
boxShadow: "0 2px 12px rgba(0, 0, 0, 0.1)",
borderRadius: "8px",
fontFamily:
'-apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Arial, sans-serif',
transition: "all 0.3s ease",
};
class TimeTracker {
constructor() {
this.startTime = null;
this.animationFrameId = null;
this.isStatsShown = false;
this.isTabActive = true;
this.timeDisplay = null;
this.statsButton = null;
this.initialize();
}
/**
* 获取时间日期
*/
getBeijingDate() {
const now = new Date();
const year = now.getFullYear();
const month = String(now.getMonth() + 1).padStart(2, "0");
const day = String(now.getDate()).padStart(2, "0");
return `${year}-${month}-${day}`;
}
/**
* 记录在线时间
*/
logTime() {
if (!this.startTime || !this.isTabActive) return;
const currentTimeInSeconds = Math.floor(
(Date.now() - this.startTime) / 1000
);
const today = this.getBeijingDate();
const currentDomain = window.location.hostname
.split(".")
.slice(-2)
.join(".");
let data = this.getStorageData();
if (!data[currentDomain]) {
data[currentDomain] = {};
}
if (!data[currentDomain][today]) {
data[currentDomain][today] = 0;
}
data[currentDomain][today] += currentTimeInSeconds;
this.setStorageData(data);
this.startTime = Date.now();
}
/**
* 更新 GM_setValue 并同步显示
*/
updateGMStorage(timeSpentInSeconds) {
if (timeSpentInSeconds <= 0) return;
const today = this.getBeijingDate();
const currentDomain = window.location.hostname
.split(".")
.slice(-2)
.join(".");
let data = this.getStorageData();
if (!data[currentDomain]) {
data[currentDomain] = {};
}
if (!data[currentDomain][today]) {
data[currentDomain][today] = 0;
}
data[currentDomain][today] += timeSpentInSeconds;
this.setStorageData(data);
this.updateDisplay();
}
/**
* 将秒数格式化为时分秒格式
*/
formatTime(secondsTotal) {
const hours = Math.floor(secondsTotal / 3600);
const minutes = Math.floor((secondsTotal % 3600) / 60);
const seconds = secondsTotal % 60;
return [
hours > 0 ? `${hours} 时` : "",
minutes > 0 ? `${minutes} 分` : "",
`${seconds} 秒`,
]
.filter(Boolean)
.join(" ");
}
/**
* 创建UI元素
*/
createUIElements() {
const container = this.createFixedElement("div", {
style: {
...DEFAULT_STYLES,
top: "20px",
right: "20px",
display: "flex",
alignItems: "center",
padding: "12px 20px",
gap: "20px",
cursor: "move",
width: "fit-content",
height: "fit-content",
maxWidth: "max-content",
whiteSpace: "nowrap",
boxSizing: "border-box",
userSelect: "none",
},
});
container.setAttribute("draggable", "true");
const dragFrame = this.createFixedElement("div", {
style: {
position: "fixed",
border: "2px dashed #2196F3",
zIndex: 999998,
pointerEvents: "none",
display: "none",
},
});
this.setupDragEvents(container, dragFrame);
const savedPosition = JSON.parse(
GM_getValue("timeTrackerPosition", "{}")
);
if (savedPosition) {
const { left, top, isShrinked } = savedPosition;
container.style.left = `${left}px`;
container.style.top = `${top}px`;
container.style.right = "auto";
this.adjustContainerStyle(container, isShrinked);
}
document.body.appendChild(dragFrame);
const timeContainer = this.createTimeContainer();
const divider = this.createDivider();
this.statsButton = this.createStatsButton();
container.appendChild(timeContainer);
container.appendChild(divider);
container.appendChild(this.statsButton);
document.body.appendChild(container);
this.addButtonHoverEffects(this.statsButton);
this.addContainerHoverEffects(container);
this.statsButton.addEventListener("click", () => {
this.logTime();
this.showStats();
this.addButtonClickEffect(this.statsButton);
});
document.addEventListener("dblclick", (event) => {
if (container.contains(event.target)) {
const savedPosition = JSON.parse(
GM_getValue("timeTrackerPosition", "{}")
);
savedPosition.isShrinked = !savedPosition.isShrinked;
GM_setValue("timeTrackerPosition", JSON.stringify(savedPosition));
this.adjustContainerStyle(container, savedPosition.isShrinked);
}
});
}
/**
* 设置拖拽事件
*/
setupDragEvents(container, dragFrame) {
container.addEventListener("dragstart", (event) => {
const transparentElement = document.createElement("div");
transparentElement.style.opacity = "0";
document.body.appendChild(transparentElement);
event.dataTransfer.setDragImage(transparentElement, 0, 0);
setTimeout(() => {
document.body.removeChild(transparentElement);
}, 0);
event.dataTransfer.effectAllowed = "move";
dragFrame.style.width = `${container.offsetWidth}px`;
dragFrame.style.height = `${container.offsetHeight}px`;
dragFrame.style.display = "block";
container.style.opacity = "0.8";
});
container.addEventListener("drag", (event) => {
event.preventDefault();
const { clientX, clientY } = event;
const { offsetWidth, offsetHeight } = container;
const { innerWidth, innerHeight } = window;
const left = clientX - offsetWidth / 2;
const top = clientY - offsetHeight / 2;
dragFrame.style.left = `${Math.max(
0,
Math.min(left, innerWidth - offsetWidth)
)}px`;
dragFrame.style.top = `${Math.max(
0,
Math.min(top, innerHeight - offsetHeight)
)}px`;
container.style.pointerEvents = "auto";
});
document.addEventListener("dragover", (event) => {
event.preventDefault();
event.dataTransfer.dropEffect = "move";
});
container.addEventListener("dragend", (event) => {
const { clientX, clientY } = event;
const { offsetWidth, offsetHeight } = container;
const { innerWidth, innerHeight } = window;
const left = clientX - offsetWidth / 2;
const top = clientY - offsetHeight / 2;
container.style.left = `${Math.max(
0,
Math.min(left, innerWidth - offsetWidth)
)}px`;
container.style.top = `${Math.max(
0,
Math.min(top, innerHeight - offsetHeight)
)}px`;
container.style.right = "auto";
dragFrame.style.display = "none";
const savedPosition = JSON.parse(
GM_getValue("timeTrackerPosition", "{}")
);
savedPosition.left = parseInt(container.style.left);
savedPosition.top = parseInt(container.style.top);
GM_setValue("timeTrackerPosition", JSON.stringify(savedPosition));
container.style.opacity = "1";
});
}
/**
* 调整容器样式
*/
adjustContainerStyle(container, isShrinked) {
if (isShrinked) {
container.style.width = "auto";
container.style.padding = "8px 12px";
if (this.statsButton) {
this.statsButton.style.display = "none";
}
} else {
container.style.width = "fit-content";
container.style.padding = "12px 20px";
if (this.statsButton) {
this.statsButton.style.display = "block";
}
}
}
/**
* 创建固定元素
*/
createFixedElement(tag, options) {
const element = document.createElement(tag);
Object.assign(element.style, options.style);
if (options.text) {
element.textContent = options.text;
}
return element;
}
/**
* 创建时间容器
*/
createTimeContainer() {
const timeContainer = document.createElement("div");
timeContainer.style.display = "flex";
timeContainer.style.alignItems = "center";
timeContainer.style.gap = "10px";
// 创建 SVG 时钟图标
const svg = document.createElementNS("http://www.w3.org/2000/svg", "svg");
svg.setAttribute("viewBox", "0 0 24 24");
svg.setAttribute("width", "20");
svg.setAttribute("height", "20");
svg.style.fill = "#666";
const path = document.createElementNS(
"http://www.w3.org/2000/svg",
"path"
);
path.setAttribute(
"d",
"M12,2A10,10 0 0,0 2,12A10,10 0 0,0 12,22A10,10 0 0,0 22,12A10,10 0 0,0 12,2M12,4A8,8 0 0,1 20,12A8,8 0 0,1 12,20A8,8 0 0,1 4,12A8,8 0 0,1 12,4M12.5,7H11V13L16.2,16.2L17,14.9L12.5,12.2V7Z"
);
svg.appendChild(path);
timeContainer.appendChild(svg);
this.timeDisplay = document.createElement("div");
this.timeDisplay.style.cursor = "pointer";
this.timeLabel = document.createElement("div");
this.timeLabel.style.fontSize = "12px";
this.timeLabel.style.color = "#666";
this.timeLabel.textContent = "今日在线";
this.timeValue = document.createElement("div");
this.timeValue.style.fontSize = "15px";
this.timeValue.style.color = "#333";
this.timeValue.style.fontWeight = "600";
this.timeDisplay.appendChild(this.timeLabel);
this.timeDisplay.appendChild(this.timeValue);
timeContainer.appendChild(this.timeDisplay);
return timeContainer;
}
/**
* 创建分隔线
*/
createDivider() {
const divider = document.createElement("div");
divider.style.width = "1px";
divider.style.height = "24px";
divider.style.backgroundColor = "#eee";
return divider;
}
/**
* 创建统计按钮
*/
createStatsButton() {
const statsButton = document.createElement("button");
statsButton.textContent = "查看统计";
Object.assign(statsButton.style, {
padding: "6px 12px",
fontSize: "14px",
fontWeight: "500",
color: "#2196F3",
backgroundColor: "rgba(33, 150, 243, 0.1)",
border: "1px solid rgba(33, 150, 243, 0.2)",
borderRadius: "6px",
cursor: "pointer",
transition: "all 0.2s ease",
outline: "none",
whiteSpace: "nowrap",
});
return statsButton;
}
/**
* 添加按钮悬停效果
*/
addButtonHoverEffects(button) {
const hoverStyle = {
backgroundColor: "rgba(33, 150, 243, 0.15)",
borderColor: "rgba(33, 150, 243, 0.4)",
};
const defaultStyle = {
backgroundColor: "rgba(33, 150, 243, 0.1)",
borderColor: "rgba(33, 150, 243, 0.2)",
};
button.addEventListener("mouseover", () => {
Object.assign(button.style, hoverStyle);
});
button.addEventListener("mouseout", () => {
Object.assign(button.style, defaultStyle);
});
}
/**
* 添加按钮点击效果
*/
addButtonClickEffect(button) {
button.style.transform = "scale(0.95)";
setTimeout(() => {
button.style.transform = "scale(1)";
}, 100);
}
/**
* 添加容器悬停效果
*/
addContainerHoverEffects(container) {
const hoverStyle = {
transform: "translateY(-1px)",
boxShadow: "0 4px 15px rgba(0, 0, 0, 0.08)",
};
const defaultStyle = {
transform: "translateY(0)",
boxShadow: "0 2px 12px rgba(0, 0, 0, 0.1)",
};
container.addEventListener("mouseover", () => {
Object.assign(container.style, hoverStyle);
});
container.addEventListener("mouseout", () => {
Object.assign(container.style, defaultStyle);
});
}
/**
* 更新显示
*/
updateDisplay() {
const today = this.getBeijingDate();
const currentDomain = window.location.hostname
.split(".")
.slice(-2)
.join(".");
const data = this.getStorageData();
const todayTime =
data[currentDomain] && data[currentDomain][today]
? data[currentDomain][today]
: 0;
const elapsedSinceLastUpdate =
this.startTime && this.isTabActive
? Math.floor((Date.now() - this.startTime) / 1000)
: 0;
const totalTime = todayTime + elapsedSinceLastUpdate;
this.timeValue.textContent = this.formatTime(totalTime);
this.animationFrameId = requestAnimationFrame(() => this.updateDisplay());
}
/**
* 获取存储数据
*/
getStorageData() {
return JSON.parse(GM_getValue("websiteTimeTracker", "{}"));
}
/**
* 设置存储数据
*/
setStorageData(data) {
GM_setValue("websiteTimeTracker", JSON.stringify(data));
}
/**
* 展示统计信息
*/
showStats() {
if (this.isStatsShown) return;
this.isStatsShown = true;
const hasSecurityRestrictions = () => {
try {
const script = document.createElement("script");
script.textContent = 'console.log("test")';
document.head.appendChild(script);
document.head.removeChild(script);
return false;
} catch (e) {
return true;
}
};
const overlay = this.createOverlay();
const modal = this.createModal();
const title = this.createTitle("在线时间统计");
modal.appendChild(title);
const useTableView =
hasSecurityRestrictions() || typeof echarts === "undefined";
if (useTableView) {
const tableContainer = document.createElement("div");
tableContainer.style.padding = "20px";
tableContainer.style.overflow = "auto";
this.createDataTable(tableContainer);
modal.appendChild(tableContainer);
} else {
const chartContainer = this.createChartContainer();
modal.appendChild(chartContainer);
setTimeout(() => {
try {
const myChart = echarts.init(chartContainer);
this.updateChart(myChart);
window.addEventListener("resize", () => {
myChart.resize();
});
} catch (error) {
modal.removeChild(chartContainer);
const tableContainer = document.createElement("div");
tableContainer.style.padding = "20px";
tableContainer.style.overflow = "auto";
this.createDataTable(tableContainer);
modal.appendChild(tableContainer);
}
}, 100);
}
overlay.addEventListener("click", () => {
document.body.removeChild(modal);
document.body.removeChild(overlay);
this.isStatsShown = false;
this.updateGMStorage(0);
});
document.body.appendChild(overlay);
document.body.appendChild(modal);
}
/**
* 创建数据表格
*/
createDataTable(container) {
const data = this.getStorageData();
const currentDomain = window.location.hostname
.split(".")
.slice(-2)
.join(".");
const domainData = data[currentDomain] || {};
const today = this.getBeijingDate();
const table = document.createElement("table");
Object.assign(table.style, {
width: "100%",
borderCollapse: "collapse",
textAlign: "center",
fontSize: "14px",
});
const thead = document.createElement("thead");
const headerRow = document.createElement("tr");
["日期", "在线时长"].forEach((text) => {
const th = document.createElement("th");
th.textContent = text;
Object.assign(th.style, {
padding: "10px",
backgroundColor: "#f5f5f5",
borderBottom: "2px solid #ddd",
});
headerRow.appendChild(th);
});
thead.appendChild(headerRow);
table.appendChild(thead);
const tbody = document.createElement("tbody");
const dates = Object.keys(domainData).sort().reverse();
dates.forEach((date) => {
const tr = document.createElement("tr");
const dateTd = document.createElement("td");
dateTd.textContent = date;
dateTd.style.padding = "10px";
dateTd.style.borderBottom = "1px solid #ddd";
tr.appendChild(dateTd);
const timeTd = document.createElement("td");
let seconds = domainData[date];
if (date === today) {
seconds += this.startTime
? Math.floor((Date.now() - this.startTime) / 1000)
: 0;
}
timeTd.textContent = this.formatTime(seconds);
timeTd.style.padding = "10px";
timeTd.style.borderBottom = "1px solid #ddd";
tr.appendChild(timeTd);
tbody.appendChild(tr);
});
table.appendChild(tbody);
container.appendChild(table);
}
/**
* 创建遮罩层
*/
createOverlay() {
return this.createFixedElement("div", {
style: {
position: "fixed",
top: "0",
left: "0",
width: "100%",
height: "100%",
backgroundColor: "rgba(0, 0, 0, 0.5)",
zIndex: 9998,
},
});
}
/**
* 创建弹窗
*/
createModal() {
return this.createFixedElement("div", {
style: {
...DEFAULT_STYLES,
top: "50%",
left: "50%",
transform: "translate(-50%, -50%)",
width: "80%",
maxWidth: "1200px",
height: "600px",
backgroundColor: "white",
zIndex: 9999,
padding: "20px",
boxSizing: "border-box",
display: "flex",
flexDirection: "column",
},
});
}
/**
* 创建标题
*/
createTitle(text) {
const title = document.createElement("div");
title.textContent = text;
Object.assign(title.style, {
fontSize: "18px",
fontWeight: "bold",
textAlign: "center",
marginBottom: "20px",
});
return title;
}
/**
* 创建图表容器
*/
createChartContainer() {
const chartContainer = document.createElement("div");
Object.assign(chartContainer.style, {
flex: 1,
width: "100%",
});
return chartContainer;
}
/**
* 更新图表数据
*/
updateChart(chart) {
const data = this.getStorageData();
const currentDomain = window.location.hostname
.split(".")
.slice(-2)
.join(".");
const domainData = data[currentDomain] || {};
const today = this.getBeijingDate();
const { factor } = this.determineUnit();
const currentTimeInSeconds = this.startTime
? Math.floor((Date.now() - this.startTime) / 1000)
: 0;
// 获取并排序日期
const dates = Object.keys(domainData).sort();
const values = dates.map((date) => {
const value = domainData[date];
return date === today
? Math.floor((value + currentTimeInSeconds) / factor)
: Math.floor(value / factor);
});
// 格式化日期显示
const formatDates = dates.map((date, index) => {
const [year, month, day] = date.split("-");
if (index === 0 || day === "01") {
return `${month}-${day}`;
} else {
return day;
}
});
const option = {
grid: {
top: 50,
right: 50,
bottom: 50,
left: 70,
},
tooltip: {
trigger: "axis",
formatter: (params) => {
const date = dates[params[0].dataIndex];
return `${date}<br/>在线时长:${params[0].value} 分钟`;
},
},
dataZoom: [
{
type: "slider",
show: true,
xAxisIndex: 0,
startValue: Math.max(0, dates.length - 30),
endValue: dates.length - 1,
},
],
xAxis: {
type: "category",
data: formatDates,
axisLabel: {
interval: 0,
rotate: 0,
margin: 15,
color: "#666",
fontSize: 12,
formatter: (value) => value,
},
axisTick: {
alignWithLabel: true,
},
axisLine: {
lineStyle: {
color: "#999",
},
},
},
yAxis: {
type: "value",
name: "时间(分钟)",
nameLocation: "middle",
nameGap: 50,
splitLine: {
lineStyle: {
type: "dashed",
color: "#eee",
},
},
axisLabel: {
color: "#666",
},
},
series: [
{
name: "在线时长",
type: "line",
data: values,
smooth: true,
showSymbol: true,
symbolSize: 6,
lineStyle: {
width: 2,
color: "#2196F3",
},
itemStyle: {
color: "#2196F3",
},
areaStyle: {
color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [
{
offset: 0,
color: "rgba(33, 150, 243, 0.3)",
},
{
offset: 1,
color: "rgba(33, 150, 243, 0.1)",
},
]),
},
},
],
};
chart.setOption(option);
}
/**
* 确定时间单位
*/
determineUnit() {
return { unit: "minutes", factor: 60 };
}
/**
* 初始化
*/
initialize() {
if (window !== window.top) {
return;
}
this.startTime = Date.now();
this.setupEventListeners();
this.createUIElements();
this.animationFrameId = requestAnimationFrame(() => this.updateDisplay());
const savedPosition = JSON.parse(
GM_getValue("timeTrackerPosition", "{}")
);
if (savedPosition && savedPosition.isShrinked) {
this.adjustContainerStyle(
document.querySelector('div[draggable="true"]'),
true
);
}
}
/**
* 设置事件监听
*/
setupEventListeners() {
window.addEventListener("focus", () => {
this.startTime = Date.now();
});
window.addEventListener("blur", () => this.logTime());
window.addEventListener("beforeunload", () => this.logTime());
document.addEventListener("visibilitychange", () => {
if (document.visibilityState === "visible") {
this.isTabActive = true;
this.startTime = Date.now();
} else {
this.isTabActive = false;
this.logTime();
}
});
}
}
// 启动应用
new TimeTracker();
})();