您需要先安装一个扩展,例如 篡改猴、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/echarts@5.4.3/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();
- })();