// ==UserScript==
// @name 网页调试
// @namespace https://gf.qytechs.cn/zh-CN/scripts/475228
// @supportURL https://github.com/WhiteSevs/TamperMonkeyScript/issues
// @version 2024.7.22
// @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.gf.qytechs.cn/scripts/483765/1360579/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
// @resource Resource_vConsoleVueDevtools https://fastly.jsdelivr.net/npm/[email protected]/dist/vue_plugin.min.js
// @require https://update.gf.qytechs.cn/scripts/483694/1414798/Eruda-2.js
// @require https://update.gf.qytechs.cn/scripts/483695/1360577/vConsole-2.js
// @require https://update.gf.qytechs.cn/scripts/483696/1412242/PageSpy-2.js
// @require https://fastly.jsdelivr.net/npm/@whitesev/[email protected]/dist/index.umd.js
// @require https://fastly.jsdelivr.net/npm/@whitesev/[email protected]/dist/index.umd.min.js
// ==/UserScript==
(function () {
if (typeof unsafeWindow === "undefined") {
var unsafeWindow =
globalThis.unsafeWindow ||
window.unsafeWindow ||
globalThis ||
window ||
self;
}
/**
* @type {Window & typeof globalThis}
*/
let unsafeWin = unsafeWindow;
let console = unsafeWin.console;
/** @type {import("@whitesev/pops").default} */
const pops = window.pops || unsafeWin.pops;
/** @type {import("@whitesev/utils").default */
const utils = (window.Utils || unsafeWin.Utils).noConflict();
/**
* 菜单对象
*/
const GM_Menu = new utils.GM_Menu({
GM_getValue,
GM_setValue,
GM_registerMenuCommand,
GM_unregisterMenuCommand,
});
/**
* @typedef PosPanelListenerData
* @property {number} id
* @property {string} key
* @property {(key: string, oldValue: any, newValue: any) => void} callback
*/
/**
* 配置面板
*/
const PopsPanel = {
/** 数据 */
$data: {
/** @type {import("@whitesev/utils/dist/src/Dictionary").UtilsDictionary<string, any>} */
__data: null,
/** @type {import("@whitesev/utils/dist/src/Dictionary").UtilsDictionary<string, number>} */
__oneSuccessExecMenu: null,
/** @type {import("@whitesev/utils/dist/src/Dictionary").UtilsDictionary<string, number>} */
__onceExec: null,
/** @type {import("@whitesev/utils/dist/src/Dictionary").UtilsDictionary<string, PosPanelListenerData>} */
__listenData: null,
/**
* 菜单项的默认值
*/
get data() {
if (PopsPanel.$data.__data == null) {
PopsPanel.$data.__data = new utils.Dictionary();
}
return PopsPanel.$data.__data;
},
/**
* 成功只执行了一次的项
*/
get oneSuccessExecMenu() {
if (PopsPanel.$data.__oneSuccessExecMenu == null) {
PopsPanel.$data.__oneSuccessExecMenu = new utils.Dictionary();
}
return PopsPanel.$data.__oneSuccessExecMenu;
},
/**
* 成功只执行了一次的项
*/
get onceExec() {
if (PopsPanel.$data.__onceExec == null) {
PopsPanel.$data.__onceExec = new utils.Dictionary();
}
return PopsPanel.$data.__onceExec;
},
/** 脚本名,一般用在设置的标题上 */
get scriptName() {
return SCRIPT_NAME;
},
/** 菜单项的总值在本地数据配置的键名 */
key: "GM_Panel",
/** 菜单项在attributes上配置的菜单键 */
attributeKeyName: "data-key",
/** 菜单项在attributes上配置的菜单默认值 */
attributeDefaultValueName: "data-default-value",
},
/** 监听器 */
$listener: {
/**
* 值改变的监听器
*/
get listenData() {
if (PopsPanel.$data.__listenData == null) {
PopsPanel.$data.__listenData = new utils.Dictionary();
}
return PopsPanel.$data.__listenData;
},
},
init() {
this.initPanelDefaultValue();
this.initExtensionsMenu();
},
/** 判断是否是顶层窗口 */
isTopWindow() {
return unsafeWindow.window.self === unsafeWindow.window.top;
},
/** 初始化进行注册(不可用)油猴菜单 */
initExtensionsMenu() {
if (!this.isTopWindow()) {
return;
}
GM_Menu.add([
{
key: "show_pops_panel_setting",
text: "⚙ 设置",
autoReload: false,
isStoreValue: false,
showText(text) {
return text;
},
callback() {
PopsPanel.showPanel();
},
},
]);
},
/**
* 初始化本地设置默认的值
*/
initPanelDefaultValue() {
let that = this;
function initDefaultValue(config) {
if (!config["attributes"]) {
return;
}
let key = config.attributes[that.$data.attributeKeyName];
let defaultValue =
config["attributes"][that.$data.attributeDefaultValueName];
if (key == null) {
console.warn(["请先配置键", config]);
return;
}
if (that.$data.data.has(key)) {
console.warn("请检查该key(已存在): " + key);
}
that.$data.data.set(key, defaultValue);
}
function loopInitDefaultValue(configList) {
for (let index = 0; index < configList.length; index++) {
let configItem = configList[index];
initDefaultValue(configItem);
let childForms = configItem.forms;
if (childForms && Array.isArray(childForms)) {
loopInitDefaultValue(childForms);
}
}
}
let contentConfigList = this.getPanelContentConfig();
for (let index = 0; index < contentConfigList.length; index++) {
let leftContentConfigItem = contentConfigList[index];
if (!leftContentConfigItem.forms) {
continue;
}
let rightContentConfigList = leftContentConfigItem.forms;
if (rightContentConfigList && Array.isArray(rightContentConfigList)) {
loopInitDefaultValue(rightContentConfigList);
}
}
},
/**
* 获取值
* @param key 键
* @param defaultValue 默认值
*/
getValue(key, defaultValue) {
let locaData = GM_getValue(this.$data.key, {});
let localValue = locaData[key];
if (localValue == null) {
if (this.$data.data.has(key)) {
return this.$data.data.get(key);
}
return defaultValue;
}
return localValue;
},
/**
* 删除值
* @param key 键
*/
deleteValue(key) {
let locaData = GM_getValue(this.$data.key, {});
let oldValue = locaData[key];
Reflect.deleteProperty(locaData, key);
GM_setValue(this.$data.key, locaData);
if (this.$listener.listenData.has(key)) {
this.$listener.listenData.get(key).callback(key, oldValue, void 0);
}
},
/**
* 监听调用setValue、deleteValue
* @param key 需要监听的键
* @param callback
*/
addValueChangeListener(key, callback, option) {
let listenerId = Math.random();
this.$listener.listenData.set(key, {
id: listenerId,
key,
callback,
});
if (option) {
if (option.immediate) {
callback(key, this.getValue(key), this.getValue(key));
}
}
return listenerId;
},
/**
* 移除监听
* @param listenerId 监听的id
*/
removeValueChangeListener(listenerId) {
let deleteKey = null;
for (const [key, value] of this.$listener.listenData.entries()) {
if (value.id === listenerId) {
deleteKey = key;
break;
}
}
if (typeof deleteKey === "string") {
this.$listener.listenData.delete(deleteKey);
} else {
console.warn("没有找到对应的监听器");
}
},
/**
* 判断该键是否存在
* @param key 键
*/
hasKey(key) {
let locaData = GM_getValue(this.$data.key, {});
return key in locaData;
},
/**
* 自动判断菜单是否启用,然后执行回调
* @param key
* @param callback 回调
* @param [isReverse=false] 逆反判断菜单启用
*/
execMenu(key, callback, isReverse = false) {
if (typeof key !== "string") {
throw new TypeError("key 必须是字符串");
}
if (!this.$data.data.has(key)) {
console.warn(`${key} 键不存在`);
return;
}
let value = PopsPanel.getValue(key);
if (isReverse) {
value = !value;
}
if (value) {
callback(value);
}
},
/**
* 自动判断菜单是否启用,然后执行回调,只会执行一次
* @param key
* @param callback 回调
* @param [isReverse=false] 逆反判断菜单启用
*/
execMenuOnce(key, callback, isReverse = false) {
if (typeof key !== "string") {
throw new TypeError("key 必须是字符串");
}
if (!this.$data.data.has(key)) {
console.warn(`${key} 键不存在`);
return;
}
if (this.$data.oneSuccessExecMenu.has(key)) {
return;
}
this.$data.oneSuccessExecMenu.set(key, 1);
let resultStyleList = [];
let pushStyleNode = (style) => {
let __value = PopsPanel.getValue(key);
changeCallBack(__value, style);
};
let changeCallBack = (currentValue, resultStyle) => {
let resultList = [];
if (currentValue) {
let result = resultStyle ?? callback(currentValue, pushStyleNode);
if (result instanceof HTMLStyleElement) {
resultList = [result];
} else if (Array.isArray(result)) {
resultList = [
...result.filter(
(item) => item != null && item instanceof HTMLStyleElement
),
];
}
}
for (let index = 0; index < resultStyleList.length; index++) {
let $css = resultStyleList[index];
$css.remove();
resultStyleList.splice(index, 1);
index--;
}
resultStyleList = [...resultList];
};
this.addValueChangeListener(key, (__key, oldValue, newValue) => {
if (isReverse) {
newValue = !newValue;
}
changeCallBack(newValue);
});
let value = PopsPanel.getValue(key);
if (isReverse) {
value = !value;
}
if (value) {
changeCallBack(value);
}
},
/**
* 根据自定义key只执行一次
* @param key 自定义key
*/
onceExec(key, callback) {
if (typeof key !== "string") {
throw new TypeError("key 必须是字符串");
}
if (this.$data.onceExec.has(key)) {
return;
}
callback();
this.$data.onceExec.set(key, 1);
},
/**
* 显示设置面板
*/
showPanel() {
pops.panel({
title: {
text: `${GM_info?.script?.name || "网页调试"}`,
position: "center",
},
content: this.getPanelContentConfig(),
mask: {
enable: true,
clickEvent: {
toClose: true,
toHide: false,
},
},
isMobile: this.isMobile(),
width: this.getWidth(),
height: this.getHeight(),
drag: true,
only: true,
zIndex: 200000000,
style: /*css*/ `
aside.pops-panel-aside{
width: 20%;
}`,
});
},
/**
* 判断是否是移动端
*/
isMobile() {
return window.innerWidth < 550;
},
/**
* 获取设置面板的宽度
*/
getWidth() {
if (window.innerWidth < 550) {
return "92vw";
} else {
return "550px";
}
},
/**
* 获取设置面板的高度
*/
getHeight() {
if (window.innerHeight > 450) {
return "80vh";
} else {
return "450px";
}
},
/**
* 获取按钮配置
* @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 {import("@whitesev/pops/dist/types/src/components/panel/switchType").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.$data.attributeKeyName] = key;
result.attributes[this.$data.attributeDefaultValueName] =
Boolean(defaultValue);
return result;
},
/**
* 获取button按钮配置
* @param {string} text 左边的文字
* @param {string} [description] 左边的文字下面的描述
* @param {string | (() => string)} buttonText 按钮的文字
* @param {PopsIcon} [buttonIcon] 按钮图标
* @param {boolean} [buttonIsRightIcon] 按钮是否在右边
* @param {boolean} [buttonIconIsLoading] 按钮图标是否旋转
* @param {import("@whitesev/pops/dist/types/src/types/button").PopsButtonStyleType} buttonType 按钮类型
* @param {((event: MouseEvent | PointerEvent) => void)} [clickCallBack] 点击回调
* @returns
*/
getButtonDetails(
text,
description = "",
buttonText,
buttonIcon,
buttonIsRightIcon,
buttonIconIsLoading,
buttonType,
clickCallBack
) {
/** @type {import("@whitesev/pops/dist/types/src/components/panel/buttonType").PopsPanelButtonDetails} */
let result = {
text: text,
type: "button",
description: description,
buttonIcon: buttonIcon,
buttonIsRightIcon: buttonIsRightIcon,
buttonIconIsLoading: buttonIconIsLoading,
buttonType: buttonType,
buttonText: buttonText,
callback(event) {
if (typeof clickCallBack === "function") {
clickCallBack(event);
}
},
afterAddToUListCallBack: void 0,
};
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 描述
* @returns {PopsPanelInputDetails}
*/
getInputDetail(
text,
key,
defaultValue,
placeholder = "",
_callback_,
description
) {
/** @type {import("@whitesev/pops/dist/types/src/components/panel/inputType").PopsPanelInputDetails} */
let result = {
text: text,
type: "input",
attributes: {},
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,
};
result.attributes[this.$data.attributeKeyName] = key;
result.attributes[this.$data.attributeDefaultValueName] = 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 描述
* @returns {PopsPanelInputDetails}
*/
getNumberInputDetail(
text,
key,
defaultValue,
placeholder = "",
_callback_,
description
) {
let config = this.getInputDetail(
text,
key,
defaultValue,
(placeholder = ""),
_callback_,
description
);
config.isNumber = true;
config.getValue = function () {
let localValue = PopsPanel.getValue(key, defaultValue);
localValue = parseInt(localValue);
if (isNaN(localValue)) {
return defaultValue;
} else {
return localValue;
}
};
config.callback = function (event, value, valueAsNumber) {
if (typeof _callback_ === "function") {
if (_callback_(event, value)) {
return;
}
}
if (value === "") {
PopsPanel.deleteValue(key);
return;
}
if (isNaN(valueAsNumber)) {
return;
}
PopsPanel.setValue(key, valueAsNumber);
};
return config;
},
/**
* 获取下拉列表配置
* @param {string} text 文字
* @param {string} key 键
* @param {any} defaultValue 默认值
* @param {{
* value: any,
* text: string,
* disable?(value: any): boolean,
* }[]} data 数据
* @param {string} description (可选)描述
* @param {(event:PointerEvent, isSelectedValue: any, isSelectedText:string)=>void} selectCallBack(可选)选择的回调
* @returns {PopsPanelSelectDetails}
*/
getSelectDetail(
text,
key,
defaultValue,
data,
description,
selectCallBack
) {
/** @type {import("@whitesev/pops/dist/types/src/components/panel/selectType").PopsPanelSelectDetails} */
let result = {
text: text,
type: "select",
description: description,
attributes: {},
getValue() {
return PopsPanel.getValue(key, defaultValue);
},
callback(event, isSelectedValue, isSelectedText) {
PopsPanel.setValue(key, isSelectedValue);
if (typeof selectCallBack === "function") {
selectCallBack(event, isSelectedValue, isSelectedText);
}
},
data: data,
};
result.attributes[this.$data.attributeKeyName] = key;
result.attributes[this.$data.attributeDefaultValueName] = defaultValue;
return result;
},
/**
* 获取配置内容
*/
getPanelContentConfig() {
/** @type {import("@whitesev/pops/dist/types/src/components/panel/indexType").PopsPanelContentConfig[]} */
let content = [
{
id: "debug-panel-config-all",
title: "总设置",
headerTitle: "总设置",
forms: [
{
text: "功能",
type: "forms",
forms: [
this.getSelectDetail(
"调试工具",
"currentDebug",
"eruda",
[
{
value: "eruda",
text: "Eruda",
},
{
value: "vconsole",
text: "VConsole",
},
{
value: "pagespy",
text: "PageSpy",
},
{
value: "chii",
text: "Chii",
},
],
void 0,
void 0
),
this.getSwtichDetail(
"允许在iframe内加载",
"allowRunInIframe",
false,
void 0,
"如果指定本脚本的容器并没有在iframe内执行本脚本,那么该功能将不会生效"
),
this.getSwtichDetail(
"主动加载调试工具",
"autoLoadDebugTool",
true,
void 0,
"关闭后将会在脚本菜单注册(不可用)按钮,有3种状态【加载并显示调试工具】、【隐藏调试工具】、【显示调试工具】"
),
],
},
],
},
{
id: "debug-panel-config-eruda",
title: "Eruda",
headerTitle: `<a href='${ToolsConfig.eruda.settingDocUrl}' target='_blank'>Eruda设置</a>`,
forms: [
{
text: "功能",
type: "forms",
forms: [
this.getButtonDetails(
"当前版本",
"",
ToolsConfig.eruda.version,
void 0,
false,
false,
"primary",
(event) => {
utils.preventEvent(event);
window.open(ToolsConfig.eruda.homeUrl, "_blank");
}
),
{
type: "own",
getLiElementCallBack(liElement) {
let $left = document.createElement("div");
$left.className = "pops-panel-item-left-text";
$left.innerHTML = /*html*/ `
<p class="pops-panel-item-left-main-text">最新版本</p>
`;
let $right = document.createElement("div");
$right.className = "pops-panel-item-right-text";
$right.innerHTML = /*html*/ `
<a href="${ToolsConfig.eruda.homeUrl}" target="_blank">
<img src="https://img.shields.io/npm/v/eruda/latest.svg?label=eruda" alt="eruda">
</a>
`;
liElement.appendChild($left);
liElement.appendChild($right);
return liElement;
},
},
this.getSwtichDetail(
"自动打开面板",
"eruda-auto-open-panel",
false,
void 0,
"加载完毕后自动显示面板内容"
),
this.getSelectDetail(
"默认展示的面板元素",
"eruda-default-show-panel-name",
"console",
[
{
text: "Console",
value: "console",
disable() {
return !PopsPanel.getValue("eruda-panel-console");
},
},
{
text: "Elements",
value: "elements",
disable() {
return !PopsPanel.getValue("eruda-panel-elements");
},
},
{
text: "Network",
value: "network",
disable() {
return !PopsPanel.getValue("eruda-panel-network");
},
},
{
text: "Resources",
value: "resources",
disable() {
return !PopsPanel.getValue("eruda-panel-resources");
},
},
{
text: "Sources",
value: "sources",
disable() {
return !PopsPanel.getValue("eruda-panel-sources");
},
},
{
text: "Info",
value: "info",
disable() {
return !PopsPanel.getValue("eruda-panel-info");
},
},
{
text: "Snippets",
value: "snippets",
disable() {
return !PopsPanel.getValue("eruda-panel-snippets");
},
},
{
text: "Monitor",
value: "monitor",
disable() {
return !PopsPanel.getValue(
"eruda_plugin_Resource_erudaMonitor"
);
},
},
{
text: "Features",
value: "features",
disable() {
return !PopsPanel.getValue(
"eruda_plugin_Resource_erudaFeatures"
);
},
},
{
text: "Timing",
value: "timing",
disable() {
return !PopsPanel.getValue(
"eruda_plugin_Resource_erudaTiming"
);
},
},
{
text: "Code",
value: "code",
disable() {
return !PopsPanel.getValue(
"eruda_plugin_Resource_erudaCode"
);
},
},
{
text: "Benchmark",
value: "benchmark",
disable() {
return !PopsPanel.getValue(
"eruda_plugin_Resource_erudaBenchmark"
);
},
},
{
text: "Geolocation",
value: "geolocation",
disable() {
return !PopsPanel.getValue(
"eruda_plugin_Resource_erudaGeolocation"
);
},
},
{
text: "Orientation",
value: "orientation",
disable() {
return !PopsPanel.getValue(
"eruda_plugin_Resource_erudaOrientation"
);
},
},
{
text: "Touches",
value: "touches",
disable() {
return !PopsPanel.getValue(
"eruda_plugin_Resource_erudaTouches"
);
},
},
{
text: "Outline",
value: "outline",
disable() {
return !PopsPanel.getValue(
"eruda_plugin_Resource_erudaOutlinePlugin"
);
},
},
{
text: "Pixel",
value: "ixel",
disable() {
return !PopsPanel.getValue(
"eruda_plugin_Resource_erudaPixel"
);
},
},
{
text: "Settings",
value: "settings",
},
],
"开启【自动打开面板】才会生效",
void 0
),
],
},
{
text: "面板",
type: "forms",
forms: [
this.getSwtichDetail(
"Console",
"eruda-panel-console",
true,
void 0,
"控制台"
),
this.getSwtichDetail(
"Elements",
"eruda-panel-elements",
true,
void 0,
"元素"
),
this.getSwtichDetail(
"Network",
"eruda-panel-network",
true,
void 0,
"网络"
),
this.getSwtichDetail(
"Resources",
"eruda-panel-resources",
true,
void 0,
"资源"
),
this.getSwtichDetail(
"Sources",
"eruda-panel-sources",
true,
void 0,
"源代码"
),
this.getSwtichDetail(
"Info",
"eruda-panel-info",
true,
void 0,
"信息"
),
this.getSwtichDetail(
"Snippets",
"eruda-panel-snippets",
true,
void 0,
"拓展"
),
],
},
{
text: "插件",
type: "forms",
forms: [
this.getSwtichDetail(
"eruda-monitor",
"eruda_plugin_Resource_erudaMonitor",
false,
void 0,
"展示页面的 fps 和内存信息"
),
this.getSwtichDetail(
"eruda-features",
"eruda_plugin_Resource_erudaFeatures",
false,
void 0,
"浏览器特性检测"
),
this.getSwtichDetail(
"eruda-timing",
"eruda_plugin_Resource_erudaTiming",
false,
void 0,
"展示性能资源数据"
),
this.getSwtichDetail(
"eruda-code",
"eruda_plugin_Resource_erudaCode",
false,
void 0,
"运行 JavaScript 代码"
),
this.getSwtichDetail(
"eruda-benchmark",
"eruda_plugin_Resource_erudaBenchmark",
false,
void 0,
"运行 JavaScript 性能测试"
),
this.getSwtichDetail(
"eruda-geolocation",
"eruda_plugin_Resource_erudaGeolocation",
false,
void 0,
"测试地理位置接口"
),
this.getSwtichDetail(
"eruda-orientation",
"eruda_plugin_Resource_erudaOrientation",
false,
void 0,
"测试重力感应接口"
),
this.getSwtichDetail(
"eruda-touches",
"eruda_plugin_Resource_erudaTouches",
false,
void 0,
"(暂时无效)可视化屏幕 Touch 事件触发"
),
this.getSwtichDetail(
"eruda-outline-plugin",
"eruda_plugin_Resource_erudaOutlinePlugin",
false,
void 0,
"给页面的元素添加边框"
),
this.getSwtichDetail(
"eruda-pixel",
"eruda_plugin_Resource_erudaPixel",
false,
void 0,
"高精度的UI恢复辅助工具"
),
],
},
],
},
{
id: "debug-panel-config-vconsole",
title: "vConsole",
headerTitle: `<a href='${ToolsConfig.vConsole.settingDocUrl}' target='_blank'>vConsole设置</a>`,
forms: [
{
text: "功能",
type: "forms",
forms: [
this.getButtonDetails(
"当前版本",
"",
ToolsConfig.vConsole.version,
void 0,
false,
false,
"primary",
(event) => {
utils.preventEvent(event);
window.open(ToolsConfig.vConsole.homeUrl, "_blank");
}
),
{
type: "own",
getLiElementCallBack(liElement) {
let $left = document.createElement("div");
$left.className = "pops-panel-item-left-text";
$left.innerHTML = /*html*/ `
<p class="pops-panel-item-left-main-text">最新版本</p>
`;
let $right = document.createElement("div");
$right.className = "pops-panel-item-right-text";
$right.innerHTML = /*html*/ `
<a href="${ToolsConfig.vConsole.homeUrl}" target="_blank">
<img src="https://img.shields.io/npm/v/vconsole/latest.svg?label=vConsole" alt="vConsole">
</a>
`;
liElement.appendChild($left);
liElement.appendChild($right);
return liElement;
},
},
this.getSwtichDetail(
"自动打开面板",
"vconsole-auto-open-panel",
false,
void 0,
"加载完毕后自动显示面板内容"
),
this.getSelectDetail(
"默认展示的面板元素",
"vconsole-default-show-panel-name",
"default",
[
{
text: "Log",
value: "default",
},
{
text: "System",
value: "system",
disable() {
return !PopsPanel.getValue("vConsole-panel-system");
},
},
{
text: "Network",
value: "network",
disable() {
return !PopsPanel.getValue("vConsole-panel-network");
},
},
{
text: "Element",
value: "element",
disable() {
return !PopsPanel.getValue("vConsole-panel-element");
},
},
{
text: "Storage",
value: "storage",
disable() {
return !PopsPanel.getValue("vConsole-panel-storage");
},
},
{
text: "Stats",
value: "stats",
disable() {
return !PopsPanel.getValue(
"vConsole_plugin_Resource_vConsole_Stats"
);
},
},
{
text: "exportLog",
value: "exportlog",
disable() {
return !PopsPanel.getValue(
"vConsole_plugin_Resource_vConsole_ExportLog"
);
},
},
{
text: "Vue",
value: "vue",
disable() {
return !PopsPanel.getValue(
"vConsole_plugin_Resource_vConsoleVueDevtools"
);
},
},
],
"开启【自动打开面板】才会生效",
void 0
),
],
},
{
text: "面板",
type: "forms",
forms: [
this.getSwtichDetail(
"System",
"vConsole-panel-system",
true,
void 0,
"控制台"
),
this.getSwtichDetail(
"Network",
"vConsole-panel-network",
true,
void 0,
"元素"
),
this.getSwtichDetail(
"Element",
"vConsole-panel-element",
true,
void 0,
"网络"
),
this.getSwtichDetail(
"Storage",
"vConsole-panel-storage",
true,
void 0,
"资源"
),
],
},
{
text: "配置",
type: "forms",
forms: [
this.getSelectDetail(
"主题",
"vConsole-theme",
"light",
[
{
value: "auto",
text: "自动",
},
{
value: "light",
text: "浅色主题",
},
{
value: "dark",
text: "深色主题",
},
],
void 0,
void 0
),
this.getSwtichDetail(
"禁止Log自动滚动",
"vconsole-disableLogScrolling",
false
),
this.getSwtichDetail(
"显示日志的输出时间",
"vconsole-showTimestamps",
false
),
this.getNumberInputDetail(
"日志的上限数量",
"vconsole-maxLogNumber",
1000,
"请输入数字"
),
this.getNumberInputDetail(
"请求记录的上限数量",
"vconsole-maxNetworkNumber",
1000,
"请输入数字"
),
],
},
{
text: "Storage配置",
type: "forms",
forms: [
this.getSwtichDetail(
"Cookies",
"vConsole-storage-defaultStorages-cookies",
true,
void 0,
"显示Cookies"
),
this.getSwtichDetail(
"LocalStorage",
"vConsole-storage-defaultStorages-localStorage",
true,
void 0,
"显示LocalStorage"
),
this.getSwtichDetail(
"SessionStorage",
"vConsole-storage-defaultStorages-sessionStorage",
true,
void 0,
"显示SessionStorage"
),
],
},
{
text: "插件",
type: "forms",
forms: [
this.getSwtichDetail(
"vconsole-stats-plugin",
"vConsole_plugin_Resource_vConsole_Stats",
false,
void 0,
"A vConsole plugin which can show Stats in front-end."
),
this.getSwtichDetail(
"vconsole-outputlog-plugin",
"vConsole_plugin_Resource_vConsole_ExportLog",
false,
void 0,
"使用该插件可以复制或下载console中打印的log"
),
this.getSwtichDetail(
"vconsole-vue-devtools-plugin",
"vConsole_plugin_Resource_vConsoleVueDevtools",
false,
void 0,
"Vue-vConsole-devtools 是一款vConsole插件,把Vue.js官方调试工具vue-devtools移植到移动端,可以直接在移动端查看调试Vue.js应用"
),
],
},
],
},
{
id: "debug-panel-config-pagespy",
title: "PageSpy",
headerTitle: `<a href='${ToolsConfig.pageSpy.settingDocUrl}' target='_blank'>PageSpy设置</a>`,
forms: [
{
text: "功能",
type: "forms",
forms: [
{
text: "注意!隐私保护!",
type: "button",
buttonType: "danger",
buttonText: "了解详情",
callback(event) {
pops.confirm({
title: {
text: "提示",
},
content: {
text: `下面默认配置的${ToolsConfig.pageSpy.defaultConfig.api}是仅供测试使用的,其他人也可以看到你的调试信息,包括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,
},
});
},
},
this.getButtonDetails(
"当前版本",
"",
ToolsConfig.pageSpy.version,
void 0,
false,
false,
"primary",
(event) => {
utils.preventEvent(event);
window.open(ToolsConfig.pageSpy.homeUrl, "_blank");
}
),
{
type: "own",
getLiElementCallBack(liElement) {
let $left = document.createElement("div");
$left.className = "pops-panel-item-left-text";
$left.innerHTML = /*html*/ `
<p class="pops-panel-item-left-main-text">最新版本</p>
`;
let $right = document.createElement("div");
$right.className = "pops-panel-item-right-text";
$right.innerHTML = /*html*/ `
<a href="${ToolsConfig.pageSpy.homeUrl}" target="_blank">
<img src="https://img.shields.io/npm/v/@huolala-tech/page-spy-browser?label=page-spy-browser" alt="page-spy-browser">
</a>
`;
liElement.appendChild($left);
liElement.appendChild($right);
return liElement;
},
},
this.getSwtichDetail(
"禁止在调试端运行",
"pagespy-disable-run-in-debug-client",
true,
void 0,
"调试端是下面配置的api/clientOrigin地址"
),
],
},
{
text: "配置",
type: "forms",
forms: [
this.getInputDetail(
"api",
"pagespy-api",
ToolsConfig.pageSpy.defaultConfig.api,
"",
function (event, value) {
PopsPanel.setValue("pagespy-api", value.trim());
},
"服务器地址的 Host"
),
this.getInputDetail(
"clientOrigin",
"pagespy-clientOrigin",
ToolsConfig.pageSpy.defaultConfig.cliennOrigin,
"",
function (event, value) {
PopsPanel.setValue("pagespy-clientOrigin", value.trim());
},
"服务器地址的 Origin"
),
this.getInputDetail(
"project",
"pagespy-project",
"default",
void 0,
void 0,
"项目名称"
),
this.getInputDetail(
"title",
"pagespy-title",
"--",
void 0,
void 0,
"自定义标题"
),
this.getSwtichDetail(
"autoRender",
"pagespy-autoRender",
true,
void 0,
"自动渲染「圆形白底带 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: `<a href='${ToolsConfig.chii.settingDocUrl}' target='_blank'>Chii设置</a>`,
forms: [
{
text: "功能",
type: "forms",
forms: [
{
text: "调试页面",
type: "button",
buttonType: "primary",
buttonText: "前往",
disable: Boolean(
PopsPanel.getValue("chii-script-embedded", true)
),
callback(event) {
let url = PopsPanel.getValue(
"chii-debug-url",
ToolsConfig.chii.defaultConfig.url
);
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.getSwtichDetail(
"检测script加载",
"chii-check-script-load",
true,
void 0,
"失败会有alert提示弹出"
),
this.getInputDetail(
"调试页面Url",
"chii-debug-url",
ToolsConfig.chii.defaultConfig.url,
"请输入链接Url",
void 0,
"配置【调试页面】的Url"
),
this.getInputDetail(
"来源js",
"chii-target-js",
ToolsConfig.chii.defaultConfig.scriptJs,
"请输入目标js文件",
void 0,
"用于注入页面来进行调试"
),
],
},
{
text: "本页展示的配置",
type: "forms",
forms: [
{
text: "高度",
type: "slider",
description: "移动端不好拖拽,使用这个配置高度",
attributes: {
"data-key": ChiiHeight.$data.key,
"data-default-value": ChiiHeight.$data.winHalfHeight,
},
getValue() {
return ChiiHeight.getLocalHeight();
},
callback(event, value) {
ChiiHeight.setGMLocalHeight(value);
ChiiHeight.setLocalHeight(value);
let chiiContainer = Array.from(
document.querySelectorAll(".__chobitsu-hide__")
).find((ele) => ele.querySelector("iframe"));
if (chiiContainer) {
chiiContainer.style.height = value + "px";
}
},
getToolTipContent(value) {
return value + "px";
},
min: 0,
max: ChiiHeight.$data.winHeight,
step: 1,
},
],
},
],
},
];
return content;
},
};
const ChiiHeight = {
$data: {
key: "chii-embedded-height",
winHeight: parseInt(globalThis.innerHeight),
winHalfHeight: parseInt(globalThis.innerHeight / 2),
},
init() {
let height = this.$data.winHalfHeight;
if (!this.isExistGMLocalHeight()) {
/* GM未创建或不是数字,设置值到油猴数据管理器中 */
this.setGMLocalHeight(height);
} else {
height = this.getGMLocalHeight();
}
this.setLocalHeight(height);
},
isExistLocalHeight() {
return typeof this.getLocalHeight() === "number";
},
/**
*
* @returns {number}
*/
getLocalHeight() {
return globalThis.localStorage.getItem(this.$data.key);
},
/**
*
* @param {number} value
*/
setLocalHeight(value) {
if (typeof value !== "number") {
console.log(value);
throw new TypeError(`${this.$data.key}的值必须是number`);
}
globalThis.localStorage.setItem(this.$data.key, value);
if (this.getLocalHeight() !== value) {
globalThis.localStorage[this.$data.key] = value;
}
},
isExistGMLocalHeight() {
return typeof this.getGMLocalHeight() === "number";
},
/**
*
* @returns {number}
*/
getGMLocalHeight() {
return PopsPanel.getValue(this.$data.key);
},
/**
*
* @param {number} value
*/
setGMLocalHeight(value) {
if (typeof value !== "number") {
console.log(value);
throw new TypeError(`${this.$data.key}的值必须是number`);
}
PopsPanel.setValue(this.$data.key, value);
},
};
const vConsolePlugin = {
State(vConsole, VConsole) {
const Stats = function () {
var mode = 0;
var localPositionStorageKey = "vConsole-Plugin-Stats-Position";
function getLocalPositionStorage() {
return GM_getValue(localPositionStorageKey, {
top: 0,
left: 0,
});
}
function setLocalPositionStorage(left, top) {
GM_setValue(localPositionStorageKey, {
left: left,
top: top,
});
}
var container = document.createElement("div");
let oldPosition = getLocalPositionStorage();
container.style.cssText = `position:fixed;top:${oldPosition.top}px;left:${oldPosition.left}px;cursor:pointer;opacity:0.9;z-index:10000`;
container.addEventListener(
"click",
function (event) {
event.preventDefault();
showPanel(++mode % container.children.length);
},
{
capture: true,
}
);
function addPanel(panel) {
container.appendChild(panel.dom);
return panel;
}
function showPanel(id) {
for (var i = 0; i < container.children.length; i++) {
container.children[i].style.display = i === id ? "block" : "none";
}
mode = id;
}
function drag() {
pops.config.Utils.drag(container, {
dragElement: container,
limit: true,
extraDistance: 2,
moveCallBack(moveElement, left, top) {
setLocalPositionStorage(left, top);
},
});
}
var beginTime = (performance || Date).now(),
prevTime = beginTime,
frames = 0;
var fpsPanel = addPanel(new Stats.Panel("FPS", "#0ff", "#002"));
var msPanel = addPanel(new Stats.Panel("MS", "#0f0", "#020"));
if (self.performance && self.performance.memory) {
var memPanel = addPanel(new Stats.Panel("MB", "#f08", "#201"));
}
showPanel(0);
drag();
return {
REVISION: 16,
dom: container,
addPanel: addPanel,
showPanel: showPanel,
begin: function () {
beginTime = (performance || Date).now();
},
end: function () {
frames++;
var time = (performance || Date).now();
msPanel.update(time - beginTime, 200);
if (time >= prevTime + 1000) {
fpsPanel.update((frames * 1000) / (time - prevTime), 100);
prevTime = time;
frames = 0;
if (memPanel) {
var memory = performance.memory;
memPanel.update(
memory.usedJSHeapSize / 1048576,
memory.jsHeapSizeLimit / 1048576
);
}
}
return time;
},
update: function () {
beginTime = this.end();
},
// Backwards Compatibility
domElement: container,
setMode: showPanel,
};
};
Stats.Panel = function (name, fg, bg) {
var min = Infinity,
max = 0,
round = Math.round;
var PR = round(window.devicePixelRatio || 1);
var WIDTH = 80 * PR,
HEIGHT = 48 * PR,
TEXT_X = 3 * PR,
TEXT_Y = 2 * PR,
GRAPH_X = 3 * PR,
GRAPH_Y = 15 * PR,
GRAPH_WIDTH = 74 * PR,
GRAPH_HEIGHT = 30 * PR;
var canvas = document.createElement("canvas");
canvas.width = WIDTH;
canvas.height = HEIGHT;
canvas.style.cssText = "width:80px;height:48px";
var context = canvas.getContext("2d");
context.font = "bold " + 9 * PR + "px Helvetica,Arial,sans-serif";
context.textBaseline = "top";
context.fillStyle = bg;
context.fillRect(0, 0, WIDTH, HEIGHT);
context.fillStyle = fg;
context.fillText(name, TEXT_X, TEXT_Y);
context.fillRect(GRAPH_X, GRAPH_Y, GRAPH_WIDTH, GRAPH_HEIGHT);
context.fillStyle = bg;
context.globalAlpha = 0.9;
context.fillRect(GRAPH_X, GRAPH_Y, GRAPH_WIDTH, GRAPH_HEIGHT);
return {
dom: canvas,
update: function (value, maxValue) {
min = Math.min(min, value);
max = Math.max(max, value);
context.fillStyle = bg;
context.globalAlpha = 1;
context.fillRect(0, 0, WIDTH, GRAPH_Y);
context.fillStyle = fg;
context.fillText(
round(value) +
" " +
name +
" (" +
round(min) +
"-" +
round(max) +
")",
TEXT_X,
TEXT_Y
);
context.drawImage(
canvas,
GRAPH_X + PR,
GRAPH_Y,
GRAPH_WIDTH - PR,
GRAPH_HEIGHT,
GRAPH_X,
GRAPH_Y,
GRAPH_WIDTH - PR,
GRAPH_HEIGHT
);
context.fillRect(
GRAPH_X + GRAPH_WIDTH - PR,
GRAPH_Y,
PR,
GRAPH_HEIGHT
);
context.fillStyle = bg;
context.globalAlpha = 0.9;
context.fillRect(
GRAPH_X + GRAPH_WIDTH - PR,
GRAPH_Y,
PR,
round((1 - value / maxValue) * GRAPH_HEIGHT)
);
},
};
};
class VConsoleStatsPlugin {
constructor(vConsole, VConsole) {
this.vConsole = vConsole;
this.VConsole = VConsole;
this.dom = null;
this.requestID = null;
this.stats = null;
return this.init();
}
init() {
this.addStyle();
const vConsoleStats = new this.VConsole.VConsolePlugin(
"Stats",
"Stats"
);
vConsoleStats.on("ready", () => {
document
.querySelectorAll(".vc-stats-buttons")
.forEach((statusButton) => {
statusButton.addEventListener("click", (event) => {
const currentType = event.target.dataset.type;
if (
currentType.toString() === "2" &&
!(self.performance && self.performance.memory)
) {
console.error(
"浏览器不支持window.performance或者window.performance.memory"
);
return;
}
this.changePanel(currentType);
});
});
});
vConsoleStats.on("renderTab", (callback) => {
const statsHTML = /*html*/ `
<div class="vc-stats-buttons">
<div class="vc-button-container">
<button class="vc-stats-button" data-type="0">show FPS</button>
<div class="vc-description">
<span>最后一秒渲染的帧。数字越高越好</span>
</div>
</div>
<div class="vc-button-container">
<button class="vc-stats-button" data-type="1">show MS</button>
<div class="vc-description">
<span>渲染帧所需的毫秒数。数字越低越好</span>
</div>
</div>
<div class="vc-button-container">
<button class="vc-stats-button" data-type="2">show MB</button>
<div class="vc-description">
<span>内存分配(MB)</span>
<a class="vc-link" href="https://caniuse.com/mdn-api_performance_memory" target="_blank">performance.memory兼容性查看</a>
<span>Chrome启用方式: --enable-precise-memory-info</span>
</div>
</div>
</div>`;
callback(statsHTML);
});
vConsoleStats.on("addTool", (callback) => {
const buttons = [
{
name: "Show Stats",
onClick: this.show,
},
{
name: "Close Stats",
onClick: this.close,
},
];
callback(buttons);
});
this.vConsole.addPlugin(vConsoleStats);
return vConsoleStats;
}
addStyle = (target) => {
if (target == null) {
target = document.head || document.body || document.documentElement;
}
const cssNode = document.createElement("style");
cssNode.setAttribute("type", "text/css");
cssNode.innerHTML = /*css*/ `
.vc-stats-button{
margin: 10px 10px;
background-color: #fbf9fe;
padding: 2px 4px;
cursor: pointer;
border-radius: 4px;
border: 1px solid;
}
.vc-button-container{
display: flex;
align-items: center;
}
.vc-description{
display: flex;
flex-direction: column;
}
.vc-description a.vc-link{
color: blue;
}`;
target.appendChild(cssNode);
};
show = () => {
if (!this.stats) {
this.stats = new Stats();
this.stats.showPanel(1); // 0: fps, 1: ms, 2: mb, 3+: custom
this.dom = this.stats.dom;
document.body.appendChild(this.dom);
this.requestID = requestAnimationFrame(this.loop);
}
};
changePanel = (type) => {
if (!this.stats) {
this.show();
}
this.stats.setMode(Number(type));
};
loop = () => {
this.stats.update();
this.requestID = requestAnimationFrame(this.loop);
};
close = () => {
if (this.requestID) {
cancelAnimationFrame(this.requestID);
}
if (this.dom) {
document.body.removeChild(this.dom);
}
this.stats = null;
this.requestID = null;
this.dom = null;
};
}
return new VConsoleStatsPlugin(vConsole, VConsole);
},
exportLog(vConsole, VConsole) {
class VConsoleOutputLogsPlugin {
constructor(vConsole, VConsole, logItemSelector) {
this.vConsole = vConsole;
this.VConsole = VConsole;
this.$ = vConsole.$;
this.dom = null;
this.logItemSelector =
logItemSelector || ".vc-content #__vc_plug_default .vc-log-row";
return this.init();
}
init() {
const vConsoleExportLogs = new this.VConsole.VConsolePlugin(
"exportLog",
"exportLog"
);
vConsoleExportLogs.on("ready", () => {
console.log("[vConsole-exportlog-plugin] -- load");
});
vConsoleExportLogs.on("renderTab", (callback) => {
const html = /*html*/ `<div class="vconsole-exportlog"></div>`;
callback(html);
});
vConsoleExportLogs.on("addTool", (callback) => {
const buttons = [
{
name: "exportLogs",
onClick: this.export,
},
{
name: "copyLogs",
onClick: this.copyText,
},
];
callback(buttons);
});
this.vConsole.addPlugin(vConsoleExportLogs);
return vConsoleExportLogs;
}
funDownload = (content, filename) => {
var eleLink = document.createElement("a");
eleLink.download = filename;
eleLink.style.display = "none";
var blob = new Blob([content]);
eleLink.href = URL.createObjectURL(blob);
document.body.appendChild(eleLink);
eleLink.click();
document.body.removeChild(eleLink);
};
getAllLogContent = () => {
let logRowsElement = document.querySelectorAll(this.logItemSelector);
let logText = "";
for (let index = 0; index < logRowsElement.length; index++) {
const ele = logRowsElement[index];
logText += `${ele.textContent}\n`;
}
return logText;
};
export = () => {
let logText = this.getAllLogContent();
this.funDownload(
logText,
`${
new Date().toLocaleDateString() +
" " +
new Date().toLocaleTimeString()
}.log`
);
};
copyText = () => {
let logText = this.getAllLogContent();
utils.setClip(logText);
};
}
return new VConsoleOutputLogsPlugin(vConsole, VConsole);
},
};
/** 全局调试工具的配置 */
const ToolsConfig = {
eruda: {
/** 版本号 */
version: "3.2.1",
/** 项目地址 */
homeUrl: "https://github.com/liriliri/eruda",
/** 项目最新的js文件地址 */
latestFileUrl: "https://cdn.jsdelivr.net/npm/eruda",
/** 设置文档 */
settingDocUrl: "https://github.com/liriliri/eruda/blob/master/README.md",
},
vConsole: {
/** 版本号 */
version: "3.15.1",
/** 项目地址 */
homeUrl: "https://github.com/Tencent/vConsole",
/** 项目最新的js文件地址 */
latestFileUrl: "https://cdn.jsdelivr.net/npm/vconsole",
/** 设置文档 */
settingDocUrl:
"https://github.com/Tencent/vConsole/blob/dev/README_CN.md",
},
pageSpy: {
/** 版本号 */
version: "1.9.2",
/** 项目地址 */
homeUrl: "https://github.com/HuolalaTech/page-spy-web",
/** 项目最新的js文件地址 */
latestFileUrl:
"https://github.com/HuolalaTech/page-spy/tree/main/packages/page-spy-browser",
/** 设置文档 */
settingDocUrl:
"https://github.com/HuolalaTech/page-spy-web/blob/main/README_ZH.md",
/** 默认配置 */
defaultConfig: {
api: "test.jikejishu.com",
cliennOrigin: "https://test.jikejishu.com",
},
},
chii: {
/** 项目地址 */
homeUrl: "https://github.com/liriliri/chii",
/** 设置文档 */
settingDocUrl:
"https://github.com/liriliri/chii/blob/master/README_CN.md",
/** 默认配置 */
defaultConfig: {
url: "https://chii.liriliri.io/",
scriptJs: "//chii.liriliri.io/target.js",
},
},
};
/**
* 全局调试
*/
const Tools = {
$data: {
/** 当前的调试工具是否已执行 */
isLoadDebugTool: false,
/** 当前已执行的调试工具名 */
loadDebugToolName: void 0,
/** 当前执行了调试工具的iframe @type {string[]} */
iframeUrlList: [],
},
$ele: {
/** 隐藏调试工具的style元素 @type {HTMLStyleElement|undefined} */
hideDebugToolCSSNode: void 0,
},
/**
* 处理当在iframe内加载时,是否允许执行,如果允许,那么把url添加到菜单中
*/
handleIframe() {
if (PopsPanel.isTopWindow()) {
return true;
}
if (!PopsPanel.getValue("allowRunInIframe")) {
return false;
}
this.$data.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.$data.isLoadDebugTool = true;
this.$data.loadDebugToolName = "vconsole";
this.vConsole();
} else if (currentDebugTool === "pagespy") {
/* PageSpy */
this.$data.isLoadDebugTool = true;
this.$data.loadDebugToolName = "pagespy";
this.pageSpy();
} else if (currentDebugTool === "eruda") {
/* eruda */
this.$data.isLoadDebugTool = true;
this.$data.loadDebugToolName = "eruda";
this.eruda();
} else if (currentDebugTool === "chii") {
/* chii */
this.$data.isLoadDebugTool = true;
this.$data.loadDebugToolName = "chii";
this.chii();
} else {
console.error("当前未配置该调试工具的运行");
}
},
/**
* 在脚本菜单中添加控制当前的调试工具状态的菜单按钮
*/
addControlDebugToolScriptMenu() {
if (!PopsPanel.isTopWindow()) {
console.warn("不在iframe内重复添加菜单按钮");
return;
}
let menuData = {
key: "debug_tool_show_hide_control",
text: "☯ 加载并显示调试工具",
autoReload: false,
isStoreValue: false,
showText(text) {
return text;
},
callback: (data) => {
changeMenu(data);
},
};
/**
*
* @param {UtilsGMMenuClickCallBackData} data
*/
const changeMenu = (data) => {
if (Tools.$data.isLoadDebugTool) {
/* 状态:已加载 */
if (Tools.$ele.hideDebugToolCSSNode) {
/* 状态:已加载且添加了隐藏CSS */
/* 进行移除隐藏CSS */
/* 菜单状态:【隐藏调试工具】 */
this.showDebugTool();
menuData.text = "🌑 隐藏调试工具";
GM_Menu.update(menuData);
} else {
/* 状态:已加载且未添加隐藏CSS */
/* 进行添加隐藏CSS */
/* 菜单状态:【显示调试工具】 */
this.hideDebugTool();
menuData.text = "🌕 显示调试工具";
GM_Menu.update(menuData);
}
} else {
/* 状态:未加载,加载并显示 */
/* 进行执行调试工具 */
/* 菜单状态:【隐藏调试工具】 */
this.showDebugTool();
menuData.text = "🌑 隐藏调试工具";
GM_Menu.update(menuData);
}
};
GM_Menu.add(menuData);
},
/**
* 判断页面中是否已存在隐藏调试工具的CSS元素节点
* @returns
*/
hasHideDebugToolCSSNode() {
return document.documentElement.contains(this.$ele.hideDebugToolCSSNode);
},
/**
* 创建隐藏调试工具的CSS元素
* @returns
*/
createHideDebugToolCSSNode() {
let cssNode = document.createElement("style");
cssNode.setAttribute("type", "text/css");
cssNode.setAttribute("data-from", "hide-debug-tool");
cssNode.innerHTML = /*css*/ `
#eruda{
display: none !important;
}
#__vconsole{
display: none !important;
}
#__pageSpy{
display: none !important;
}
.__chobitsu-hide__ > iframe,
.__chobitsu-hide__:has(iframe){
display: none !important;
}
`;
return cssNode;
},
/**
* 隐藏当前的调试工具
*/
hideDebugTool() {
if (this.$ele.hideDebugToolCSSNode == null) {
console.log("未创建隐藏【调试工具】的style元素 => 创建元素");
this.$ele.hideDebugToolCSSNode = this.createHideDebugToolCSSNode();
}
if (!this.hasHideDebugToolCSSNode()) {
console.log("页面不存在隐藏【调试工具】的style元素 => 添加元素");
document.documentElement.appendChild(this.$ele.hideDebugToolCSSNode);
}
},
/**
* 显示当前的调试工具
*/
showDebugTool() {
if (this.$ele.hideDebugToolCSSNode) {
console.log("页面存在隐藏【调试工具】的style元素 => 移除元素");
document.documentElement.removeChild(this.$ele.hideDebugToolCSSNode);
this.$ele.hideDebugToolCSSNode = null;
}
if (!this.$data.isLoadDebugTool) {
console.log("尚未运行【调试工具】 => 运行调试工具");
this.runDebugTool();
}
},
eruda() {
initEruda("Eruda", unsafeWin);
let Eruda = unsafeWin.Eruda || globalThis.Eruda;
if (!Eruda) {
alert("调试工具【eruda】注册(不可用)全局失败,请反馈开发者");
return;
}
let inintPanelList = [];
if (PopsPanel.getValue("eruda-panel-console")) {
inintPanelList.push("console");
}
if (PopsPanel.getValue("eruda-panel-elements")) {
inintPanelList.push("elements");
}
if (PopsPanel.getValue("eruda-panel-network")) {
inintPanelList.push("network");
}
if (PopsPanel.getValue("eruda-panel-resources")) {
inintPanelList.push("resources");
}
if (PopsPanel.getValue("eruda-panel-sources")) {
inintPanelList.push("sources");
}
if (PopsPanel.getValue("eruda-panel-info")) {
inintPanelList.push("info");
}
if (PopsPanel.getValue("eruda-panel-snippets")) {
inintPanelList.push("snippets");
}
ToolsConfig.eruda.version = Eruda.version;
Eruda.init({
tool: inintPanelList,
});
console.log(`eruda当前版本:${Eruda.version}`);
console.log(`eruda项目地址:${ToolsConfig.eruda.homeUrl}`);
console.log("eruda的全局变量名: Eruda");
if (PopsPanel.getValue("eruda_plugin_Resource_erudaMonitor")) {
try {
unsafeWin.eval(GM_getResourceText("Resource_erudaMonitor"));
Eruda.add(erudaMonitor);
} catch (error) {
console.error("插件【eruda-monitor】加载失败,原因:", error);
}
}
if (PopsPanel.getValue("eruda_plugin_Resource_erudaFeatures")) {
try {
unsafeWin.eval(GM_getResourceText("Resource_erudaFeatures"));
Eruda.add(erudaFeatures);
} catch (error) {
console.error("插件【eruda-features】加载失败,原因:", error);
}
}
if (PopsPanel.getValue("eruda_plugin_Resource_erudaTiming")) {
try {
unsafeWin.eval(GM_getResourceText("Resource_erudaTiming"));
Eruda.add(erudaTiming);
} catch (error) {
console.error("插件【eruda-timing】加载失败,原因:", error);
}
}
if (PopsPanel.getValue("eruda_plugin_Resource_erudaCode")) {
try {
unsafeWin.eval(GM_getResourceText("Resource_erudaCode"));
Eruda.add(erudaCode);
} catch (error) {
console.error("插件【eruda-code】加载失败,原因:", error);
}
}
if (PopsPanel.getValue("eruda_plugin_Resource_erudaBenchmark")) {
try {
unsafeWin.eval(GM_getResourceText("Resource_erudaBenchmark"));
Eruda.add(erudaBenchmark);
} catch (error) {
console.error("插件【eruda-benchmark】加载失败,原因:", error);
}
}
if (PopsPanel.getValue("eruda_plugin_Resource_erudaGeolocation")) {
try {
unsafeWin.eval(GM_getResourceText("Resource_Leaflet"));
unsafeWin.eval(GM_getResourceText("Resource_erudaGeolocation"));
Eruda.add(erudaGeolocation);
} catch (error) {
console.error("插件【eruda-geolocation】加载失败,原因:", error);
}
}
if (PopsPanel.getValue("eruda_plugin_Resource_erudaOrientation")) {
try {
unsafeWin.eval(GM_getResourceText("Resource_erudaOrientation"));
Eruda.add(erudaOrientation);
} catch (error) {
console.error("插件【eruda-orientation】加载失败,原因:", error);
}
}
if (PopsPanel.getValue("eruda_plugin_Resource_erudaTouches")) {
try {
unsafeWin.eval(GM_getResourceText("Resource_erudaTouches"));
Eruda.add(erudaTouches);
} catch (error) {
console.error("插件【eruda-touches】加载失败,原因:", error);
}
}
if (PopsPanel.getValue("eruda_plugin_Resource_erudaOutlinePlugin")) {
try {
unsafeWin.eval(GM_getResourceText("Resource_erudaOutlinePlugin"));
Eruda.add(erudaOutlinePlugin);
} catch (error) {
console.error("插件【eruda-outline-plugin】加载失败,原因:", error);
}
}
if (PopsPanel.getValue("eruda_plugin_Resource_erudaPixel")) {
try {
unsafeWin.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", unsafeWin);
let VConsole = unsafeWin.VConsole || globalThis.VConsole;
if (!VConsole) {
alert("调试工具【vConsole】注册(不可用)全局失败,请反馈开发者");
return;
}
let initPanelList = [];
if (PopsPanel.getValue("vConsole-panel-system")) {
initPanelList.push("system");
}
if (PopsPanel.getValue("vConsole-panel-network")) {
initPanelList.push("network");
}
if (PopsPanel.getValue("vConsole-panel-element")) {
initPanelList.push("element");
}
if (PopsPanel.getValue("vConsole-panel-storage")) {
initPanelList.push("storage");
}
let theme = "light";
if (PopsPanel.getValue("vConsole-theme") === "auto") {
if (utils.isThemeDark()) {
theme = "dark";
}
} else {
theme = PopsPanel.getValue("vConsole-theme");
}
let defaultStorages = [];
if (PopsPanel.getValue("vConsole-storage-defaultStorages-cookies")) {
defaultStorages.push("cookies");
}
if (PopsPanel.getValue("vConsole-storage-defaultStorages-localStorage")) {
defaultStorages.push("localStorage");
}
if (
PopsPanel.getValue("vConsole-storage-defaultStorages-sessionStorage")
) {
defaultStorages.push("sessionStorage");
}
let vConsole = new VConsole({
defaultPlugins: initPanelList,
theme: "light",
onReady() {
if (PopsPanel.getValue("vconsole-auto-open-panel")) {
vConsole.show();
}
},
disableLogScrolling: PopsPanel.getValue("vconsole-disableLogScrolling"),
log: {
maxLogNumber: PopsPanel.getValue("vconsole-maxLogNumber", 1000),
showTimestamps: PopsPanel.getValue("vconsole-showTimestamps"),
maxNetworkNumber: PopsPanel.getValue(
"vconsole-maxNetworkNumber",
1000
),
},
storage: {
defaultStorages: defaultStorages,
},
});
ToolsConfig.vConsole.version = vConsole.version;
unsafeWindow.vConsole = vConsole;
console.log(`VConsole当前版本:${vConsole.version}`);
console.log(`VConsole项目地址:${ToolsConfig.vConsole.homeUrl}`);
console.log("VConsole的实例化的全局变量名: vConsole");
if (PopsPanel.getValue("vConsole_plugin_Resource_vConsole_Stats")) {
try {
vConsolePlugin.State(vConsole, VConsole);
} catch (error) {
console.error("插件【vconsole-stats-plugin】加载失败,原因:", error);
}
}
if (PopsPanel.getValue("vConsole_plugin_Resource_vConsole_ExportLog")) {
try {
vConsolePlugin.exportLog(vConsole, VConsole);
} catch (error) {
console.error(
"插件【vconsole-outputlog-plugin】加载失败,原因:",
error
);
}
}
if (PopsPanel.getValue("vConsole_plugin_Resource_vConsoleVueDevtools")) {
try {
unsafeWin.eval(GM_getResourceText("Resource_vConsoleVueDevtools"));
const Devtools = unsafeWin.vueVconsoleDevtools;
Devtools.initPlugin(vConsole);
} catch (error) {
console.error(
"插件【vconsole-vue-devtools-plugin】加载失败,原因:",
error
);
}
}
if (PopsPanel.getValue("vconsole-auto-open-panel")) {
let defaultShowName = PopsPanel.getValue(
"vconsole-default-show-panel-name",
"default"
);
vConsole.show();
setTimeout(() => {
vConsole.showPlugin(defaultShowName);
}, 250);
}
},
pageSpy() {
let api = PopsPanel.getValue(
"pagespy-api",
ToolsConfig.pageSpy.defaultConfig.api
);
let clientOrigin = PopsPanel.getValue(
"pagespy-clientOrigin",
ToolsConfig.pageSpy.defaultConfig.cliennOrigin
);
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(unsafeWin);
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);
ToolsConfig.pageSpy.version = 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",
ToolsConfig.chii.defaultConfig.url
);
if (
window.location.href.startsWith(debugUrl) &&
PopsPanel.getValue("chii-disable-run-in-debug-url", true)
) {
console.log("禁止在调试端运行");
return;
}
ChiiHeight.init();
if (PopsPanel.getValue("chii-check-script-load")) {
function checkChiiScriptLoad(event) {
if (event.target === scriptNode) {
globalThis.alert(
`调试工具【Chii】脚本加载失败
可能原因1:CSP策略阻止了加载第三方域的js文件
可能原因2:目标js无效`
);
unsafeWindow.removeEventListener("error", checkChiiScriptLoad, {
capture: true,
});
}
}
unsafeWindow.addEventListener("error", checkChiiScriptLoad, {
capture: true,
});
}
let scriptJsUrl = PopsPanel.getValue(
"chii-target-js",
ToolsConfig.chii.defaultConfig.scriptJs
);
let scriptEmbedded = PopsPanel.getValue("chii-script-embedded", true);
let scriptNode = document.createElement("script");
scriptNode.src = scriptJsUrl;
scriptNode.setAttribute("type", "application/javascript");
if (scriptEmbedded) {
scriptNode.setAttribute("embedded", true);
}
(document.head || document.body || document.documentElement).appendChild(
scriptNode
);
},
};
PopsPanel.init();
if (Tools.handleIframe()) {
if (PopsPanel.getValue("autoLoadDebugTool")) {
Tools.runDebugTool();
} else {
Tools.addControlDebugToolScriptMenu();
}
}
})();