您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
用于追踪和统计用户在每个二级域名下的在线时长,并提供友好的可视化统计界面
// ==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(); })();