|
|
|
@ -1,9 +1,11 @@
|
|
|
|
|
/** |
|
|
|
|
* 当前脚本为仿照的版本,并非篡改猴插件的源码,仅供学习参考。 |
|
|
|
|
* 篡改猴(Tampermonkey)| 油猴(Greasemonkey)浏览器脚本扩展 |
|
|
|
|
* |
|
|
|
|
* @name 篡改猴(Tampermonkey)| 油猴(Greasemonkey)浏览器脚本扩展 |
|
|
|
|
* @author 由 Wang Liang(王良)仿照的 |
|
|
|
|
* @version 0.1.1 |
|
|
|
|
* @since 2024-04-24 12:00 |
|
|
|
|
* @author 王良 |
|
|
|
|
* @authorHomePage https://wangliang1024.cn |
|
|
|
|
* @remark 当前脚本为仿照的版本,并非篡改猴插件的源码,仅供学习参考。 |
|
|
|
|
* @description 篡改猴 (Tampermonkey) 是拥有 超过 1000 万用户 的最流行的浏览器扩展之一。 它适用于 Chrome、Microsoft Edge、Safari、Opera Next 和 Firefox。 |
|
|
|
|
* 有些人也会把篡改猴(Tampermonkey)称作油猴(Greasemonkey),尽管后者只是一款仅适用于 Firefox 浏览器的浏览器扩展程序。 |
|
|
|
|
* 它允许用户自定义并增强您最喜爱的网页的功能。用户脚本是小型 JavaScript 程序,可用于向网页添加新功能或修改现有功能。使用 篡改猴,您可以轻松在任何网站上创建、管理和运行这些用户脚本。 |
|
|
|
@ -14,22 +16,29 @@
|
|
|
|
|
*/ |
|
|
|
|
'use strict'; |
|
|
|
|
(function () { |
|
|
|
|
const version = "0.1.1"; |
|
|
|
|
const PRE = "DS-Tampermonkey:"; // 前缀 |
|
|
|
|
const MENU_ID_PRE = PRE + "menu-"; |
|
|
|
|
|
|
|
|
|
const context = { |
|
|
|
|
initialized: false, // 是否已经初始化 |
|
|
|
|
defaultPluginOptions: {}, // 默认插件选项 |
|
|
|
|
pluginElement: null, // 插件div |
|
|
|
|
menusElement: null, // 菜单列表div |
|
|
|
|
pluginOptions: {}, // 插件选项 |
|
|
|
|
styleElement: null, // 插件样式元素 |
|
|
|
|
pluginElement: null, // 插件元素 |
|
|
|
|
arrowElement: null, // 箭头元素 |
|
|
|
|
menusElement: null, // 菜单列表元素 |
|
|
|
|
userMenusElement: null, // 用户菜单列表元素 |
|
|
|
|
menus: {}, // 菜单集合 |
|
|
|
|
menuIndex: 0, // 菜单索引,用于生成menuCmdId |
|
|
|
|
lastNotification: null // 最后一次通知 |
|
|
|
|
/*{ |
|
|
|
|
obj: null, // 通知对象 |
|
|
|
|
/* 最后一次通知的对象结构如下: |
|
|
|
|
{ |
|
|
|
|
obj: null, // 通知对象,类型:Notification |
|
|
|
|
options: null, // 通知选项 |
|
|
|
|
timeout: null // 通知定时器 |
|
|
|
|
}*/ |
|
|
|
|
} |
|
|
|
|
*/ |
|
|
|
|
}; |
|
|
|
|
|
|
|
|
|
|
|
|
|
@ -48,25 +57,32 @@
|
|
|
|
|
|
|
|
|
|
// 创建一个新的<style>元素 |
|
|
|
|
const styleElement = document.createElement('style'); |
|
|
|
|
styleElement.id = PRE + "plugin-style"; |
|
|
|
|
// 设置<style>元素的type属性 |
|
|
|
|
styleElement.type = 'text/css'; |
|
|
|
|
|
|
|
|
|
// 设置<style>元素的内容 |
|
|
|
|
let cssContent = ` |
|
|
|
|
.____ds-icon____{ |
|
|
|
|
.___ds-tampermonkey___{ |
|
|
|
|
position: fixed; |
|
|
|
|
right: 5px; |
|
|
|
|
right: 11px; |
|
|
|
|
top: 30%; |
|
|
|
|
z-index: 9999; |
|
|
|
|
width: 36px; |
|
|
|
|
height: 36px; |
|
|
|
|
border-radius: 8px; |
|
|
|
|
user-select: none; /* Standard syntax */ |
|
|
|
|
-webkit-user-select: none; /* Safari */ |
|
|
|
|
-ms-user-select: none; /* IE 10+/Edge */ |
|
|
|
|
background-color: #DDD; |
|
|
|
|
background-repeat: no-repeat; |
|
|
|
|
background-size: cover; |
|
|
|
|
background-image: url(""); |
|
|
|
|
} |
|
|
|
|
.____ds-menus____ { |
|
|
|
|
.___ds-tampermonkey-hide___{ |
|
|
|
|
width: 0; |
|
|
|
|
} |
|
|
|
|
.___ds-menus___{ |
|
|
|
|
display: none; |
|
|
|
|
position: absolute; |
|
|
|
|
right: 36px; |
|
|
|
@ -79,10 +95,13 @@
|
|
|
|
|
border: 1px solid #52525E; |
|
|
|
|
overflow: hidden; |
|
|
|
|
} |
|
|
|
|
.____ds-icon____:hover .____ds-menus____ { |
|
|
|
|
.___ds-tampermonkey___:hover:not(.___ds-tampermonkey-hide___) .___ds-menus___{ |
|
|
|
|
display: block; |
|
|
|
|
} |
|
|
|
|
.____ds-menu____ { |
|
|
|
|
.___ds-tampermonkey-hide___ .___ds-menus___{ |
|
|
|
|
display: none; |
|
|
|
|
} |
|
|
|
|
.___ds-menu___{ |
|
|
|
|
height: 35px; |
|
|
|
|
line-height: 35px; |
|
|
|
|
padding: 0 10px; |
|
|
|
@ -91,20 +110,38 @@
|
|
|
|
|
cursor: pointer; |
|
|
|
|
margin-left: 26px; |
|
|
|
|
} |
|
|
|
|
.____ds-menu____:hover { |
|
|
|
|
.___ds-menu___:hover{ |
|
|
|
|
background-color: #855F16; |
|
|
|
|
} |
|
|
|
|
.____ds-menu0____ { |
|
|
|
|
.___ds-menu0___{ |
|
|
|
|
margin-left: 0; |
|
|
|
|
font-size: 16px; |
|
|
|
|
font-weight: bold; |
|
|
|
|
} |
|
|
|
|
.____ds-menu0____ img{ |
|
|
|
|
.___ds-menu0___ img{ |
|
|
|
|
width: 23px; |
|
|
|
|
height: 23px; |
|
|
|
|
vertical-align: middle; |
|
|
|
|
margin: 0 8px 3px 8px; |
|
|
|
|
} |
|
|
|
|
.___ds-arrow___{ |
|
|
|
|
width: 0; |
|
|
|
|
height: 0; |
|
|
|
|
position: absolute; |
|
|
|
|
top: 11px; |
|
|
|
|
left: 37px; |
|
|
|
|
cursor: pointer; |
|
|
|
|
border-top: 7px solid transparent; |
|
|
|
|
border-bottom: 7px solid transparent; |
|
|
|
|
border-left: 10px solid #665c5c; |
|
|
|
|
} |
|
|
|
|
.___ds-tampermonkey-hide___ .___ds-arrow___{ |
|
|
|
|
border-top: 7px solid transparent; |
|
|
|
|
border-bottom: 7px solid transparent; |
|
|
|
|
border-right: 10px solid #665c5c; |
|
|
|
|
border-left: 0; |
|
|
|
|
left: 0; |
|
|
|
|
} |
|
|
|
|
`; |
|
|
|
|
// 如果有自定义样式,则添加到 CSS 内容中 |
|
|
|
|
if (options.style) { |
|
|
|
@ -121,23 +158,31 @@
|
|
|
|
|
|
|
|
|
|
// 将<style>元素添加到<head>中 |
|
|
|
|
document.head.append(styleElement); |
|
|
|
|
|
|
|
|
|
// 将<style>元素保存在上下文中 |
|
|
|
|
context.styleElement = styleElement; |
|
|
|
|
}; |
|
|
|
|
|
|
|
|
|
// 创建插件div |
|
|
|
|
api.createPluginDiv = (options) => { |
|
|
|
|
options = { |
|
|
|
|
...{ name: "油猴脚本" }, |
|
|
|
|
...{ name: "未知名的脚本" }, |
|
|
|
|
...options |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// 创建插件div |
|
|
|
|
context.pluginElement = document.createElement('div'); |
|
|
|
|
context.pluginElement.title = "油猴脚本" + (options.name ? ":" + options.name : ""); |
|
|
|
|
context.pluginElement.className = "____ds-icon____"; |
|
|
|
|
context.pluginElement.id = PRE + "plugin"; |
|
|
|
|
context.pluginElement.title = "油猴插件" + (options.name ? ":" + options.name : ""); |
|
|
|
|
context.pluginElement.className = "___ds-tampermonkey___"; |
|
|
|
|
if (api.GM_getValue("ds_hide")) { |
|
|
|
|
context.pluginElement.classList.add("___ds-tampermonkey-hide___"); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// 创建菜单列表div |
|
|
|
|
context.menusElement = document.createElement('div'); |
|
|
|
|
context.menusElement.className = "____ds-menus____"; |
|
|
|
|
context.menusElement.id = PRE + "menus"; |
|
|
|
|
context.menusElement.className = "___ds-menus___"; |
|
|
|
|
if (options.width > 0) { |
|
|
|
|
context.menusElement.style['min-width'] = options.width + "px"; |
|
|
|
|
} |
|
|
|
@ -149,16 +194,16 @@
|
|
|
|
|
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.className = "___ds-menu___ ___ds-menu0___"; |
|
|
|
|
switchMenuElement.innerHTML = (enabled ? "✅" : "❌") + icon + options.name; |
|
|
|
|
switchMenuElement.title = `点击${enabled ? "关闭" : "开启"}此脚本功能`; |
|
|
|
|
switchMenuElement.onclick = function () { |
|
|
|
|
let enabled = api.GM_getValue("ds_enabled", true) |
|
|
|
|
if (enabled) { |
|
|
|
|
api.hideMenus(); |
|
|
|
|
api.hideUserMenus(); |
|
|
|
|
enabled = false; |
|
|
|
|
} else { |
|
|
|
|
api.showMenus(); |
|
|
|
|
api.showUserMenus(); |
|
|
|
|
enabled = true; |
|
|
|
|
} |
|
|
|
|
switchMenuElement.innerHTML = (enabled ? "✅" : "❌") + icon + options.name; |
|
|
|
@ -172,44 +217,110 @@
|
|
|
|
|
// 将开关菜单添加到菜单列表div中 |
|
|
|
|
context.menusElement.append(switchMenuElement); |
|
|
|
|
|
|
|
|
|
// 创建用户菜单列表div |
|
|
|
|
context.userMenusElement = document.createElement('div'); |
|
|
|
|
context.userMenusElement.id = PRE + "user-menus"; |
|
|
|
|
context.userMenusElement.className = "___ds-user-menus___"; |
|
|
|
|
// 将用户菜单div添加到菜单div中 |
|
|
|
|
context.menusElement.append(context.userMenusElement); |
|
|
|
|
|
|
|
|
|
// 获取body元素 |
|
|
|
|
const body = document.getElementsByTagName('body')[0]; |
|
|
|
|
// 将插件div添加到body中 |
|
|
|
|
body.prepend(context.pluginElement); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// 显示菜单列表 |
|
|
|
|
api.showMenus = () => { |
|
|
|
|
for (const menuCmdId in context.menus) { |
|
|
|
|
const menuElement = context.menus[menuCmdId].element; |
|
|
|
|
menuElement.style.display = "block"; |
|
|
|
|
// 创建箭头 |
|
|
|
|
api.createArrow = (options) => { |
|
|
|
|
// 创建箭头元素 |
|
|
|
|
context.arrowElement = document.createElement('div'); |
|
|
|
|
context.arrowElement.id = PRE + "arrow"; |
|
|
|
|
context.arrowElement.className = "___ds-arrow___"; |
|
|
|
|
// 初始化title |
|
|
|
|
api.initArrowTitle(); |
|
|
|
|
// 绑定点击事件 |
|
|
|
|
context.arrowElement.onclick = () => { |
|
|
|
|
if (__ds_global__.getContext().pluginElement.classList.contains("___ds-tampermonkey-hide___")) { |
|
|
|
|
api.showPlugin(); |
|
|
|
|
api.initArrowTitle(false); |
|
|
|
|
} else { |
|
|
|
|
api.hidePlugin(); |
|
|
|
|
api.initArrowTitle(true); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
// 将箭头元素添加到插件div中 |
|
|
|
|
context.pluginElement.append(context.arrowElement); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
api.initArrowTitle = (isHidden) => { |
|
|
|
|
if (isHidden == null) { |
|
|
|
|
isHidden = context.pluginElement.classList.contains("___ds-tampermonkey-hide___"); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
if (isHidden) { |
|
|
|
|
context.arrowElement.title = "点击展示「油猴插件」的操作界面"; |
|
|
|
|
} else { |
|
|
|
|
context.arrowElement.title = "点击隐藏「油猴插件」的操作界面"; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// 隐藏插件 |
|
|
|
|
api.hidePlugin = () => { |
|
|
|
|
if (context.pluginElement) { |
|
|
|
|
context.pluginElement.classList.add("___ds-tampermonkey-hide___"); |
|
|
|
|
} |
|
|
|
|
api.GM_setValue("ds_hide", true); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// 显示插件 |
|
|
|
|
api.showPlugin = () => { |
|
|
|
|
if (context.pluginElement) { |
|
|
|
|
context.pluginElement.classList.remove("___ds-tampermonkey-hide___"); |
|
|
|
|
} |
|
|
|
|
api.GM_setValue("ds_hide", false); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// 隐藏菜单列表 |
|
|
|
|
api.hideMenus = () => { |
|
|
|
|
for (const menuCmdId in context.menus) { |
|
|
|
|
const menuElement = context.menus[menuCmdId].element; |
|
|
|
|
menuElement.style.display = "none"; |
|
|
|
|
// 显示用户菜单列表 |
|
|
|
|
api.showUserMenus = () => { |
|
|
|
|
if (context.userMenusElement) { |
|
|
|
|
context.userMenusElement.style.display = "block"; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// 隐藏用户菜单列表 |
|
|
|
|
api.hideUserMenus = () => { |
|
|
|
|
if (context.userMenusElement) { |
|
|
|
|
context.userMenusElement.style.display = "none"; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// 初始化篡改猴操作界面 |
|
|
|
|
api.DS_init = (options) => { |
|
|
|
|
try { |
|
|
|
|
// 如果已经初始化过,则直接返回 |
|
|
|
|
if (context.initialized) return; |
|
|
|
|
|
|
|
|
|
// 合并默认参数 |
|
|
|
|
options = { |
|
|
|
|
...context.defaultPluginOptions, |
|
|
|
|
...options |
|
|
|
|
}; |
|
|
|
|
|
|
|
|
|
// 创建样式元素 |
|
|
|
|
api.createPluginStyle(options); |
|
|
|
|
// 创建插件div |
|
|
|
|
api.createPluginDiv(options); |
|
|
|
|
// 创建箭头 |
|
|
|
|
api.createArrow(options); |
|
|
|
|
// 保存参数 |
|
|
|
|
context.pluginOptions = options; |
|
|
|
|
|
|
|
|
|
// 初始化完成 |
|
|
|
|
context.initialized = true; |
|
|
|
|
|
|
|
|
|
console.log("ds_tampermonkey: initialization completed(篡改猴插件初始化完成,篡改猴图标已显示在页面右侧,鼠标移到上面可展示功能列表!)") |
|
|
|
|
console.log(`ds_tampermonkey_${version}: initialization completed(篡改猴插件初始化完成,篡改猴图标已显示在页面右侧,鼠标移到上面可展示功能列表!)`) |
|
|
|
|
} catch (e) { |
|
|
|
|
console.error("ds_tampermonkey: initialization failed(篡改猴插件初始化失败):", e); |
|
|
|
|
console.error(`ds_tampermonkey_${version}: initialization failed(篡改猴插件初始化失败):`, e); |
|
|
|
|
} |
|
|
|
|
}; |
|
|
|
|
|
|
|
|
@ -225,10 +336,18 @@
|
|
|
|
|
// 生成菜单ID |
|
|
|
|
let menuCmdId; |
|
|
|
|
if (options.id) { |
|
|
|
|
if (options.id.indexOf(MENU_ID_PRE) === 0) { |
|
|
|
|
menuCmdId = options.id; |
|
|
|
|
} else { |
|
|
|
|
menuCmdId = MENU_ID_PRE + options.id; |
|
|
|
|
if (typeof options.id !== "string") { |
|
|
|
|
options.id = options.id.toString(); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
menuCmdId = (options.id.indexOf(MENU_ID_PRE) === 0 ? '' : MENU_ID_PRE) + options.id; |
|
|
|
|
|
|
|
|
|
// 如果是数字ID,为了避免与自增ID索引冲突,将数字ID赋值给自增ID索引 |
|
|
|
|
if (options.id.match("^\\d+$")) { |
|
|
|
|
const numberId = parseInt(options.id); |
|
|
|
|
if (numberId > context.menuIndex) { |
|
|
|
|
context.menuIndex = numberId; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} else { |
|
|
|
|
menuCmdId = MENU_ID_PRE + (++context.menuIndex); |
|
|
|
@ -237,7 +356,7 @@
|
|
|
|
|
// 创建菜单元素 |
|
|
|
|
const menuElement = document.createElement('div'); |
|
|
|
|
menuElement.id = menuCmdId; |
|
|
|
|
menuElement.className = "____ds-menu____"; |
|
|
|
|
menuElement.className = "___ds-menu___"; |
|
|
|
|
menuElement.innerHTML = name; |
|
|
|
|
if (options.title) { |
|
|
|
|
menuElement.title = typeof options.title === "function" ? options.title() : options.title; |
|
|
|
@ -250,7 +369,7 @@
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// 将菜单元素添加到菜单列表div中 |
|
|
|
|
context.menusElement.append(menuElement); |
|
|
|
|
context.userMenusElement.append(menuElement); |
|
|
|
|
|
|
|
|
|
// 将菜单添加到菜单集合中 |
|
|
|
|
context.menus[menuCmdId] = { |
|
|
|
@ -270,15 +389,24 @@
|
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
if (typeof menuCmdId !== "string") { |
|
|
|
|
menuCmdId = menuCmdId.toString(); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
if (menuCmdId.indexOf(MENU_ID_PRE) !== 0) { |
|
|
|
|
menuCmdId = MENU_ID_PRE + menuCmdId; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
const menu = context.menus[menuCmdId]; |
|
|
|
|
if (menu) { |
|
|
|
|
menu.element.remove(); |
|
|
|
|
delete context.menus[menuCmdId]; |
|
|
|
|
} else { |
|
|
|
|
const menuElement = document.getElementById(menuCmdId) |
|
|
|
|
if (menuElement) { |
|
|
|
|
menuElement.remove(); |
|
|
|
|
} |
|
|
|
|
delete context.menus[menuCmdId]; |
|
|
|
|
} |
|
|
|
|
}; |
|
|
|
|
|
|
|
|
|
// 打开新标签 |
|
|
|
@ -321,7 +449,7 @@
|
|
|
|
|
// 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); |
|
|
|
|
console.error(`ds_tampermonkey_${version}: GM_notification: 无效的参数值:details_or_text = ` + details_or_text); |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
// param2 |
|
|
|
@ -330,19 +458,19 @@
|
|
|
|
|
} 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); |
|
|
|
|
console.warn(`ds_tampermonkey_${version}: 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); |
|
|
|
|
console.warn(`ds_tampermonkey_${version}: GM_notification: 无效的参数值:image = ` + image); |
|
|
|
|
} |
|
|
|
|
// param4 |
|
|
|
|
if (typeof onclick === "function") { |
|
|
|
|
options.onclick = onclick; |
|
|
|
|
} else if (onclick != null) { |
|
|
|
|
console.warn("GM_notification: 无效的参数值:onclick = " + onclick); |
|
|
|
|
console.warn(`ds_tampermonkey_${version}: GM_notification: 无效的参数值:onclick = ` + onclick); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
let text = options.text; |
|
|
|
@ -416,5 +544,5 @@
|
|
|
|
|
module.exports = api; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
console.log("ds_tampermonkey: completed") |
|
|
|
|
console.log(`ds_tampermonkey_${version}: completed`) |
|
|
|
|
})(); |