feature: 实现篡改猴(Tampermonkey,别名:油猴)浏览器插件的几个 `GM_xxx` 方法,并参照篡改猴浏览器扩展界面开发了可操作界面 (#300)
parent
d9453b8437
commit
162d9d92f3
|
@ -1,12 +0,0 @@
|
|||
(function(){
|
||||
const config = {};
|
||||
window.__ds_global__={
|
||||
GM_registerMenuCommand: () => {},
|
||||
GM_unregisterMenuCommand: () => {},
|
||||
GM_openInTab: () => {},
|
||||
GM_getValue: (key) => config[key],
|
||||
GM_setValue: (key, value) => { config[key] = value; },
|
||||
GM_notification: () => {}
|
||||
}
|
||||
}();
|
||||
console.log("ds_global completed")
|
|
@ -0,0 +1,335 @@
|
|||
/**
|
||||
* @name 篡改猴(Tampermonkey)| 油猴(Greasemonkey)浏览器脚本扩展
|
||||
* @description 篡改猴 (Tampermonkey) 是拥有 超过 1000 万用户 的最流行的浏览器扩展之一。 它适用于 Chrome、Microsoft Edge、Safari、Opera Next 和 Firefox。
|
||||
* 有些人也会把篡改猴(Tampermonkey)称作油猴(Greasemonkey),尽管后者只是一款仅适用于 Firefox 浏览器的浏览器扩展程序。
|
||||
* 它允许用户自定义并增强您最喜爱的网页的功能。用户脚本是小型 JavaScript 程序,可用于向网页添加新功能或修改现有功能。使用 篡改猴,您可以轻松在任何网站上创建、管理和运行这些用户脚本。
|
||||
* 例如,使用 篡改猴,您可以向网页添加一个新按钮,可以快速在社交媒体上分享链接,或自动填写带有个人信息的表格。在数字化时代,这特别有用,因为网页常常被用作访问广泛的服务和应用程序的用户界面。
|
||||
* 此外,篡改猴 使您轻松找到并安装其他用户创建的用户脚本。这意味着您可以快速轻松地访问为您喜爱的网页定制的广泛库,而无需花费数小时编写自己的代码。
|
||||
* 无论您是希望为您的站点添加新功能的 Web 开发人员,还是只是希望 改善在线体验的普通用户,篡改猴 都是您的工具箱中的一个很好的工具。
|
||||
* @homepageUrl https://www.tampermonkey.net
|
||||
*/
|
||||
'use strict';
|
||||
(function () {
|
||||
const PRE = "DS-Tampermonkey:"; // 前缀
|
||||
|
||||
const context = {
|
||||
initialized: false, // 是否已经初始化
|
||||
defaultPluginOptions: {}, // 默认插件选项
|
||||
pluginElement: null, // 插件div
|
||||
menusElement: null, // 菜单列表div
|
||||
menus: {}, // 菜单集合
|
||||
menuIndex: 0 // 菜单索引,用于生成menuCmdId
|
||||
};
|
||||
|
||||
// 创建插件样式
|
||||
function createPluginStyle (options) {
|
||||
options = options || {};
|
||||
|
||||
// 创建一个新的<style>元素
|
||||
const styleElement = document.createElement('style');
|
||||
// 设置<style>元素的type属性
|
||||
styleElement.type = 'text/css';
|
||||
|
||||
// 设置<style>元素的内容
|
||||
let cssContent = `
|
||||
.____ds-icon____{
|
||||
position: fixed;
|
||||
right: 5px;
|
||||
top: 35%;
|
||||
z-index: 9999;
|
||||
width: 36px;
|
||||
height: 36px;
|
||||
border-radius: 8px;
|
||||
background-color: #DDD;
|
||||
background-repeat: no-repeat;
|
||||
background-size: cover;
|
||||
background-image: url("");
|
||||
}
|
||||
.____ds-menus____ {
|
||||
display: none;
|
||||
position: absolute;
|
||||
right: 36px;
|
||||
top: 0;
|
||||
z-index: 10000;
|
||||
min-width: 200px;
|
||||
min-height: 35px;
|
||||
border-radius: 8px;
|
||||
background-color: #323231;
|
||||
border: 1px solid #52525E;
|
||||
overflow: hidden;
|
||||
}
|
||||
.____ds-icon____:hover .____ds-menus____ {
|
||||
display: block;
|
||||
}
|
||||
.____ds-menu____ {
|
||||
height: 35px;
|
||||
line-height: 35px;
|
||||
padding: 0 10px;
|
||||
white-space: nowrap;
|
||||
color: #FFF;
|
||||
cursor: pointer;
|
||||
margin-left: 26px;
|
||||
}
|
||||
.____ds-menu____:hover {
|
||||
background-color: #855F16;
|
||||
}
|
||||
.____ds-menu0____ {
|
||||
margin-left: 0;
|
||||
font-size: 16px;
|
||||
font-weight: bold;
|
||||
}
|
||||
.____ds-menu0____ img{
|
||||
width: 23px;
|
||||
height: 23px;
|
||||
vertical-align: middle;
|
||||
margin: 0 8px 3px 8px;
|
||||
}
|
||||
`;
|
||||
// 如果有自定义样式,则添加到 CSS 内容中
|
||||
if (options.style) {
|
||||
cssContent += options.style;
|
||||
}
|
||||
|
||||
// 添加 CSS 内容到<style>元素中
|
||||
if (styleElement.styleSheet) {
|
||||
// 兼容 IE
|
||||
styleElement.styleSheet.cssText = cssContent;
|
||||
} else {
|
||||
styleElement.appendChild(document.createTextNode(cssContent));
|
||||
}
|
||||
|
||||
// 将<style>元素添加到<head>中
|
||||
document.head.append(styleElement);
|
||||
}
|
||||
|
||||
// 创建插件div
|
||||
function createPluginDiv (options) {
|
||||
options = {
|
||||
...{ name: "油猴脚本" },
|
||||
...options
|
||||
}
|
||||
|
||||
// 创建插件div
|
||||
context.pluginElement = document.createElement('div');
|
||||
context.pluginElement.title = "油猴脚本" + (options.name ? ":" + options.name : "");
|
||||
context.pluginElement.className = "____ds-icon____";
|
||||
|
||||
// 创建菜单列表div
|
||||
context.menusElement = document.createElement('div');
|
||||
context.menusElement.className = "____ds-menus____";
|
||||
// 将菜单列表div添加到插件div中
|
||||
context.pluginElement.append(context.menusElement);
|
||||
|
||||
// 创建开关菜单
|
||||
const enabled = window.__ds_global__.GM_getValue("ds_enabled", true)
|
||||
const switchMenuElement = document.createElement('div');
|
||||
const icon = (options.icon ? `<img alt="icon" src="${options.icon}"/>` : " ");
|
||||
switchMenuElement.id = PRE + "menu-0";
|
||||
switchMenuElement.className = "____ds-menu____ ____ds-menu0____";
|
||||
switchMenuElement.innerHTML = (enabled ? "✅" : "❌") + icon + options.name;
|
||||
switchMenuElement.title = `点击${enabled ? "关闭" : "开启"}此脚本功能`;
|
||||
switchMenuElement.onclick = function () {
|
||||
let enabled = window.__ds_global__.GM_getValue("ds_enabled", true)
|
||||
if (enabled) {
|
||||
hideMenus();
|
||||
enabled = false;
|
||||
} else {
|
||||
showMenus();
|
||||
enabled = true;
|
||||
}
|
||||
switchMenuElement.innerHTML = (enabled ? "✅" : "❌") + icon + options.name;
|
||||
switchMenuElement.title = `点击${enabled ? "关闭" : "开启"}此脚本功能`;
|
||||
window.__ds_global__.GM_setValue("ds_enabled", enabled)
|
||||
window.__ds_global__.GM_notification({
|
||||
text: `已${enabled ? "开启" : "关闭"} 「${options.name}」 功能\n(刷新网页后生效)`,
|
||||
timeout: 3500
|
||||
});
|
||||
};
|
||||
// 将开关菜单添加到菜单列表div中
|
||||
context.menusElement.append(switchMenuElement);
|
||||
|
||||
// 获取body元素
|
||||
const body = document.getElementsByTagName('body')[0];
|
||||
// 将插件div添加到body中
|
||||
body.prepend(context.pluginElement);
|
||||
}
|
||||
|
||||
function showMenus () {
|
||||
for (const menuCmdId in context.menus) {
|
||||
const menuElement = context.menus[menuCmdId].element;
|
||||
menuElement.style.display = "block";
|
||||
}
|
||||
}
|
||||
|
||||
function hideMenus () {
|
||||
for (const menuCmdId in context.menus) {
|
||||
const menuElement = context.menus[menuCmdId].element;
|
||||
menuElement.style.display = "none";
|
||||
}
|
||||
}
|
||||
|
||||
window.__ds_global__ = {
|
||||
// 获取上下文
|
||||
getContext: () => {
|
||||
return context;
|
||||
},
|
||||
// 初始化
|
||||
DS_init: (options) => {
|
||||
try {
|
||||
if (context.initialized) return;
|
||||
// 合并默认参数
|
||||
options = {
|
||||
...context.defaultPluginOptions,
|
||||
...options
|
||||
};
|
||||
createPluginStyle(options);
|
||||
createPluginDiv(options);
|
||||
context.initialized = true;
|
||||
|
||||
console.log("ds_tampermonkey: initialization completed")
|
||||
} catch (e) {
|
||||
console.error("ds_tampermonkey: initialization failed:", e);
|
||||
}
|
||||
},
|
||||
// 注册菜单
|
||||
GM_registerMenuCommand: (name, callback, options_or_accessKey) => {
|
||||
const options = typeof options_or_accessKey === "string" ? { accessKey: options_or_accessKey } : options_or_accessKey;
|
||||
|
||||
// 生成菜单ID
|
||||
const menuCmdId = PRE + "menu-" + (++context.menuIndex);
|
||||
|
||||
// 创建菜单元素
|
||||
const menuElement = document.createElement('div');
|
||||
menuElement.id = menuCmdId;
|
||||
menuElement.className = "____ds-menu____";
|
||||
menuElement.innerHTML = name;
|
||||
if (callback) {
|
||||
menuElement.onclick = callback;
|
||||
}
|
||||
|
||||
// 将菜单元素添加到菜单列表div中
|
||||
context.menusElement.append(menuElement);
|
||||
|
||||
// 将菜单添加到菜单集合中
|
||||
context.menus[menuCmdId] = {
|
||||
name: name,
|
||||
callback: callback,
|
||||
options: options,
|
||||
element: menuElement
|
||||
};
|
||||
|
||||
// 返回菜单ID
|
||||
return menuCmdId;
|
||||
},
|
||||
// 删除菜单
|
||||
GM_unregisterMenuCommand: (menuCmdId) => {
|
||||
const menuElement = document.getElementById(menuCmdId)
|
||||
if (menuElement) {
|
||||
menuElement.remove();
|
||||
}
|
||||
delete context.menus[menuCmdId];
|
||||
},
|
||||
// 打开新标签
|
||||
GM_openInTab: (url, options_or_loadInBackground) => {
|
||||
// const options = typeof options_or_loadInBackground === "boolean"
|
||||
// ? { loadInBackground: options_or_loadInBackground }
|
||||
// : (options_or_loadInBackground || {});
|
||||
|
||||
window.open(url)
|
||||
},
|
||||
// 获取配置
|
||||
GM_getValue: (key, defaultValue) => {
|
||||
key = PRE + key;
|
||||
const valueStr = localStorage.getItem(key);
|
||||
if (valueStr == null || valueStr === '') {
|
||||
return defaultValue;
|
||||
}
|
||||
try {
|
||||
return JSON.parse(valueStr).v;
|
||||
} catch (e) {
|
||||
}
|
||||
return valueStr;
|
||||
},
|
||||
// 设置配置
|
||||
GM_setValue: (key, value) => {
|
||||
key = PRE + key;
|
||||
localStorage.setItem(key, JSON.stringify({ v: value }));
|
||||
},
|
||||
// 删除设置
|
||||
GM_deleteValue: (key) => {
|
||||
key = PRE + key;
|
||||
localStorage.removeItem(key);
|
||||
},
|
||||
// 通知
|
||||
GM_notification: (details_or_text, ondone_or_title, image, onclick) => {
|
||||
// param1
|
||||
let options = typeof details_or_text === "string" ? { text: details_or_text } : details_or_text;
|
||||
if (typeof options !== "object") {
|
||||
console.error("GM_notification: 无效的参数值:details_or_text = " + details_or_text);
|
||||
return;
|
||||
}
|
||||
// param2
|
||||
if (typeof ondone_or_title === "string") {
|
||||
options.title = ondone_or_title;
|
||||
} else if (typeof ondone_or_title === "function") {
|
||||
options.ondone = ondone_or_title;
|
||||
} else if (ondone_or_title != null) {
|
||||
console.warn("GM_notification: 无效的参数值:ondone_or_title = " + ondone_or_title);
|
||||
}
|
||||
// param3
|
||||
if (typeof image === "string") {
|
||||
options.image = image;
|
||||
} else if (onclick != null) {
|
||||
console.warn("GM_notification: 无效的参数值:image = " + image);
|
||||
}
|
||||
// param4
|
||||
if (typeof onclick === "function") {
|
||||
options.onclick = onclick;
|
||||
} else if (onclick != null) {
|
||||
console.warn("GM_notification: 无效的参数值:onclick = " + onclick);
|
||||
}
|
||||
|
||||
let text = options.text;
|
||||
if (options.title) {
|
||||
text = options.title + ": " + text;
|
||||
}
|
||||
|
||||
// 显示通知方法
|
||||
const showNotification = () => {
|
||||
const notification = new Notification(text);
|
||||
if (options.timeout) {
|
||||
setTimeout(function () {
|
||||
notification.close();
|
||||
if (options.ondone) options.ondone(); // 回调
|
||||
}, options.timeout)
|
||||
}
|
||||
return notification;
|
||||
};
|
||||
const showAlert = () => {
|
||||
alert(text);
|
||||
if (options.ondone) options.ondone(); // 回调
|
||||
};
|
||||
|
||||
// 检查浏览器是否支持Notification API
|
||||
if (!("Notification" in window)) {
|
||||
showAlert(); // 不支持,直接使用alert显示通知
|
||||
}
|
||||
// 检查用户是否已授予权限
|
||||
else if (Notification.permission === "granted") {
|
||||
// 如果用户已授予权限,我们可以显示通知
|
||||
showNotification();
|
||||
}
|
||||
// 否则,先请求权限
|
||||
else if (Notification.permission !== 'denied') {
|
||||
Notification.requestPermission(function (permission) {
|
||||
if (permission === "granted") {
|
||||
showNotification(); // 用户接受权限,我们可以显示通知
|
||||
} else {
|
||||
showAlert(); // 用户驳回了权限,直接使用alert显示通知
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
})();
|
||||
console.log("ds_tampermonkey: completed")
|
|
@ -1,5 +1,5 @@
|
|||
const monkey = require('../../../monkey')
|
||||
const CryptoJs = require('crypto-js')
|
||||
// const CryptoJs = require('crypto-js')
|
||||
const lodash = require('lodash')
|
||||
const log = require('../../../../utils/util.log')
|
||||
|
||||
|
@ -9,9 +9,9 @@ const REMOVE = '[remove]' // 标记需要移除的头信息
|
|||
|
||||
function getScript (key, script) {
|
||||
const scriptUrl = SCRIPT_URL_PRE + key
|
||||
|
||||
const hash = CryptoJs.SHA256(script).toString(CryptoJs.enc.Base64)
|
||||
return `<script crossorigin="anonymous" defer="defer" type="application/javascript" src="${scriptUrl}" integrity="sha256-${hash}"></script>`
|
||||
// const hash = CryptoJs.SHA256(script).toString(CryptoJs.enc.Base64)
|
||||
// return `<script crossorigin="anonymous" defer="defer" type="application/javascript" src="${scriptUrl}" integrity="sha256-${hash}"></script>`
|
||||
return `<script crossorigin="anonymous" defer="defer" type="application/javascript" src="${scriptUrl}"></script>`
|
||||
}
|
||||
function getScriptByUrlOrPath (scriptUrlOrPath) {
|
||||
return `<script crossorigin="anonymous" defer="defer" type="application/javascript" src="${scriptUrlOrPath}"></script>`
|
||||
|
|
|
@ -4,20 +4,61 @@ const log = require('../../utils/util.log')
|
|||
let scripts
|
||||
|
||||
function buildScript (sc, content, scriptName) {
|
||||
const scriptKey = `ds_${scriptName}${sc.version ? ('_' + sc.version) : ''}:`
|
||||
|
||||
// 代码1:监听事件
|
||||
const runAt = sc['run-at'] || 'document-end'
|
||||
let eventStr
|
||||
if (runAt === 'document-end') {
|
||||
eventStr = 'document.addEventListener("DOMContentLoaded"'
|
||||
} else {
|
||||
eventStr = 'window.addEventListener("load"'
|
||||
}
|
||||
|
||||
// 代码2:初始化
|
||||
const options = {
|
||||
name: sc.name,
|
||||
icon: sc.icon
|
||||
}
|
||||
const initStr = `
|
||||
const DS_init = (window.__ds_global__ || {})['DS_init']
|
||||
if (typeof DS_init === 'function') {
|
||||
\tconsole.log("${scriptKey} do DS_init")
|
||||
\tDS_init(${JSON.stringify(options)});
|
||||
} else {
|
||||
\tconsole.log("${scriptKey} has no DS_init")
|
||||
}`
|
||||
|
||||
// 代码3:判断是否启用了脚本
|
||||
const checkEnabledStr = `
|
||||
if (!((window.__ds_global__ || {}).GM_getValue || (() => true))("ds_enabled", true)) {
|
||||
\tconsole.log("${scriptKey} disabled")
|
||||
\treturn
|
||||
}`
|
||||
|
||||
// 代码4:`GM_xxx` 方法读取
|
||||
let grantStr = ''
|
||||
for (const item of sc.grant) {
|
||||
if (grantStr.length > 0) {
|
||||
grantStr += '\r\n'
|
||||
}
|
||||
grantStr += (item.indexOf('.') > 0 ? '' : 'const ') + item + ' = window.__ds_global__[\'' + item + '\']'
|
||||
|
||||
if (item.indexOf('.') > 0) {
|
||||
grantStr += item + ' = (window.__ds_global__ || {})[\'' + item + '\'];'
|
||||
} else {
|
||||
grantStr += 'const ' + item + ' = (window.__ds_global__ || {})[\'' + item + '\'] || (() => {});'
|
||||
}
|
||||
}
|
||||
|
||||
return 'window.addEventListener("load", ()=> {\r\n' +
|
||||
grantStr + ';\r\n' +
|
||||
// 拼接脚本
|
||||
return eventStr + ', () => {' +
|
||||
initStr + '\r\n' +
|
||||
checkEnabledStr + '\r\n\r\n' +
|
||||
(grantStr ? (grantStr + '\r\n\r\n') : '') +
|
||||
content +
|
||||
(scriptName ? `\r\nconsole.log("ds_${scriptName} completed")` : '') +
|
||||
`\r\nconsole.log("${scriptKey} completed")` +
|
||||
'\r\n})' +
|
||||
(scriptName ? `\r\nconsole.log("ds_${scriptName} loaded")` : '')
|
||||
`\r\nconsole.log("${scriptKey} loaded")`
|
||||
}
|
||||
|
||||
function loadScript (content, scriptName) {
|
||||
|
@ -78,7 +119,7 @@ const api = {
|
|||
scripts.github = loadScript(readFile(rootDir, 'github.script'), 'github')
|
||||
scripts.google = loadScript(readFile(rootDir, 'google.js'), 'google')
|
||||
// scripts.jquery = { script: readFile(rootDir, 'jquery.min.js') }
|
||||
scripts.global = { script: readFile(rootDir, 'global.script') }
|
||||
scripts.tampermonkey = { script: readFile(rootDir, 'tampermonkey.script') }
|
||||
return scripts
|
||||
},
|
||||
loadScript
|
||||
|
|
Loading…
Reference in New Issue