// ==UserScript== // @name 网页调试 // @namespace https://greasyfork.org/zh-CN/scripts/475228 // @supportURL https://github.com/WhiteSevs/TamperMonkeyScript/issues // @version 2024.3.11 // @author WhiteSevs // @description 内置多种网页调试工具,包括:Eruda、vConsole、PageSpy、Chii,可在设置菜单中进行详细配置 // @icon  // @license MIT // @match *://*/* // @run-at document-start // @grant unsafeWindow // @grant GM_registerMenuCommand // @grant GM_unregisterMenuCommand // @grant GM_info // @grant GM_getValue // @grant GM_setValue // @grant GM_deleteValue // @grant GM_setClipboard // @grant GM_getResourceText // @resource Resource_erudaMonitor https://fastly.jsdelivr.net/npm/eruda-monitor // @resource Resource_erudaFeatures https://fastly.jsdelivr.net/npm/eruda-features // @resource Resource_erudaTiming https://fastly.jsdelivr.net/npm/eruda-timing // @resource Resource_erudaCode https://fastly.jsdelivr.net/npm/eruda-code // @resource Resource_erudaBenchmark https://fastly.jsdelivr.net/npm/eruda-benchmark // @resource Resource_Leaflet https://update.greasyfork.org/scripts/483765/1309677/Leaflet.js // @resource Resource_erudaGeolocation https://fastly.jsdelivr.net/gh/WhiteSevs/eruda-geolocation/eruda-geolocation.js // @resource Resource_erudaOrientation https://fastly.jsdelivr.net/gh/WhiteSevs/eruda-orientation/eruda-orientation.js // @resource Resource_erudaTouches https://fastly.jsdelivr.net/npm/eruda-touches // @resource Resource_erudaOutlinePlugin https://fastly.jsdelivr.net/npm/eruda-outline-plugin // @resource Resource_erudaPixel https://fastly.jsdelivr.net/npm/eruda-pixel // @require https://update.greasyfork.org/scripts/456485/1340659/pops.js // @require https://update.greasyfork.org/scripts/483694/1319661/Eruda-2.js // @require https://update.greasyfork.org/scripts/483695/1319662/vConsole-2.js // @require https://update.greasyfork.org/scripts/483696/1331578/PageSpy-2.js // @require https://update.greasyfork.org/scripts/455186/1341032/WhiteSevsUtils.js // @downloadURL none // ==/UserScript== (function () { if(typeof unsafeWindow === "undefined"){ unsafeWindow = globalThis; } /** * @type {import("../库/pops")} */ const pops = window.pops; /** * @type {import("../库/Utils")} */ const utils = window.Utils.noConflict(); /** * 菜单对象 */ let GM_Menu = new utils.GM_Menu({ GM_getValue, GM_setValue, GM_registerMenuCommand, GM_unregisterMenuCommand, }); /** * @type {Window & typeof globalThis} */ let currentWin = this || self; let console = currentWin.console; /** * 配置面板 */ const PopsPanel = { /** * 本地存储的总键名 */ key: "GM_Panel", /** * 属性attributes的data-key */ attributeDataKey_Name: "data-key", /** * 属性attributes的data-default-value */ attributeDataDefaultValue_Name: "data-default-value", /** * 初始化菜单 */ initMenu() { this.initLocalDefaultValue(); if (unsafeWindow.top !== unsafeWindow.self) { return; } GM_Menu.add([ { key: "show_pops_panel_setting", text: "⚙ 设置", autoReload: false, isStoreValue: false, showText(text) { return text; }, callback: () => { this.showPanel(); }, }, ]); }, /** * 初始化本地设置默认的值 */ initLocalDefaultValue() { let content = this.getContent(); content.forEach((item) => { if (!item["forms"]) { return; } item.forms.forEach((__item__) => { if (__item__.forms) { __item__.forms.forEach((containerItem) => { if (!containerItem.attributes) { return; } let key = containerItem.attributes[this.attributeDataKey_Name]; let defaultValue = containerItem.attributes[this.attributeDataDefaultValue_Name]; if (this.getValue(key) == null) { this.setValue(key, defaultValue); } }); } else { } }); }); }, /** * 设置值 * @param {string} key 键 * @param {any} value 值 */ setValue(key, value) { let localValue = GM_getValue(this.key, {}); localValue[key] = value; GM_setValue(this.key, localValue); }, /** * 获取值 * @param {string} key 键 * @param {any} defaultValue 默认值 * @returns {any} */ getValue(key, defaultValue) { let localValue = GM_getValue(this.key, {}); return localValue[key] ?? defaultValue; }, /** * 删除值 * @param {string} key 键 */ deleteValue(key) { let localValue = GM_getValue(this.key, {}); delete localValue[key]; GM_setValue(this.key, localValue); }, /** * 显示设置面板 */ showPanel() { pops.panel({ title: { text: `${GM_info?.script?.name || "网页调试"}`, position: "center", }, content: this.getContent(), mask: { enable: true, clickEvent: { toClose: true, }, }, width: utils.isPhone() ? "92vw" : "70vw", height: utils.isPhone() ? "80vh" : "80vh", drag: true, only: true, zIndex: 200000000, }); }, /** * 获取按钮配置 * @param {string} text 文字 * @param {string} key 键 * @param {boolean} defaultValue 默认值 * @param {?(event:Event,value: boolean)=>boolean} _callback_ 点击回调 * @param {string|undefined} description 描述 */ getSwtichDetail(text, key, defaultValue, _callback_, description) { /** * @type {PopsPanelSwitchDetails} */ let result = { text: text, type: "switch", description: description, attributes: {}, getValue() { return Boolean(PopsPanel.getValue(key, defaultValue)); }, callback(event, value) { console.log(`${value ? "开启" : "关闭"} ${text}`); if (typeof _callback_ === "function") { if (_callback_(event, value)) { return; } } PopsPanel.setValue(key, Boolean(value)); }, }; result.attributes[this.attributeDataKey_Name] = key; result.attributes[this.attributeDataDefaultValue_Name] = Boolean(defaultValue); return result; }, /** * 获取输入框配置 * @param {string} text 文字 * @param {string} key 键 * @param {boolean} defaultValue 默认值 * @param {string} [placeholder=""] 提示 * @param {?(event:Event,value: string)=>boolean} _callback_ 输入回调 * @param {string|undefined} description 描述 */ getInputDetail( text, key, defaultValue, placeholder = "", _callback_, description ) { /** * @type {PopsPanelButtonDetails} */ return { text: text, type: "input", attributes: { "data-key": key, "data-default-value": defaultValue, }, description: description, getValue() { let localValue = PopsPanel.getValue(key, defaultValue); return localValue; }, callback(event, value) { if (typeof _callback_ === "function") { if (_callback_(event, value)) { return; } } PopsPanel.setValue(key, value); }, placeholder: placeholder, }; }, /** * 获取配置内容 */ getContent() { /** * @type {PopsPanelContentConfig[]} */ let content = [ { id: "debug-panel-config-all", title: "总设置", headerTitle: "总设置", forms: [ { text: "功能", type: "forms", forms: [ { text: "调试工具", type: "select", attributes: { "data-key": "currentDebug", "data-default-value": "eruda", }, getValue() { return PopsPanel.getValue( this.attributes["data-key"], this.attributes["data-default-value"] ); }, callback(event, isSelectedValue, isSelectedText) { PopsPanel.setValue( this.attributes["data-key"], isSelectedValue ); }, data: [ { value: "eruda", text: "Eruda", }, { value: "vconsole", text: "VConsole", }, { value: "pagespy", text: "PageSpy", }, { value: "chii", text: "Chii", }, ], }, this.getSwtichDetail( "允许在iframe内运行", "allowRunInIframe", false ), ], }, ], }, { id: "debug-panel-config-eruda", title: "Eruda", headerTitle: "Eruda设置", forms: [ { text: "功能", type: "forms", forms: [ { text: "版本", type: "button", attributes: { "data-key": "eruda-version", "data-default-value": GlobalDebug.erudaVersion, }, buttonType: "primary", buttonText: GlobalDebug.erudaVersion, callback(event) { window.open("https://github.com/liriliri/eruda", "_blank"); }, }, this.getSwtichDetail( "自动打开面板", "eruda-auto-open-panel", false, void 0, "加载完毕后自动显示面板内容" ), ], }, { text: "配置", type: "forms", forms: [ this.getInputDetail( "默认展示的面板元素", "eruda-default-show-panel-name", "Console", "请输入面板", function (event, value) { PopsPanel.setValue( "eruda-default-show-panel-name", value.trim() ); }, "开启【自动打开面板】才会生效" ), ], }, { text: "插件", type: "forms", forms: [ this.getSwtichDetail( "eruda-monitor", "eruda_plugin_Resource_erudaMonitor", false, undefined, "展示页面的 fps 和内存信息" ), this.getSwtichDetail( "eruda-features", "eruda_plugin_Resource_erudaFeatures", false, undefined, "浏览器特性检测" ), this.getSwtichDetail( "eruda-timing", "eruda_plugin_Resource_erudaTiming", false, undefined, "展示性能资源数据" ), this.getSwtichDetail( "eruda-code", "eruda_plugin_Resource_erudaCode", false, undefined, "运行 JavaScript 代码" ), this.getSwtichDetail( "eruda-benchmark", "eruda_plugin_Resource_erudaBenchmark", false, undefined, "运行 JavaScript 性能测试" ), this.getSwtichDetail( "eruda-geolocation", "eruda_plugin_Resource_erudaGeolocation", false, undefined, "测试地理位置接口" ), this.getSwtichDetail( "eruda-orientation", "eruda_plugin_Resource_erudaOrientation", false, undefined, "测试重力感应接口" ), this.getSwtichDetail( "eruda-touches", "eruda_plugin_Resource_erudaTouches", false, undefined, "(暂时无效)可视化屏幕 Touch 事件触发" ), this.getSwtichDetail( "eruda-outline-plugin", "eruda_plugin_Resource_erudaOutlinePlugin", false, undefined, "给页面的元素添加边框" ), this.getSwtichDetail( "eruda-pixel", "eruda_plugin_Resource_erudaPixel", false, undefined, "高精度的UI恢复辅助工具" ), ], }, ], }, { id: "debug-panel-config-vconsole", title: "vConsole", headerTitle: "vConsole设置", forms: [ { text: "功能", type: "forms", forms: [ { text: "版本", type: "button", attributes: { "data-key": "vconsole-version", "data-default-value": GlobalDebug.vConsoleVersion, }, buttonType: "primary", buttonText: GlobalDebug.vConsoleVersion, callback(event) { window.open( "https://github.com/Tencent/vConsole", "_blank" ); }, }, this.getSwtichDetail( "自动打开面板", "vconsole-auto-open-panel", false, void 0, "加载完毕后自动显示面板内容" ), ], }, { text: "配置", type: "forms", forms: [ this.getSwtichDetail( "禁止Log自动滚动", "vconsole-disableLogScrolling", false ), this.getSwtichDetail( "显示日志的输出时间", "vconsole-showTimestamps", false ), { text: "日志的上限数量", type: "input", attributes: { "data-key": "vconsole-maxLogNumber", "data-default-value": 1000, }, getValue() { let localValue = PopsPanel.getValue( this.attributes["data-key"], this.attributes["data-default-value"] ); localValue = parseInt(localValue); if (isNaN(localValue)) { return this.attributes["data-default-value"]; } else { return localValue; } }, callback(event, value) { let inputValue = parseInt(value); if (isNaN(inputValue)) { return; } PopsPanel.setValue(this.attributes["data-key"], inputValue); }, placeholder: "请输入数字", isNumber: true, }, { text: "请求记录的上限数量", type: "input", attributes: { "data-key": "vconsole-maxNetworkNumber", "data-default-value": 1000, }, getValue() { let localValue = PopsPanel.getValue( this.attributes["data-key"], this.attributes["data-default-value"] ); localValue = parseInt(localValue); if (isNaN(localValue)) { return this.attributes["data-default-value"]; } else { return localValue; } }, callback(event, value) { let inputValue = parseInt(value); if (isNaN(inputValue)) { return; } PopsPanel.setValue(this.attributes["data-key"], inputValue); }, placeholder: "请输入数字", isNumber: true, }, ], }, ], }, { id: "debug-panel-config-pagespy", title: "PageSpy", headerTitle: "PageSpy设置", forms: [ { text: "功能", type: "forms", forms: [ { text: "注意!隐私保护!", type: "button", buttonType: "danger", buttonText: "了解详情", callback(event) { pops.confirm({ title: { text: "提示", }, content: { text: `下面默认配置的${GlobalDebug.pageSpyDefaultApi}是仅供测试使用的,其他人也可以看到你的调试信息,包括Cookie等信息,如果想用,请自己搭建一个调试端`, }, btn: { reverse: true, position: "end", ok: { text: "前往了解更多", callback() { window.open( "https://github.com/HuolalaTech/page-spy-web/wiki/%F0%9F%90%9E-%E5%B8%B8%E8%A7%81%E9%97%AE%E9%A2%98%E8%A7%A3%E7%AD%94#user-content-testjikejishucom-%E6%98%AF%E5%AE%98%E6%96%B9%E6%8F%90%E4%BE%9B%E7%9A%84%E5%9F%9F%E5%90%8D%E5%90%97%E4%B8%80%E7%9B%B4%E5%8F%AF%E4%BB%A5%E7%94%A8%E5%90%97", "_blank" ); }, }, }, mask: { enable: true, }, }); }, }, { text: "版本", type: "button", attributes: { "data-key": "pagespy-version", "data-default-value": GlobalDebug.pageSpyVersion, }, buttonType: "primary", buttonText: GlobalDebug.pageSpyVersion, callback(event) { window.open( "https://github.com/HuolalaTech/page-spy-web", "_blank" ); }, }, this.getSwtichDetail( "禁止在调试端运行", "pagespy-disable-run-in-debug-client", true, undefined, "调试端是下面配置的api/clientOrigin地址" ), ], }, { text: "配置", type: "forms", forms: [ this.getInputDetail( "api", "pagespy-api", GlobalDebug.pageSpyDefaultApi, "", function (event, value) { PopsPanel.setValue("pagespy-api", value.trim()); }, "服务器地址的 Host" ), this.getInputDetail( "clientOrigin", "pagespy-clientOrigin", GlobalDebug.pageSpyDefaultCliennOrigin, "", function (event, value) { PopsPanel.setValue("pagespy-clientOrigin", value.trim()); }, "服务器地址的 Origin" ), this.getInputDetail( "project", "pagespy-project", "default", undefined, undefined, "项目名称" ), this.getInputDetail( "title", "pagespy-title", "--", undefined, undefined, "自定义标题" ), this.getSwtichDetail( "autoRender", "pagespy-autoRender", true, undefined, "自动渲染「圆形白底带 Logo」" ), { text: "enableSSL", description: "是否https", type: "select", attributes: { "data-key": "pagespy-enableSSL", "data-default-value": true, }, getValue() { return PopsPanel.getValue( this.attributes["data-key"], this.attributes["data-default-value"] ); }, callback(event, isSelectedValue, isSelectedText) { PopsPanel.setValue( this.attributes["data-key"], isSelectedValue ); }, data: [ { value: null, text: "默认(自动分析)", }, { value: true, text: "开启", }, { value: false, text: "关闭", }, ], }, ], }, ], }, { id: "debug-panel-config-chii", title: "Chii", headerTitle: "Chii设置", forms: [ { text: "功能", type: "forms", forms: [ { text: "调试页面", type: "button", buttonType: "primary", buttonText: "前往", callback(event) { let url = PopsPanel.getValue( "chii-debug-url", GlobalDebug.chiiUrl ); window.open(url, "_blank"); }, }, ], }, { text: "配置", type: "forms", forms: [ this.getSwtichDetail( "本页展示", "chii-script-embedded", true, (event, value) => { let $shadowRoot = event.target.getRootNode(); let button = $shadowRoot.querySelector( "li.pops-panel-forms-container-item ul > li > .pops-panel-button button" ); if (value) { button.setAttribute("disabled", true); } else { button.removeAttribute("disabled"); } }, "将调试器展示在同一页面中" ), this.getSwtichDetail( "禁止在调试端运行", "chii-disable-run-in-debug-url", true, void 0, "调试端是下面配置的【调试页面Url】" ), this.getInputDetail( "调试页面Url", "chii-debug-url", GlobalDebug.chiiUrl, "请输入链接Url", void 0, "配置【调试页面】的Url" ), this.getInputDetail( "来源js", "chii-target-js", GlobalDebug.chiiDetaultScriptJs, "请输入目标js文件", void 0, "用于注入页面来进行调试" ), ], }, ], }, ]; return content; }, }; /** * 全局调试 */ const GlobalDebug = { erudaVersion: "3.0.1", vConsoleVersion: "3.15.1", pageSpyVersion: "1.7.6", pageSpyDefaultApi: "test.jikejishu.com", pageSpyDefaultCliennOrigin: "https://test.jikejishu.com", chiiUrl: "https://chii.liriliri.io/", chiiDetaultScriptJs: "//chii.liriliri.io/target.js", iframeUrlList: [], /** * 处理当在iframe内加载时,是否允许执行,如果允许,那么把url添加到菜单中 */ handleIframe() { if (top === self) { return true; } if (!PopsPanel.getValue("allowRunInIframe")) { return false; } this.iframeUrlList.push(window.location.href); top.console.log("iframe信息:" + window.location.href); GM_Menu.add({ key: "iframeUrl", text: window.location.href, autoReload: false, isStoreValue: false, showText(text) { return text; }, callback() { GM_setClipboard(window.location.href); }, }); return true; }, /** * 执行当前的调试工具 */ runDebugTool() { /* 当前的调试工具,默认为eruda */ let currentDebugTool = PopsPanel.getValue("currentDebug"); currentDebugTool = currentDebugTool.toString().toLowerCase(); console.log(`网页调试:当前使用的调试工具【${currentDebugTool}】`); if (currentDebugTool === "vconsole") { /* vConsole */ this.vConsole(); } else if (currentDebugTool === "pagespy") { /* PageSpy */ this.pageSpy(); } else if (currentDebugTool === "eruda") { /* eruda */ this.eruda(); } else if (currentDebugTool === "chii") { this.chii(); } else { console.error("当前未配置该调试工具的运行"); } }, eruda() { initEruda("Eruda", currentWin); let eruda = currentWin.Eruda; if (!eruda) { alert("调试工具【eruda】注册全局失败,请反馈开发者"); return; } GlobalDebug.erudaVersion = eruda.version; eruda.init(); console.log(`eruda当前版本:${eruda.version}`); console.log(`eruda项目地址:https://github.com/liriliri/eruda`); unsafeWindow._eruda_ = eruda; console.log("eruda的全局变量名:_eruda_"); if (PopsPanel.getValue("eruda_plugin_Resource_erudaMonitor")) { try { eval(GM_getResourceText("Resource_erudaMonitor")); eruda.add(erudaMonitor); } catch (error) { console.error("插件【eruda-monitor】加载失败,原因:", error); } } if (PopsPanel.getValue("eruda_plugin_Resource_erudaFeatures")) { try { eval(GM_getResourceText("Resource_erudaFeatures")); eruda.add(erudaFeatures); } catch (error) { console.error("插件【eruda-features】加载失败,原因:", error); } } if (PopsPanel.getValue("eruda_plugin_Resource_erudaTiming")) { try { eval(GM_getResourceText("Resource_erudaTiming")); eruda.add(erudaTiming); } catch (error) { console.error("插件【eruda-timing】加载失败,原因:", error); } } if (PopsPanel.getValue("eruda_plugin_Resource_erudaCode")) { try { eval(GM_getResourceText("Resource_erudaCode")); eruda.add(erudaCode); } catch (error) { console.error("插件【eruda-code】加载失败,原因:", error); } } if (PopsPanel.getValue("eruda_plugin_Resource_erudaBenchmark")) { try { eval(GM_getResourceText("Resource_erudaBenchmark")); eruda.add(erudaBenchmark); } catch (error) { console.error("插件【eruda-benchmark】加载失败,原因:", error); } } if (PopsPanel.getValue("eruda_plugin_Resource_erudaGeolocation")) { try { eval(GM_getResourceText("Resource_Leaflet")); eval(GM_getResourceText("Resource_erudaGeolocation")); eruda.add(erudaGeolocation); } catch (error) { console.error("插件【eruda-geolocation】加载失败,原因:", error); } } if (PopsPanel.getValue("eruda_plugin_Resource_erudaOrientation")) { try { currentWin.eval(GM_getResourceText("Resource_erudaOrientation")); eruda.add(currentWin.erudaOrientation); } catch (error) { console.error("插件【eruda-orientation】加载失败,原因:", error); } } if (PopsPanel.getValue("eruda_plugin_Resource_erudaTouches")) { try { eval(GM_getResourceText("Resource_erudaTouches")); eruda.add(erudaTouches); } catch (error) { console.error("插件【eruda-touches】加载失败,原因:", error); } } if (PopsPanel.getValue("eruda_plugin_Resource_erudaOutlinePlugin")) { try { eval(GM_getResourceText("Resource_erudaOutlinePlugin")); eruda.add(erudaOutlinePlugin); } catch (error) { console.error("插件【eruda-outline-plugin】加载失败,原因:", error); } } if (PopsPanel.getValue("eruda_plugin_Resource_erudaPixel")) { try { eval(GM_getResourceText("Resource_erudaPixel")); eruda.add(erudaPixel); } catch (error) { console.error("插件【eruda-pixel】加载失败,原因:", error); } } if (PopsPanel.getValue("eruda-auto-open-panel")) { let defaultShowName = PopsPanel.getValue( "eruda-default-show-panel-name", "" ); eruda.show(); setTimeout(() => { eruda.show(defaultShowName); }, 250); } }, vConsole() { initVConsole("VConsole", currentWin); let vConsole = currentWin.VConsole; if (!vConsole) { alert("调试工具【vConsole】注册全局失败,请反馈开发者"); return; } let vconsole = new vConsole({ defaultPlugins: ["system", "network", "element", "storage"], theme: "light", disableLogScrolling: PopsPanel.getValue("vconsole-disableLogScrolling"), maxLogNumber: PopsPanel.getValue("vconsole-maxLogNumber"), onReady() { if (PopsPanel.getValue("vconsole-auto-open-panel")) { vconsole.show(); } }, }); GlobalDebug.vConsoleVersion = vconsole.version; vconsole.setOption( "log.showTimestamps", PopsPanel.getValue("vconsole-showTimestamps") ); vconsole.setOption( "log.maxLogNumber", PopsPanel.getValue("vconsole-maxLogNumber", 1000) ); vconsole.setOption( "log.maxNetworkNumber", PopsPanel.getValue("vconsole-maxNetworkNumber", 1000) ); console.log(`VConsole当前版本:${vconsole.version}`); console.log(`VConsole项目地址:https://github.com/Tencent/vConsole`); unsafeWindow._vConsole_ = vconsole; console.log("VConsole的全局变量名:_vConsole_"); }, pageSpy() { let api = PopsPanel.getValue( "pagespy-api", GlobalDebug.pageSpyDefaultApi ); let clientOrigin = PopsPanel.getValue( "pagespy-clientOrigin", GlobalDebug.pageSpyDefaultCliennOrigin ); if (PopsPanel.getValue("pagespy-disable-run-in-debug-client")) { if (window.location.hostname.includes(api)) { return; } if (window.location.origin.includes(clientOrigin)) { return; } } let __pageSpy__ = new initPageSpy(); if (!__pageSpy__) { alert("调试工具【PageSpy】获取失败,请反馈开发者"); return; } let $pageSpy = new __pageSpy__({ // SDK 会从引入的路径自动分析并决定 Server 的地址(api)和调试端的地址(clientOrigin) // 假设你从 https://example.com/page-spy/index.min.js 引入,那么 SDK 会在内部设置: // - api: "example.com" // - clientOrigin: "https://example.com" // 如果你的服务部署在别处,就需要在这里手动指定去覆盖。 api: api, clientOrigin: clientOrigin, // project 作为信息的一种聚合,可以在调试端房间列表进行搜索 project: PopsPanel.getValue("pagespy-project", true), // title 供用户提供自定义参数,可以用于区分当前调试的客户端 // 对应的信息显示在每个调试连接面板的「设备id」下方 title: PopsPanel.getValue("pagespy-title", true), // 指示 SDK 初始化完成,是否自动在客户端左下角渲染「圆形白底带 Logo」的控件 // 如果设置为 false, 可以调用 window.$pageSpy.render() 手动渲染 autoRender: PopsPanel.getValue("pagespy-autoRender", true), // 手动指定 PageSpy 服务的 scheme。 // 这在 SDK 无法正确分析出 scheme 可以使用,例如 PageSpy 的浏览器插件 // 是通过 chrome-extension://xxx/sdk/index.min.js 引入 SDK,这会 // 被 SDK 解析成无效的 "chrome-extension://" 并回退到 ["http://", "ws://"]。 // - (默认)传值 undefined 或者 null:SDK 会自动分析; // - 传递 boolean 值: // - true:SDK 将通过 ["https://", "wss://"] 访问 PageSpy 服务 // - false:SDK 将通过 ["http://", "ws://"] 访问 PageSpy 服务 enableSSL: PopsPanel.getValue("pagespy-enableSSL", true), }); unsafeWindow.$pageSpy = $pageSpy; console.log($pageSpy); GlobalDebug.pageSpyVersion = unsafeWindow.$pageSpy.version; console.log("PageSpy全局变量:$pageSpy"); utils .waitPropertyByInterval( unsafeWindow.$pageSpy, function () { return unsafeWindow.$pageSpy.root != null; }, 250, 10000 ) .then(() => { /** * @type {HTMLElement} */ let contentElement = unsafeWindow.$pageSpy.root.querySelector(".page-spy-content"); let goToRoomListElement = document.createElement("div"); let goToDebugElement = document.createElement("div"); goToDebugElement.className = "page-spy-content__btn"; goToDebugElement.innerHTML = "前往调试"; goToRoomListElement.className = "page-spy-content__btn"; goToRoomListElement.innerHTML = "前往房间列表"; goToDebugElement.addEventListener( "click", function () { window.open( `${clientOrigin}/#/devtools?version=${encodeURIComponent( unsafeWindow.$pageSpy.name )}&address=${encodeURIComponent( unsafeWindow.$pageSpy.address )}`, "_blank" ); }, { capture: true, } ); goToRoomListElement.addEventListener( "click", function () { window.open(`${clientOrigin}/#/room-list`, "_blank"); }, { capture: true, } ); contentElement.appendChild(goToRoomListElement); contentElement.appendChild(goToDebugElement); }); }, chii() { let debugUrl = PopsPanel.getValue("chii-debug-url", this.chiiUrl); if ( window.location.href.startsWith(debugUrl) && PopsPanel.getValue("chii-disable-run-in-debug-url", true) ) { console.log("禁止在调试端运行"); return; } let scriptJs = PopsPanel.getValue( "chii-target-js", this.chiiDetaultScriptJs ); let scriptEmbedded = PopsPanel.getValue("chii-script-embedded", true); let scriptNode = document.createElement("script"); scriptNode.src = scriptJs; scriptNode.setAttribute("type", "application/javascript"); if (scriptEmbedded) { scriptNode.setAttribute("embedded", true); } unsafeWindow.addEventListener( "error", function (event) { if (event.target === scriptNode) { globalThis.alert( `调试工具【Chii】脚本加载失败 可能原因1:CSP策略阻止了加载第三方域的js文件 可能原因2:目标js无效` ); } }, { capture: true, } ); (document.head || document.body || document.documentElement).appendChild( scriptNode ); }, }; PopsPanel.initMenu(); if (GlobalDebug.handleIframe()) { GlobalDebug.runDebugTool(); } })();