website-time-tracker

用于追踪和统计用户在每个二级域名下的在线时长,并提供友好的可视化统计界面

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

(У мене вже є менеджер скриптів, дайте мені встановити його!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==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();
})();