网页内容选择截图

基于SnapDOM插件实现的网页内容选取截图脚本

// ==UserScript==
// @name         网页内容选择截图
// @version      0.0.15
// @author       ymzhao
// @description  基于SnapDOM插件实现的网页内容选取截图脚本
// @license      MIT
// @icon         
// @include      *
// @match        *
// @require      https://unpkg.com/@zumer/snapdom@latest/dist/snapdom.min.js
// @grant        GM_addStyle
// @grant        GM_registerMenuCommand
// @namespace https://greasyfork.org/users/703945
// ==/UserScript==

(function () {
  'use strict';

  const d=new Set;const importCSS = async e=>{d.has(e)||(d.add(e),(t=>{typeof GM_addStyle=="function"?GM_addStyle(t):document.head.appendChild(document.createElement("style")).append(t);})(e));};

  const indexCss = ".snap-target{box-shadow:inset 0 0 4px 2px green,0 0 10px 4px green!important}";
  importCSS(indexCss);
  function manualDelay(delay = 500) {
    return new Promise((resolve) => setTimeout(resolve, delay));
  }
  function shining(message, color = "rgb(50, 177, 108)", fontSize = "xxx-large") {
    if (!message) return;
    const i = document.createElement("span");
    i.textContent = message;
    i.style.cssText = `
    position: fixed;
    top: 50%;
    left: 50%;
    color: ${color};
    font-size: ${fontSize};
    font-weight: bold;
    transform: translate(-50%, -50%);
    user-select: none;
    z-index: 2002;`;
    document.body.appendChild(i);
    const duration = 2500;
    i.animate(
      [{ top: "50%", opacity: 1 }, { top: "30%", opacity: 0 }],
      { duration, fill: "forwards" }
    );
    setTimeout(i.remove, duration);
  }
  class SnapdomUtil {
constructor() {
      this._getSnapDom();
      this._isChosing = false;
      this._loading = false;
      this._hoverEl = null;
    }
    get isChosing() {
      return this._isChosing;
    }
    async _getSnapDom() {
      let tmp = window?.snapdom;
      this._snapdom = tmp;
    }
startChosing() {
      if (this._isChosing || this._loading) {
        console.warn("正在选中,已重新开始");
        this.stopChosing();
      }
      this._isChosing = true;
      document.body.addEventListener("mousemove", this._handleMousemove);
      setTimeout(() => {
        document.body.addEventListener("click", this._handleConfirmTarget);
        document.body.addEventListener("keydown", this._handleEsc);
      }, 200);
    }
stopChosing() {
      this._isChosing = false;
      document.body.removeEventListener("mousemove", this._handleMousemove);
      document.body.removeEventListener("click", this._handleConfirmTarget);
      document.body.removeEventListener("keydown", this._handleEsc);
      if (this._hoverEl) {
        this._hoverEl.classList.remove("snap-target");
        this._hoverEl = null;
      }
    }
    destroy() {
      this._loading = false;
      this._isChosing = false;
      this.startChosing = () => {
      };
      this.stopChosing();
    }
_handleMousemove = (e) => {
      if (!this._isChosing || !e.target) return;
      if (this._hoverEl) this._hoverEl.classList.remove("snap-target");
      e.target.classList.add("snap-target");
      this._hoverEl = e.target;
    };

_handleConfirmTarget = async (e) => {
      if (!this._hoverEl) {
        shining("未选中目标", "orange");
        return;
      }
      const tmpEl = this._hoverEl;
      this.stopChosing();
      this._loading = true;
      await manualDelay(50);
      try {
        await this._execSnapDom(tmpEl);
        shining("下载成功");
      } catch (err) {
        shining("下载失败", "red");
        console.error(err);
      }
      this._loading = false;
    };
_handleEsc = (e) => {
      if (e.keyCode !== 27) return;
      this.stopChosing();
    };
async _execSnapDom(targetEl) {
      if (!this._snapdom) {
        console.error("未加载插件snapdom");
        return;
      }
      if (!targetEl) {
        console.error("未指定捕获元素");
        return;
      }
      console.time("捕获耗时:");
      const capture = await this._snapdom(targetEl, { embedFonts: true, compress: true });
      console.timeEnd("捕获耗时:");
      console.time("下载耗时:");
      const options = {
        format: "jpg",
        filename: "chart-report-2025",
        backgroundColor: getBackgroundColor(targetEl)
      };
      await capture.download(options);
      console.timeEnd("下载耗时:");
    }
  }
  function getBackgroundColor(element) {
    if (!(element instanceof Element)) return "#FFFFFF";
    let computedStyle = window.getComputedStyle(element);
    let backgroundColor = computedStyle.getPropertyValue("background-color");
    if (backgroundColor === "rgba(0, 0, 0, 0)" || backgroundColor === "transparent") {
      let parentElement = element.parentElement;
      if (parentElement) {
        return getBackgroundColor(parentElement);
      } else {
        return "#FFFFFF";
      }
    }
    return backgroundColor;
  }
  var _GM_registerMenuCommand = (() => typeof GM_registerMenuCommand != "undefined" ? GM_registerMenuCommand : void 0)();
  /**
   * 网页截图插件
   *
   * @description 基于 [snapdom](https://github.com/zumerlab/snapdom) 实现的网页截图功能,开始截图后,可移动鼠标选中网页元素,点击以下载
   * @author [email protected]
   * @license MIT
   */
  const snapdomUtil = new SnapdomUtil();
  try {
    const menu_command_id_1 = _GM_registerMenuCommand(
      "选中并截图",
      snapdomUtil.startChosing.bind(snapdomUtil),
      {
        accessKey: "s",
        autoClose: true,
        title: "点击后,可选中网页元素以截图"
      }
    );
  } catch (error) {
  }
  window.onload = function() {
    document.addEventListener("keydown", function(e) {
      if (e.ctrlKey && e.shiftKey && (e.key === "," || e.key === "<")) {
        e.preventDefault();
        if (snapdomUtil.isChosing) snapdomUtil.stopChosing();
        else snapdomUtil.startChosing();
      }
    });
  };

})();