website-time-tracker

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

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