// ==UserScript==
// @name BeautyToMac
// @namespace http://tampermonkey.net/
// @version 0.1.8rc1
// @icon 
// @description 给你的百度和360搜索一个类似Mac/Gnome的浏览体验
// @author CloundMark
// @match *://www.baidu.com/*
// @match *://baidu.com/*
// @match *://www.so.com/*
// @match *://so.com/*
// @license GPLv3
// @run-at document-start
// @grant GM_xmlhttpRequest
// @grant GM.xmlHttpRequest
// @grant GM_registerMenuCommand
// @grant GM_setValue
// @grant GM_getValue
// @connect *
// ==/UserScript==
(function () {
/* 全局配置项 start*/
const userGlobalConfigs = {
// 网站开关 true启用,false关闭;
baidu: true,
sou360: true,
// 调试模式开关
isDebug: false,
// 0、全局背景; bg代表背景图像,你可以从网络上引用自己喜欢的,used 为true表示启用当前分类或具体的图像,false弃用;
// 内部的name建议您始终加上(但也不是必须),主要是便于哪天不需要的时候可以快速定位;
// 对于本地图像,可以上传到免费的在线图床之后再获取链接加入,比如 https://imgse.com/
// 请仿照示例添加背景
bg: {
used: true,
sources: {
'cartoon': {
used: true,
sources: [
{ used: true, name: '粉红猫', url: 'https://i0.hdslb.com/bfs/article/f45de07102e3fe83331d68dd455336ab0b2a08b3.png@942w_566h_progressive.png' },
{ used: true, name: '可爱猫', url: 'https://i0.hdslb.com/bfs/article/dec6e7a1969748f2b4462c688367d772798d4134.png@942w_668h_progressive.png' },
{ used: true, name: '两仪式', url: 'https://i0.hdslb.com/bfs/article/c7cb8fecdd0cd7d34e89319c4b9eeb4ab543cfd0.jpg@1320w_740h.jpg' },
{ used: true, name: '远坂凛', url: 'https://img9.51tietu.net/pic/2019-091402/jr354qpj03ujr354qpj03u.jpg' },
{ used: true, name: '红猫', url: 'https://mms0.baidu.com/it/u=2246948513,3190507093&fm=253&app=138&f=JPEG&fmt=auto&q=75?w=800&h=500' },
//不清晰{ used: true, name: '红猫', url: 'https://gimg2.baidu.com/image_search/src=http%3A%2F%2Fhbimg.b0.upaiyun.com%2F011e9ec28bb2ef90550a090d2121296ee2cfe88233f56-lV1t8X_fw658&refer=http%3A%2F%2Fhbimg.b0.upaiyun.com&app=2002&size=f9999,10000&q=a80&n=0&g=0n&fmt=auto?sec=1661358344&t=a4aeb02bf2d960bed1a219c17de13543' },
{ used: true, name: '蓝狗', url: 'https://i0.hdslb.com/bfs/article/78c032adb6df5717fc4ae465982a9d2096984223.jpg@942w_531h_progressive.jpg' },
{ used: true, name: '两仪式2', url: 'https://img2.baidu.com/it/u=4177750439,794883634&fm=253&fmt=auto&app=138&f=JPEG?w=935&h=500' },
{ used: true, name: 'saber', url: 'https://i0.hdslb.com/bfs/article/49c217c9ec1e9cd707a228a509df0e7d57e0dd6e.jpg@1320w_740h.jpg' },
{ used: true, name: '雅尔贝德', url: 'https://i0.hdslb.com/bfs/article/38f4427201389a1fe54c12d9a62ca57681e06b3a.jpg@1320w_740h.webp' },
{ used: true, name: 'saber2', url: 'https://i0.hdslb.com/bfs/article/0fa36a51ebd26d1d2a5621385910d9b35cdc681a.jpg@942w_590h_progressive.jpg' },
{ used: true, name: '雅尔贝德2', url: 'https://i0.hdslb.com/bfs/article/e7cfb2d9768b707f16685a83d7126db4787042b2.jpg@942w_668h_progressive.webp' },
{ used: true, name: '白猫', url: 'https://i0.hdslb.com/bfs/article/64f062d70fdba96b39dd346ab9d33dc5bf77c17f.jpg@942w_498h_progressive.webp' },
{ used: true, name: '贞子', url: 'https://i0.hdslb.com/bfs/article/87f6a5c7b25307c4cc7bd86acca8ed51dec25dcc.jpg@942w_666h_progressive.webp' },
{ used: true, name: '大鱼', url: 'https://i0.hdslb.com/bfs/article/73b003ea1c04fb3d1ed736773b3f752a8f3acfc5.jpg@942w_596h_progressive.webp' },
{ used: true, name: '空蓝', url: 'https://i0.hdslb.com/bfs/article/86bed00f68bbff662e202aa91215336bb3dd251e.jpg@942w_587h_progressive.webp' },
{ used: true, name: '樱', url: 'https://i0.hdslb.com/bfs/article/fa372e5e0a70b630b5b7eb9b4a0d1125a83535ce.png' },
{ used: true, name: '薇尔莉特', url: 'https://i0.hdslb.com/bfs/article/59af6572e5ea374e434a0d0cbb78c5a83e159939.jpg@942w_531h_progressive.webp' },
{ used: true, name: '紫', url: 'https://img1.baidu.com/it/u=2725650397,3822860203&fm=253&fmt=auto&app=120&f=JPEG?w=1422&h=800' },
{ used: true, name: '伊莉雅', url: 'https://i0.hdslb.com/bfs/article/59b79113f3a8be33ae75009c862a4bb47fc6d87a.jpg@942w_531h_progressive.webp' },
{ used: true, name: '薇尔莉特2', url: 'https://i0.hdslb.com/bfs/article/0a5f6397857f8f840e9c4bb6a7b52b9623aefcd7.jpg@942w_590h_progressive.webp' },
{ used: true, name: '薇尔莉特3', url: 'https://i0.hdslb.com/bfs/article/ec68b813ca3a56a1e23190f9e8a255feb1d2d5e7.jpg@1320w_740h.webp' },
{ used: true, name: '薇尔莉特4', url: 'https://i0.hdslb.com/bfs/article/7085568797e6a1923056990f71888fe9706d644b.jpg@1320w_740h.webp' },
{ used: true, name: '伊莉雅2', url: 'https://img0.baidu.com/it/u=3169921451,1554186398&fm=253&fmt=auto&app=120&f=JPEG?w=1422&h=800' },
{ used: true, name: '希丝缇娜', url: 'https://i0.hdslb.com/bfs/article/65da66f0bd9e888ad572c9ab9100e4b741f2019d.jpg@942w_530h_progressive.webp' },
// 引自B站 https://www.bilibili.com/read/cv6710298?from=search
{ used: true, name: '深', url: 'https://i0.hdslb.com/bfs/article/5c40299551a8dc24fe8c55016283206b370132aa.jpg@942w_545h_progressive.webp' },
// 引自B站 https://www.bilibili.com/read/cv17176820?from=search 原画师:杉87
{ used: true, name: '晚霞', url: 'https://i0.hdslb.com/bfs/article/61e693800a1f2dbab634bbb30524b151523d09fa.jpg@942w_531h_progressive.webp' },
// 引自B站 https://www.bilibili.com/read/cv17176820?from=search 原画师:Teardrops
{ used: true, name: '望', url: 'https://i0.hdslb.com/bfs/article/7ff810942db808887d96ee65cb0f469bb68eb9aa.jpg@942w_531h_progressive.webp' },
// 原画师 ohara_tometa(小原トメ太)
{ used: true, name: 'Bunnies', url: 'https://iknow-pic.cdn.bcebos.com/c2fdfc039245d6888182ff59a5c27d1ed31b244e' },
{ used: true, name: '艾莉丝', url: 'https://gimg2.baidu.com/image_search/src=http%3A%2F%2Fp2.itc.cn%2Fimages01%2F20210309%2Ff14fae9f68c443f1835e002513fe34fd.jpeg&refer=http%3A%2F%2Fp2.itc.cn&app=2002&size=f9999,10000&q=a80&n=0&g=0n&fmt=auto?sec=1670083703&t=5b75cd70ffbb1e67877a30ae22c5d140' },
{ used: true, name: '近月少女', url: 'https://p7.itc.cn/q_70/images03/20220624/1b26d1898ee041e2a16b54a1a5181886.jpeg' },
{ used: true, name: '近月少女2', url: 'https://i0.hdslb.com/bfs/article/75fe779e3694b006a6cb98bdeb0d440d5b8bb83d.jpg' },
{ used: true, name: '近月少女3', url: 'https://gimg2.baidu.com/image_search/src=http%3A%2F%2Fimg1.gamersky.com%2Fupimg%2Fusers%2F2022%2F06%2F18%2Forigin_202206180231253771.jpg&refer=http%3A%2F%2Fimg1.gamersky.com&app=2002&size=f9999,10000&q=a80&n=0&g=0n&fmt=auto?sec=1673282937&t=58d7062b4cab1dc2c25204c72e1d8e4e' },
{ used: true, name: '', url: '' },
{ used: true, name: '', url: '' },
{ used: true, name: '', url: '' },
]
},
'natures': {
used: false,
sources: [
{ used: true, name: '', url: '' },
{ used: true, name: '', url: '' },
]
},
'Girls': {
used: false,
sources: [
{ used: true, name: '', url: '' },
{ used: true, name: '', url: '' },
]
},
}
},
// 1、背景切换方式
bgSwitchMode: 'multi',// 可选枚举值:['once','multi'];
// 2、设置背景切换的最短时间间隔,单位s;小于3s将设为3s;
duration: 5,
// 3、启用低速网络模式,非false以开启;建议始终选择low;一下网速依次越差;
// 可选枚举值:[false,'low','middle','heigh','infinity'];
netThrottlMode: 'low',
// 4、枚举值,设定中央版块的主题色
// 可选枚举值:[各种颜色的16进制表示];false禁用中央面板色彩(当前全局为护眼色#C7EDCC80)并启用默认背景
// 你可以在下方给每个站点分别配置
centerColor: '#C7EDCC80',
// 5、请求任意站点;如果你的图像来自可能被拒的其他网站服务提供者,那么启用此选项可以绕过浏览器限制
requestAllWebsite: true,
// 6、启用键盘导航,设为false以禁用
keyBoardNav: true,
// 7、启用左右空白导航,设为false以禁用
whiteNav: true,
// 8、开启快捷键ctrl+y搜索聚焦
allowSearchBox: true,
};
class debugLog {
static get Priority() {
return ['log', 'summary', 'warn', 'error', 'critical', 'dir'];
}
constructor(isDebuger, debugPriority) {
this.isDebuger = isDebuger; // 要使用全局配置
this.debugPriority = debugPriority || this.constructor.Priority;
this.log = this.wrapperLogPriority();
this.summary = this.wrapperLogPriority('summary', 'color:cyan');
this.warn = this.wrapperLogPriority('warn', 'color:yellow');
this.error = this.wrapperLogPriority('error', 'color:red');
this.critical = this.wrapperLogPriority('critical', 'color:black;background-color:red;');
this.dir = this.wrapperObjPriority('dir');
for (let obj of [this, this.debugPriority]) Object.freeze(obj);
}
wrapperLogPriority(actionPriority, style) {
let priority = actionPriority || 'log'; // log,即info消息级别
if (!this.constructor.Priority.includes(priority)) throw TypeError(`Cant allow [${priority}]`);
let outpuMethod = console.log;
return (...rest) => {
if (this.isDebuger && this.debugPriority.includes(priority)) {
outpuMethod(`%c${rest}`, style || 'color:lime');
}
}
}
wrapperObjPriority(actionPriority) {
let priority = actionPriority;
if (!this.constructor.Priority.includes(priority)) throw TypeError(`Cant allow [${priority}]`);
let outpuMethod = console.dir;
return (...rest) => {
if (this.isDebuger && this.debugPriority.includes(priority)) {
outpuMethod(...rest);
}
}
}
}
//const logger = new debugLog(userGlobalConfigs.isDebug,['log','summary','warn','error','critical']);
const logger = new debugLog(userGlobalConfigs.isDebug);
const URL = window.URL || window.webkitURL || window.mozURL;
const MutationObserver = window.MutationObserver || window.WebKitMutationObserver || window.MozMutationObserver;
// eslint-disable-next-line no-undef
const GMxmlHttpRequest = typeof GM_xmlhttpRequest === 'function' ? GM_xmlhttpRequest : GM.xmlHttpRequest;
/* 基类 */
class Base {
constructor() {
this.priority = 'Base';
}
setBFInterval(fn, interval, ...rest) {
let id = setInterval((...rest) => {
fn.call(this, ...rest);
}, interval, ...rest);
fn.call(this, ...rest);
return id;
}
findElement(selector, timoout = 5, fq = false, intvl = 100, signal = null) {
if (!(selector && typeof selector === 'string')) throw TypeError('selector must be non-zero string.')
return new Promise((success, reject) => {
let timer4min, timer4big, mt, timer4HTML;
// 启用用户中断
if (signal && Object.prototype.toString.call(signal).slice(8, -1) === 'AbortSignal') {
signal.addEventListenner('abort', () => {
logger.warn('异步查询,用户中断查找:', selector);
reject(signal.reason);
clearInterval(timer4min);
clearTimeout(timer4big);
mt && mt.disconnect();
});
}
// 开启快查询
if (fq) {
timer4min = this.setBFInterval(() => {
let result = document.querySelectorAll(selector);
if (result.length) {
success(result);
clearInterval(timer4min);
clearTimeout(timer4big);
mt && mt.disconnect();
}
}, intvl);
}
//开启超时策略
timer4big = setTimeout(() => {
reject(`TimeOut[${selector}]`);
clearInterval(timer4min);
mt && mt.disconnect();
}, timoout * 1000);
//开启DOM监测
mt = new MutationObserver((_, obs) => {
let result = document.querySelectorAll(selector);
if (result.length) {
success(result);
clearInterval(timer4min);
clearTimeout(timer4big);
obs.disconnect();
}
});
if (document && document.documentElement && document.documentElement.tagName.toUpperCase() === 'HTML') {
mt.observe(document.documentElement, { childList: true, subtree: true });
} else {
//说明网速极度不好
timer4HTML = this.setBFInterval(function () {
if (document && document.documentElement && document.documentElement.tagName.toUpperCase() === 'HTML') {
mt.observe(document.documentElement, { childList: true, subtree: true });
clearInterval(timer4HTML);
}
}, 15)
}
});
}
sendGMXHR(url, method, loadCB, errorCB, timeoutCB, timeout) {
GMxmlHttpRequest({
url,
method: method || 'get',
timeout,
responseType: 'blob',
onload: loadCB,
onerror: errorCB,
ontimeout: timeoutCB
});
}
getHeadersFromGMXHR(headersString = '') {
// 油猴响应头转换成对象
let head = new Headers();
let items = headersString.split('\n').filter(o => o);
let k = '', v = '', idx;
for (let item of items) {
item = item.trim();
idx = item.indexOf(':');
if (idx === -1) continue;
k = item.slice(0, idx).trimEnd(), v = item.slice(idx + 1).trimStart();
if (k && v) head.append(k, v);
}
return head;
}
shuffle(arr = []) {
if (arr.length < 2) return arr;
let l = arr.length;
while (l) {
let j = Math.floor(Math.random() * l--);
[arr[l], arr[j]] = [arr[j], arr[l]];
}
return arr;
}
getType(val) {
return Object.prototype.toString.call(val).slice(8, -1);
}
forceCheck(fn, argsType = [], returnType = []) {
if (!(typeof fn === 'function' &&
Array.isArray(argsType) &&
Array.isArray(returnType))) throw TypeError('类型错误:必须是[function, array, array]');
return function () {
//if(fnWrapper.prototype[Symbol.toStringTag]!==fn.name) fnWrapper.prototype[Symbol.toStringTag]=fn.name;
let tp = '';
if (arguments.length) {
for (let i = 0; i < argsType.length; i++) {
tp = this.getType(arguments[i]);
if (tp !== argsType[i]) {
throw TypeError(`函数:[${fn.name}]的[第${i + 1}个]参数类型要求是[${argsType[i]}],但传入的是[${tp}]`);
}
}
}
let result = fn.prototype ? fn.call(this, ...arguments) : fn(...arguments);
if (returnType.length) {
tp = this.getType(result);
if (returnType.includes(tp)) {
return result;
} else {
throw TypeError(`函数:[${fn.name}]的返回参数类型要求是[${returnType}],但返回的是[${tp}]`);
}
} else {
return result;
}
};
}
openCaches(cacheName) {
if (window.caches && window.isSecureContext) {
return caches.open(cacheName)
} else {
logger.log('Current context is not secure');
return Promise.resolve();
}
}
setCaches(cacheName, url, response) {
this.openCaches(cacheName).then(myCache => {
myCache && myCache.keys().then(urls => {
let urlsStr = urls.map(item => item.url);
if (!urlsStr.includes(url)) {
myCache.put(url, response)
.then(
() => logger.log(`Cache set ok.-->[${url}]`)
).catch(
(e) => logger.error(`Cache set is wrong.->info[${e.message}]-->[${url}]`)
)
}
})
})
}
getCache(cacheName, url, options) {
return new Promise((success) => {
this.openCaches(cacheName).then(myCache => {
if (myCache) {
myCache.match(url, options || { ignoreVary: true })
.then(rsp => success(rsp))
.catch(err => {
logger.error(`Get cache failed. Info-->${err}\nfor ${url}`);
success(false);
})
} else {
success(null);
}
})
})
}
addCaches(cacheName, url, failCallBack, initSuccessCallBack) {
// 无则添加,有则忽略
this.openCaches(cacheName).then(myCache => {
myCache && myCache.keys().then(urls => {
let urlsStr = urls.map(item => item.url);
if (urlsStr.includes(url)) {
logger.log(`Add cache, but url exists, ignore. [${url}]`);
} else {
myCache.add(url)
.then(() => {
logger.log(`First add cache success, [${url}]`);
if (initSuccessCallBack) logger.log('InitSuccessCallBack add cache Callback exist. Try to callback.');
initSuccessCallBack && initSuccessCallBack();
})
.catch(err => {
logger.error(`Add cache failed. Info-->${err}\nfor ${url}`);
if (failCallBack) logger.log('Add cache Callback exist. Try to callback.');
failCallBack && failCallBack();
})
}
})
})
}
updateCaches(cacheName, url, failCallBack) {
this.openCaches(cacheName).then(myCache => {
myCache && myCache.delete(url, { ignoreVary: true }).then(() => {
myCache.add(url).catch(err => {
logger.error(`Update cache failed. Info-->${err}\nfor ${url}`);
if (failCallBack) logger.log('Update cache Callback exist. Try to callback.')
failCallBack && failCallBack();
})
}, () => {
logger.error('Update cache something is wrong, get fault.')
})
})
}
deleteCaches(cacheName, url, options) {
this.openCaches(cacheName).then(myCache => {
myCache && myCache.keys().then(urls => {
if (urls.includes(url)) {
myCache.delete(url, options || { ignoreVary: true }).then(
ok => logger.log(`Delete cache success. Info-->${ok}\nfor ${url}`)
).catch(
err => logger.error(`Delete cache failed. Info-->${err}\nfor ${url}`)
)
} else {
logger.warn(`No url -->[${url}],ignore.`);
}
})
})
}
getAllCachesUrl(cacheName) {
return new Promise(success => {
this.openCaches(cacheName).then(myCache => {
if (myCache) {
myCache.keys().then(urls => {
success(urls.map(item => item.url));
})
} else {
success(null);
}
})
})
}
hasCaches(cacheName, url) {
return new Promise(success => {
this.openCaches(cacheName).then(myCache => {
if (myCache) {
myCache.keys().then(urls => {
let urlsStr = urls.map(item => item.url);
if (urlsStr.includes(url)) {
success(true);
} else {
success(false);
}
})
} else {
success(null);
}
})
})
}
run() {
throw TypeError('ChildrenClass must have this method!');
}
}
class NoAdBase extends Base {
constructor(selector) {
super();
this.selector = selector;
this.allElements = [];
}
}
/* 反广告 */
class AntiAdSoft extends NoAdBase {
constructor(selector) {
super(selector);
this.allElements = [...document.querySelectorAll(this.selector)];
Reflect.defineProperty(this, 'priority', { value: 'NoAdSoft', configurable: false, enumerable: true, writable: false });
}
run() {
for (let i = 0; i < this.allElements.length; i++) {
this.allElements[i].style.cssText += 'display:none !important';
}
}
}
class AntiAdHard extends NoAdBase {
constructor(selector) {
super(selector);
this.allElements = null;
Reflect.defineProperty(this, 'priority', { value: 'NoAdHard', configurable: false, enumerable: true, writable: false });
}
run() {
this.findElement(this.selector).then(results => {
this.allElements = [...results];
if (this.allElements.length) {
this.allElements.forEach(item => item.remove());
}
}).catch(info => {
logger.warn(`AntiHard No ${info}.`);
})
}
}
class AntiAdDynamic extends NoAdBase {
constructor(adDynamic) {
super();
//dynamicRules
if (!this.constructor.UNIQUEINSTANCE) {
this.dynamicRules = adDynamic;
this.maid = [];
Reflect.defineProperty(this, 'priority', { value: 'NoAdDynamic', configurable: false, enumerable: true, writable: false });
Reflect.defineProperty(this.constructor, 'UNIQUEINSTANCE', { value: this, configurable: false, enumerable: false, writable: false });
} else {
return this.constructor.UNIQUEINSTANCE;
}
}
run() {
for (let item of this.dynamicRules) {
let mt = new MutationObserver(() => {
let tmpNodeList = document.querySelectorAll(item.ref);
if (tmpNodeList.length) {
let resultWrapper = item.fire(tmpNodeList);
resultWrapper.forEach(o => o.remove());
}
})
this.maid.push(mt);
let eye = document.querySelector(item.eyeSlector);
if (eye) {
mt.observe(eye, item.options);
} else {
this.findElement(item.eyeSlector)
.then(eles => mt.observe(eles[0], item.options))
.catch(info => logger.warn(`${info}---无广告参照点:${item.ref} from ${item.eyeSlector}.`));
}
}
}
}
/* 处理样式的基类 */
class ModifyStyleBase extends Base {
constructor(selector) {
super(selector);
}
getNewElement(tag, options = {}) {
let tmp = document.createElement(tag);
Object.assign(tmp, options);
return tmp;
}
setStyle() {
throw TypeError('ChildrenClass must have this method!');
}
removeStyle() {
throw TypeError('ChildrenClass must have this method!');
}
getStyle() {
throw TypeError('ChildrenClass must have this method!');
}
observeStyle() {
throw TypeError('ChildrenClass must have this method!');
}
stringToMap(styleText) {
// 将 “div{height:20px;width:30px;content:':;'}span a{color:'red';text-align:center;}” 转换成
// Map对象 {'selector':{'attr1':'value1','attr2':'value2'}}
// 原因是用户输入可能不规范,转成统一格式便于处理
// 输入content属性不能包含 {};
let result = new Map();
if (!styleText) return result;
let pattern4InnerText = /[\w+~|*$()[\]\s#:=>,"'-^]+{[^}]*?}/gi;
let pattern4PerRule = /([\w+~|*$()[\]\s#:=>,"'-^]+){([^}]*?)}/i;
let cssRulesText = [...styleText.matchAll(pattern4InnerText)].flat();
for (let rule of cssRulesText) {
let [selector, values] = rule.match(pattern4PerRule).slice(1);
// if (!selector || !values) throw Error(selector + 'someThing is Wrong');
selector = selector.trim();
values = values.split(';').filter(o => o.trim());
let style = result.get(selector);
if (!style) {
style = new Map();
result.set(selector, style);
}
for (let attr of values) {
let idx = attr.indexOf(":");
let k = attr.slice(0, idx).trim();
let v = attr.slice(idx + 1).trim();
if (!k) throw Error(selector + ' attr is Wrong');
if (v == '') continue;
style.set(k, v.replace('\uea60', ';'));
}
}
return result;
}
mapToString(map) {
if (Object.prototype.toString.call(map).slice(8, -1) !== 'Map') throw TypeError('map Must be Map');
let result = '';
for (let [selector, valMaps] of map) {
result = result + selector + '{';
for (let [k, v] of valMaps) {
result = result + k + ":" + v + ";";
}
result += "}";
}
return result
}
string2DToMap(styleText) {
let result = new Map();
if (!styleText) return result;
for (let item of styleText.split(';').filter(i => i.trim())) {
let key = item.slice(0, item.indexOf(':'));
let val = item.slice(item.indexOf(':') + 1);
result.set(key.trim(), val.trim());
}
return result;
}
getAllAvailableBg() {
//与项目的配置项耦合在一起了
let tempImgLists = [];
if (this.configs.bg.used) {
let bg = this.configs.bg.sources;
for (let modules of Object.values(bg)) {
if (modules.used) {
for (let item of modules.sources) {
item.url = item.url.trim();
if (item.used && item.url && (/^https?:/i).test(item.url)) {
tempImgLists.push(item);
}
}
}
}
}
return tempImgLists;
}
getSingleBg(availableList) {
// 为空的时候会返回undefined
if (window.isSecureContext && window.crypto && window.crypto.getRandomValues) {
return availableList[Math.floor(crypto.getRandomValues((new Uint32Array(1)))[0] / 0xFFFFFFFF * availableList.length)];
} else {
return availableList[Math.floor(Math.random() * availableList.length)];
}
}
getDifferPreviousSingleBg(previousBgObj, availableList) {
let curObj = this.getSingleBg(availableList);
for (; ;) {
if (previousBgObj === curObj) {
curObj = this.getSingleBg(availableList);
} else {
return curObj;
}
}
}
pageTest(urls = []) {
// 成功则返回该有的对象,否则就解决为null
if (!urls.length) return Promise.resolve(true);
let controllers = [], flag = true, errCounter = 0;
return new Promise(success => {
// 1、先检查本地
this.getAllCachesUrl(this.configs.cache).then(urlArrs => {
for (let item of urls) {
if (urlArrs.includes(item.url)) {
this.getCache(this.configs.cache, item.url)
.then(rsp => {
rsp.blob().then(bb => {
item.blobUrl = URL.createObjectURL(bb);
success(item);
flag = false;
for (let obj of controllers) {
obj.ctrl.abort();
}
logger.log('首页取到了缓存对象');
logger.dir(item);
})
})
logger.log('首页查询到了缓存对象');
break;
}
}
})
// 2、同时发送网络请求
for (let item of urls) {
let ctrl = new AbortController();
controllers.push({ ctrl, item });
fetch(item.url, { signal: ctrl.signal, method: 'head', priority: 'high' })
.then(rsp => {
if (rsp.status === 200 || rsp.status === 304) {
if (flag) { // 此处有bug
flag = false;
success(item);
logger.log(`首页成功拿到了head的响应`);
for (let obj of controllers) {
if (obj.item !== item) obj.ctrl.abort();
}
// 如果失败就算了,不再使用回调必须成功
this.addCaches(this.configs.cache, item.url)
}
// 其他成功就已经毫无意义了,不再计数
} else {
errCounter++
if (errCounter >= urls.length) success(null);
}
})
.catch((err) => {
errCounter++
logger.error(err.message);
if (errCounter >= urls.length) success(null);
})
}
})
}
getFirstPage(urlObjs = [], perNumber = 5) {
// 从数据库拿出固定的数据对象
if (!urlObjs.length) return Promise.resolve(null);
urlObjs = this.shuffle(urlObjs.slice());
let urlIterators = urlObjs[Symbol.iterator]();
let iteratorCounter = 0, flag = true;
return new Promise(success => {
this.pageTest((function () {
let tmpResult = [];
if (flag) {
for (let i = 0; i < perNumber; i++) {
if (++iteratorCounter === urlObjs.length) flag = false;
tmpResult.push(urlIterators.next().value);
}
}
return tmpResult.filter(o => o);
}())).then(result => {
logger.log(`首页一共发起${iteratorCounter}次head请求.`)
if (result) {
if (result === true) {
success(true);
} else {
logger.log('这里的是真对象');
logger.dir(result);
success(result);
}
} else {
return success(this.getFirstPage(urlObjs));
}
})
})
}
asyncLoadImg(bgObj, signal4Element, signal4Fetch, timeout4GM = 8000) {
// 只加载是403跨域不允许的资源,其他的都由浏览器同步请求;
let init = this.getCache(this.configs.cache, bgObj.url)
return init.then(result => {
// 1、查询缓存
if (result) {
logger.summary('Match Cache.');
clearTimeout(signal4Element.timer);
clearTimeout(signal4Fetch.timer);
return result.blob().then(bb => {
return URL.createObjectURL(bb);
})
} else {
logger.log('Start Request.');
return Promise.reject();
}
})
.then(result => result, () => new Promise((success, rej) => {
// 2、图像跨域测试并短时间获得缓存响应
let img = new Image();
img.onload = function () {
if (img.naturalHeight) {
bgObj.directLinkUsed = true; // 直接链接可用
success(bgObj.url);
// logger.summary(`正常请求${bgObj.url}`);
} else {
logger.warn(`img假成功-->[${bgObj.name}]--[${bgObj.url}]`);
rej();
}
clearTimeout(signal4Element.timer);
signal4Element.signal.onabort = null;
}
img.onerror = function () {
logger.warn('Wallpaper loading Error,NEXT STEP');
signal4Element.signal.onabort = null;
if (img.naturalHeight) {
success({ onlyBgAvailable: true, url: bgObj.url });
clearTimeout(signal4Fetch.timer);
logger.warn(`img假失败-->[${bgObj.name}]--[${bgObj.url}]`)
} else {
rej();
}
}
// img.crossOrigin = '';
img.src = bgObj.url;
signal4Element.signal.onabort = () => {
rej();
logger.warn('User timeout.');
img.onerror = null;
img.src = '';
img = null;
}
}))
.then(url => url, () => {
// 3、对可能失败的缓存响应中做处本地应急处理
return fetch(bgObj.url, { signal: signal4Fetch.signal })
.then(
rsp => { clearTimeout(signal4Fetch.timer); logger.log('Deal with response'); return rsp },
() => { return Promise.reject('Request Exception...') }
)
})
.then(rsp => {
// rsp 是来自于缓存的blob,或来自于fetch的rsp,或来自于img元素的url,或onlyBgAvailable对象;
if (rsp.onlyBgAvailable) return rsp.url;
if (rsp === bgObj.url) {
// 缓存url,但是存在一种情况,就是图片src被禁止,返回200,ok,fetch请求也会失败,
// 但是当图像作为背景却不会被禁止(远坂凛)
return new Promise((resolve, reject) => {
if (rsp.toLowerCase().startsWith('https:')) {
this.addCaches(this.configs.cache, bgObj.url, () => {
if (this.configs.requestAllWebsite) {
this.sendGMXHR(bgObj.url, null, rsp => {
// 对GM补救成功的同样进行blob化处理
if (/^image.*?\//i.test(rsp.response.type)) {
resolve(URL.createObjectURL(rsp.response));
let rspDup = new Response(rsp.response.slice(), { status: 200, statusText: 'OK', headers: this.getHeadersFromGMXHR(rsp.responseHeaders) });
this.setCaches(this.configs.cache, bgObj.url, rspDup);
} else {
logger.warn(`GM补救请求异常1,请及时检查[${bgObj.url}-->[${rsp.response}]-->[${rsp.response.type}]`);
if (bgObj.directLinkUsed) {
// 直链可用的话
resolve(bgObj.url);
logger.warn(`不可缓存1,但直链可用[${bgObj.url}]`)
} else {
reject('GM补救异常1');
}
}
}, function () {
reject('GM cross Origin Request failture.');
}, function () {
reject('GM timeout2');
}, timeout4GM);
} else {
console.warn('Maybe,,,,Please switch requestAllWebsite on.');
reject('GM add cache cant cors.');
}
}, () => {
// 成功,但是是第一次添加的回调
this.getCache(this.configs.cache, bgObj.url).then(myRsp => {
myRsp.blob().then(bb => {
logger.log('初次直接返回了blob');
resolve(URL.createObjectURL(bb));
})
})
})
} else {
resolve(rsp);
}
})
}
if (typeof rsp === 'string' && rsp.startsWith('blob:')) return rsp;
return new Promise((resolve, reject) => {
if (rsp.status === 403) {
logger.error('WebServer rejects request. ' + bgObj.url);
if (this.configs.requestAllWebsite) {
logger.warn('403,Start Cross Origin Request.');
this.sendGMXHR(bgObj.url, null, rsp => {
resolve(URL.createObjectURL(rsp.response));
let rspDup = new Response(rsp.response.slice(), { status: 200, statusText: 'OK', headers: this.getHeadersFromGMXHR(rsp.responseHeaders) });
this.setCaches(this.configs.cache, bgObj.url, rspDup);
}, function (rsp) {
logger.error('Cross Origin Request failture.');
reject(rsp)
}, function () {
reject('timeout');
}, timeout4GM);
} else {
console.warn(403, 'Maybe,,,,Please switch requestAllWebsite on.');
reject(rsp.status);
}
} else if (rsp.status > 399 && rsp.status !== 403) {
logger.error('Maybe internet or server inter error.')
reject(rsp.status);
} else {
// 几乎都是浏览器请求成功的rsp
try {
let rspDup = rsp.clone();
rsp.blob().then(bb => {
resolve(URL.createObjectURL(bb));
});
this.setCaches(this.configs.cache, bgObj.url, rspDup);
} catch (e) {
logger.error(`浏览器几乎请求成功处有异常-->[${e.message}]`);
resolve(bgObj.url);
}
}
});
}, reason => {
return new Promise((resolve, reject) => {
logger.warn(`BrowserInfo: [${reason}]`);
if (this.configs.requestAllWebsite) {
logger.warn('Start Cross Origin Request.');
this.sendGMXHR(bgObj.url, 'get', rsp => {
// 对GM补救成功的同样进行blob化处理
if (/^image.*?\//i.test(rsp.response.type)) {
resolve(URL.createObjectURL(rsp.response));
let rspDup = new Response(rsp.response.slice(), { status: 200, statusText: 'OK', headers: this.getHeadersFromGMXHR(rsp.responseHeaders) });
this.setCaches(this.configs.cache, bgObj.url, rspDup);
} else {
logger.warn(`GM补救请求异常2,请及时检查[${bgObj.url}-->[${rsp.response}]-->[${rsp.response.type}]`);
if (bgObj.directLinkUsed) {
// 直链可用的话
resolve(bgObj.url);
logger.warn(`不可缓存2,但直链可用[${bgObj.url}]`)
} else {
reject('GM补救异常2');
}
}
}, function (rsp) {
logger.error('Cross Origin Request failture.');
reject(rsp);
}, function () {
reject('GM timeout2');
}, 8000);
} else {
reject('cant cors.');
}
})
})
}
handleImg(context, timeout4Element, timeout4Fetch, timeout4GM) {
let bgObj = context.bgObjNext;
logger.log('下一张背景图像: ' + (bgObj.name || bgObj.url));
let aborter4Element = new AbortController(),
aborter4Fetch = new AbortController(),
timer4ele, timer4fet;
timer4ele = setTimeout(() => { aborter4Element.abort('UserTimeOutEle.'); }, (timeout4Element || this.configs.duration) * 1000 * 0.8);
timer4fet = setTimeout(() => { aborter4Fetch.abort('UserTimeOutDom.') }, (timeout4Fetch || this.configs.duration) * 1000 * 2 * 0.8);
this.asyncLoadImg(bgObj, { signal: aborter4Element.signal, timer: timer4ele }, { signal: aborter4Fetch.signal, timer: timer4fet }, timeout4GM)
.then(rsp => {
logger.summary('success get');
if (rsp.startsWith('data:')) {
context.bgStr = rsp.replace(';', '\uea60'); // 60000码点字符
} else {
context.bgStr = rsp;
}
bgObj.bgObjPrev = bgObj.bgObjNext;
context.lock = true;
})
.catch(reason => {
logger.warn(reason, '保留上一张');
bgObj.bgObjNext = bgObj.bgObjPrev;
context.lock = true;
});
}
wrapperEvent(customEvent, callback) {
// 用于自定义包装事件
if (typeof customEvent !== 'string') throw TypeError('EventName must be string.');
if (typeof callback !== 'function') throw TypeError('callback must be function.');
let customElement = new Event(customEvent, { cancelable: false, bubbles: true });
return function () {
customElement.content = arguments;
let result = callback.apply(this, [...arguments]);
window.dispatchEvent(customElement);
return result;
}
}
}
/* 定义样式在标签内处理的类 */
/* 此处的设置不会被网站设置覆盖的样式,为弱类型设置;全局唯一实例的style标签 */
class ModifyStyleFromTag extends ModifyStyleBase {
constructor() {
super();
if (!this.constructor.UNIQUEINSTANCE) {
this.observers = [];
this.styleEle = this.getNewElement('style');
this.styleEle.id = this.styleEle.id || '$' + Math.random().toString(36).slice(2);
Reflect.defineProperty(this, 'priority', { value: 'Tag', configurable: false, enumerable: true, writable: false });
Reflect.defineProperty(this.constructor, 'UNIQUEINSTANCE', { value: this, configurable: false, enumerable: false, writable: false });
} else {
return this.constructor.UNIQUEINSTANCE;
}
}
setStyle(css) {
// css 可以是样式对象或者是css文本;filter指定给满足条件的元素添加样式,可以是NodeList或[[HTMLElement]],或者回调函数;
// let cssTest = "div {width:30px ;height: 40px;}span a ,b> strong {color:'red';line-height:1rem}";
let newStyleMap = this.stringToMap(css);
let currentMap = this.stringToMap(this.styleEle.innerText);
for (let [selector, values] of newStyleMap) {
let currentStyleMap = currentMap.get(selector);
if (!currentStyleMap) {
// 空说明是新选择器;
currentMap.set(selector, values);
} else {
// 说明是旧选择器;
for (let [k, v] of values) {
currentStyleMap.set(k, v);
}
}
}
this.styleEle.innerText = this.mapToString(currentMap);
}
getStyle(selector, styleKey) {
// 返回一个样式字符串值
selector = selector.trim();
return this.stringToMap(this.styleEle.innerText).get(selector).get(styleKey);
}
getAllStyle() {
// 返回的是map对象;
return this.stringToMap(document.getElementById(this.styleEle.id).innerText);
}
removeStyle(selector, styleKeys = []) {
// styleKey应该是数组;eg,['width','height',...];
selector = selector.trim();
let currentMap = this.stringToMap(this.styleEle.innerText);
let flag = true;
if (!styleKeys.length) {
// 1 如果styleKey为空的话,则删除整个选择器
flag = currentMap.delete(selector);
} else {
// 2 非空的话则要每个处理
let currentStyleMap = currentMap.get(selector);
if (!currentStyleMap) return false;
for (let k of styleKeys) {
k = k.trim();
if (!currentStyleMap.delete(k)) logger.warn(`${selector}->${k} dont exist`);
}
if (!currentStyleMap.size) currentMap.delete(selector);
}
this.styleEle.innerText = this.mapToString(currentMap);
return flag;
}
observeStyle() {
// 监测要监视的style标签内容
this.observers[0] = new MutationObserver(() => {
if (!this.styleEle.isConnected) {
this.connect();
logger.warn('style tag is missing, Re-Connected ' + this.styleEle.id);
}
});
// this.observer.observe(document.head,{childList:true,subtree:true,characterData:true,attributeFilter:['class','style','id']});
if (document.head) {
this.observers[0].observe(document.head, { childList: true });
} else {
this.findElement('head').then(rsp => this.observers[0].observe(rsp[0], { childList: true }));
}
}
connect() {
if (!this.styleEle) throw Error('无样式对象');
if (!document.head) {
this.findElement('head')
.then(head => head[0].appendChild(this.styleEle))
.catch((e) => {
logger.warn('Tag cant insert to document.head immediately, info: ', e);
let timer = this.setBFInterval(() => {
let result = document.querySelectorAll('head');
if (result.length) {
result[0].appendChild(this.styleEle);
clearInterval(timer);
logger.log('Tag low performace insert head!')
clearTimeout(t4o);
}
}, 100);
let t4o = setTimeout(() => {
clearInterval(timer);
logger.error('head cant be found.');
}, 60 * 1000);
});
} else {
document.head.appendChild(this.styleEle);
}
}
run() {
this.connect();
this.observeStyle();
}
}
/* 定义样式在行内处理的类 */
/* 此处的样式设置可能会被网站设置覆盖的样式,为强类型设置 */
class ModifyStyleFromLine extends ModifyStyleBase {
constructor() {
super();
if (!this.constructor.UNIQUEINSTANCE) {
this.observers = null;
// 这是一个补救,用于针对固有样式进行保留(选择器:函数对象);
this.fixedStyleObj = null;
this.configs = null;
this.pEvents = {};
this.variableSelector = {}; // 此属性处理每次必须变更的对象;[]
Reflect.defineProperty(this, 'priority', { value: 'Line', configurable: false, enumerable: true, writable: false });
Reflect.defineProperty(this, 'styleMaps', { value: new Map(), configurable: false, enumerable: true, writable: false });
Reflect.defineProperty(this.constructor, 'UNIQUEINSTANCE', { value: this, configurable: false, enumerable: false, writable: false });
} else {
return this.constructor.UNIQUEINSTANCE;
}
}
getAllStyle() {
return this.styleMaps;
}
getStyle(selector, styleKey) {
// 返回一个样式字符串值
return this.styleMaps.get(selector.trim()).get(styleKey.trim());
}
setStyle(css) {
// css 可以是样式对象或者是css文本;filter指定给满足条件的元素添加样式,可以是NodeList或[[HTMLElement]],或者回调函数;
// 需要覆盖固有样式中希望被覆盖掉的;
// let cssTest = "div {width:30px ;height: 40px;}span a ,b> strong {color:'red';line-height:1rem}";
let newStyleMap = this.stringToMap(css);
let currentMap = this.styleMaps;
for (let [selector, values] of newStyleMap) {
let currentStyleMap = currentMap.get(selector);
if (!currentStyleMap) {
currentMap.set(selector, values);
} else {
for (let [k, v] of values) {
currentStyleMap.set(k, v);
}
}
}
}
updateGlobalStyle() {
for (let [selector, styleItems] of this.styleMaps) {
let nodeList = document.querySelectorAll(selector);
//logger.log(selector,'设置样式===>',nodeList);
let cssText = '';
for (let [k, v] of styleItems) {
cssText = cssText + k + ":" + v + ";";
}
// 处理每个元素固有样式中想被保留的;
for (let ele of nodeList) {
let cssTmp = cssText;
if (selector in this.fixedStyleObj) {
let filterStyleObj = this.fixedStyleObj[selector](ele);
cssTmp += filterStyleObj.cur;
let oldStyle = [...this.string2DToMap(ele.style.cssText).keys()];
oldStyle.sort();
let newStyle = [...this.string2DToMap(cssTmp + filterStyleObj.delay).keys()];
newStyle.sort();
if (newStyle.toString() != oldStyle.toString()) ele.style.cssText = cssTmp;
} else {
let oldStyle = [...this.string2DToMap(ele.style.cssText).keys()];
oldStyle.sort();
let newStyle = [...this.string2DToMap(cssTmp).keys()];
newStyle.sort();
if (newStyle.toString() != oldStyle.toString()) ele.style.cssText = cssTmp;
}
// 处理计算样式
if (Object.keys(this.variableSelector).includes(selector)) {
for (let stykey of this.variableSelector[selector]) {
// stykey 是每个需要计算和检查的stykey;
if (styleItems.get(stykey) !== this.string2DToMap(ele.style.cssText).get(stykey)) {
ele.style[stykey] = styleItems.get(stykey);
}
}
}
}
}
}
removeStyle(selector, styleKeys = []) {
// 调试用方法
// styleKey应该是数组;eg,['width','height',...];
let currentMap = this.styleMaps.get(selector);
if (!currentMap) return false;
for (let key of styleKeys) {
if (!currentMap.delete(key)) logger.log(key + ' is not exist.');
}
let nodeList = document.querySelectorAll(selector);
let cssText = '';
if (!currentMap.size) {
this.styleMaps.delete(selector);
} else {
for (let [k, v] of currentMap) {
cssText = cssText + k + " :" + v + ";";
}
}
// 合并固有样式(始终保留,不能移除的)
for (let ele of nodeList) {
let cssTmp = cssText;
if (selector in this.fixedStyleObj) {
cssTmp += this.fixedStyleObj[selector](ele).cur;
}
ele.style.cssText = cssTmp;
}
}
observeStyle() {
for (let item of this.observers) {
item.observer = new MutationObserver((_, obs) => {
// 主要是根据DOM变化确定每一次变化的时候被观察元素依然存在;
let flag = true;
let l = 1;
while (l--) {
// 初始化 空,需要赋值元素
if (!item.target) {
item.target = document.querySelector(item.selector);
if (!item.target) {
logger.error('变更.但是未查找到元素');
break;
}
this.updateGlobalStyle();
obs.observe(document.querySelector(item.eyeSlector), item.options);
logger.warn('变更,填充空target');
}
if (!item.target.isConnected) {
// 说明DOM变更,前后查找到的元素不一样,但是不影响,
item.target = document.querySelector(item.selector);
if (!item.target) break;
logger.warn('变更,更换旧target-修正');
// 这里还要重新监测observe
obs.observe(document.querySelector(item.eyeSlector), item.options);
this.updateGlobalStyle();
} else {
// 连接,但是变化太慢导致下次触发DOM变更以前为空,出现样式空白;
for (let sp of item.specialJudge) {
if (!item.target.style.cssText.includes(sp)) {
logger.warn(`变更,样式空白-修正`);
this.updateGlobalStyle();
break;
}
}
}
flag = false;
}
if (flag) {
// 为真说明元素丢失;设置定时器,直到查找到并重新赋值 观察 归位
logger.warn('暂时丢失,等待变更');
}
});
}
// 开始观察
for (let item of this.observers) {
let tmpNode = document.querySelector(item.eyeSlector);
if (!tmpNode) {
this.updateGlobalStyle();
this.findElement(item.eyeSlector).then(rsp => {
item.observer.observe(rsp[0], item.options);
// 找到后先执行一次添加样式,因为可能eyeselector要等很久才能变更节点
this.updateGlobalStyle();
}).catch(selectorAsync => logger.error(`异步查询,未找到:${selectorAsync}`));
} else {
item.observer.observe(tmpNode, item.options);
}
}
}
subcribeGlobal() {
//开启全局事件监听--私有事件另行监听
// 1、监听URL地址变化,地址变化就更新this.styleMaps里面的背景样式;
let imgsAvi = this.getAllAvailableBg(this.configs.bg);
let context = { lock: false, bgStr: '', bgObjNext: null, bgObjPrev: null };
let bgImgObj = this.getSingleBg(imgsAvi);
if (!bgImgObj) {
this.setStyle(`body{background-image:none;}`);
logger.log('当前无有效背景图像');
return;
} else {
this.getFirstPage(imgsAvi, 5).then(imgObj => {
if (imgObj !== true) {
context.bgObjPrev = imgObj;
if (!context.bgStr) context.bgStr = imgObj.blobUrl || imgObj.url;
logger.log('当前背景图像: ' + (imgObj.name || imgObj.url))
this.setStyle(`body{background-image:url("${imgObj.blobUrl || imgObj.url}");}`);
} else {
logger.log("所有在线请求失败,无有效图像.")
this.setStyle("body{background-image:none;}");
}
this.updateGlobalStyle();
});
}
if (this.configs.bgSwitchMode === 'multi' && imgsAvi.length > 1) {
history.pushState = this.wrapperEvent('pushstate', history.pushState);
let changeBg = (() => {
// 设定时间阀值,避免可能的重复设置背景
let timePre = new Date();
let timeCur = 0;
let pt = /^url\((['"]?)(blob[^"']*?)\1\)/i;
// 初始化第一次的图片
context.bgObjNext = this.getDifferPreviousSingleBg(bgImgObj, imgsAvi);
this.handleImg(context);
return function () {
timeCur = new Date();
if (!context.lock) return;
if (timeCur - timePre < this.configs.duration * 1000) return;
context.lock = false;
let bg = this.getStyle('body', 'background-image');
logger.log(`前一张:[${bg}]\n当前的:[url("${context.bgStr}")]`);
if (bg !== `url("${context.bgStr}")`) {
if (pt.test(bg)) {
URL.revokeObjectURL(bg.match(pt)[2]);
logger.summary(`已经释放:${bg}`);
}
this.setStyle(`body{background-image:url("${context.bgStr}");}`);
this.updateGlobalStyle();
logger.log('当前背景图像: ' + (context.bgObjNext.name || context.bgObjNext.url));
timePre = timeCur;
}
logger.log('全局调试对象$_SM:');
logger.dir($_SM);
// 准备下一次点击的图片;
context.bgObjNext = this.getDifferPreviousSingleBg(context.bgObjNext, imgsAvi);
this.handleImg(context);
};
})();
window.addEventListener('pushstate', changeBg.bind(this));
window.addEventListener('popstate', changeBg.bind(this));
}
}
subcribePrivate() {
//定义私有事件;
for (let key of Object.keys(this.pEvents)) {
this.pEvents[key].call(this);
}
}
run() {
this.subcribeGlobal();
this.subcribePrivate();
this.updateGlobalStyle();
this.observeStyle();
}
}
/* 此类用于实例化不同网站的设置 */
class Context extends Base {
constructor(tagRules, lineRules, adRules, fixedStyleObj, observers, configs, pEvents) {
super();
this.startUrl = location.host;
if (!(tagRules || lineRules || adRules)) throw TypeError('无效对象');
this.lineStyleSatndardRules = lineRules.standardStyle || '';
this.lineStyleVariableSelector = lineRules.variableStyleSelector || '';
this.tagStyleRules = tagRules || '';
this.adSoft = adRules.adSoft || [];
this.adHard = adRules.adHard || [];
this.adDynamic = adRules.adDynamic || [];
this.pEvents = pEvents || {};
this.fixedStyleObj = fixedStyleObj || {};
this.observers = observers || [];
this.configs = configs || userGlobalConfigs;
this.maid = [];
}
configInit() {
// 1、根据网速选择不同,重新包装findElement方法;
let curObj = Object.getPrototypeOf(this);
while (curObj) {
if (Object.prototype.hasOwnProperty.call(curObj, 'findElement')) {
let findElement = curObj.findElement;
curObj.findElement = (() => {
let timeout, fq, intvl;
switch (this.configs.netThrottlMode) {
// 'low','middle','heigh','infinity'
case 'low':
({ timeout, fq, intvl } = { timeout: 7, fq: true, intvl: 100 });
break;
case 'middle':
({ timeout, fq, intvl } = { timeout: 12, fq: true, intvl: 120 });
break;
case 'heigh':
({ timeout, fq, intvl } = { timeout: 20, fq: true, intvl: 300 });
break;
case 'infinity':
({ timeout, fq, intvl } = { timeout: 30, fq: true, intvl: 500 });
break;
default:
({ timeout, fq, intvl } = { timeout: 3, fq: false, intvl: 75 });
}
return function (selector, signal = null) {
return findElement.call(this, selector, timeout, fq, intvl, signal);
}
})();
break;
} else {
curObj = Object.getPrototypeOf(curObj);
}
}
// 2、移除禁用配置
if (Object.getPrototypeOf(this.configs) !== Object.prototype) {
for (let key of Reflect.ownKeys(this.configs)) {
if (['bg', 'isDebug', 'baidu', 'sou360'].includes(key)) {
delete this.configs[key];
logger.warn(`禁用用户 [${key}] 配置项.`)
}
}
}
// 3、添加用户不可见全局配置
this.configs.cache = 'myImgs';
// 修订限制性配置
this.configs.duration = this.configs.duration < 3 ? 3 : this.configs.duration;
}
globalTodo() {
//每个网站都要做的事情
// 广告软消除
window.addEventListener('load', () => {
this.adSoft.forEach(selector => { (new AntiAdSoft(selector.trim())).run(); })
});
document.addEventListener('abort', () => {
this.adSoft.forEach(selector => { (new AntiAdSoft(selector.trim())).run(); })
});
// 广告硬消除
for (let selector of this.adHard) {
(new AntiAdHard(selector.trim())).run();
}
}
go() {
// 此方法用于启动所有样式广告监听等
this.configInit();
let line = new ModifyStyleFromLine();
line.pEvents = this.pEvents;
line.configs = this.configs;
line.observers = this.observers;
line.fixedStyleObj = this.fixedStyleObj;
line.variableSelector = this.lineStyleVariableSelector;
let tag = new ModifyStyleFromTag();
// 动态广告置于此处
let adDy = new AntiAdDynamic(this.adDynamic);
line.setStyle(this.lineStyleSatndardRules); //冷处理
tag.setStyle(this.tagStyleRules);//冷处理
for (let item of [line, tag, adDy]) {
this.maid.push(item);
item.run();
}
this.globalTodo();
}
}
/* 此类用于生成图形化用户级配置面板 */
class ConfiguraturesByUserModify {
constructor() {
this.name = '设置面板';
this.panelId = '$2BeautyToMacConfigPanel';
}
getPanel() {
// 没有面板就创建面板,否则返回面板的引用;
let configPanel = document.getElementById(this.panelId);
if (!configPanel) {
configPanel = document.createElement('div');
configPanel.id = this.panelId;
configPanel.style.cssText = `position:fixed;right:48px;top:48px;z-index:5000;`;
let root = configPanel.attachShadow({ mode: 'open' });
root.innerHTML = `<style>
ol{
padding-left: 0px;
list-style-position: inside;
}
div.panel{
border: 1px gray solid;
border-radius: 12px;
padding: 16px;
min-width: 280px;
background-color: wheat;
position: relative;
}
h4.title{
font-size: large;
text-align: center;
margin: 12px auto;
}
.item-title{
font-size: 16px;
font-weight: bold;
}
.panel-item{
padding-bottom: 6px;
border-bottom: 1px black dotted;
margin-bottom: 5px;
}
.item-container{
margin-top: 8px;
}
.item-content{
display: inline-block;
margin-left: 12px;
}
input[type="color"],input[type="range"],input[type="number"]{
width: 110px;
}
/* close-logo处理 */
.close-logo{
position: absolute;
right: 12px;
top:12px;
cursor: pointer;
}
</style>
<div class="panel">
<h4 class='title'>BeautyToMac设置面板</h4>
<span class="close-logo" title="关闭" id="xclose">[<span style="color: red;">×</span>]</span>
<ol>
<li class="panel-item">
<span class="item-title">内容区域颜色:</span>
<div class="item">
<div class="item-container">
<label class="item-content">
<input type="color" value="#C7EDCC"><span class="item-content-commit">(点击取色)</span>
</label>
</div>
</div>
</li>
<li class="panel-item">
<span class="item-title">内容区域透明度:</span>
<div class="item">
<div class="item-container">
<label class="item-content">
<input type="range" name="opacity" max="100" min="0" value="80">
</label>
</div>
</div>
</li>
<li class="panel-item">
<span class="item-title">背景切换模式:</span>
<div class="item-container">
<label class="item-content">
<input type="radio" name="bgmode" value="轮播" checked><span class="item-content-commit">轮播</span>
</label>
<label class="item-content">
<input type="radio" name="bgmode" value="唯一"><span class="item-content-commit">唯一</span>
</label>
</div>
</li>
<li class="panel-item">
<span class="item-title">切换时间间隔:</span>
<div class="item-container">
<label class="item-content">
<input type="number" min="3" max="20" value="5" oninput="value=value.replace(/[^\d]/g,'');value<3?value=3:value>20?value=20:value=value;"><span class="item-content-commit">秒</span>
</label>
</div>
</li>
<li class="panel-item">
<span class="item-title">网络节流模式:</span>
<div class="item-container">
<label class="item-content"><input type="radio" name="ntm" value="server"><span class="item-content-commit">server</span></label>
<label class="item-content"><input type="radio" name="ntm" value="4/5G" checked><span class="item-content-commit">4/5G</span></label>
<label class="item-content"><input type="radio" name="ntm" value="3G"><span class="item-content-commit">3G</span></label>
<label class="item-content"><input type="radio" name="ntm" value="2G"><span class="item-content-commit">2G</span></label>
</div>
</li>
<li class="panel-item">
<span class="item-title">键盘左右导航:</span>
<div class="item-container">
<label class="item-content">
<input type="radio" name="kbn" value="开启" checked><span class="item-content-commit">开启</span>
</label>
<label class="item-content">
<input type="radio" name="kbn" value="关闭"><span class="item-content-commit">关闭</span>
</label>
</div>
</li>
<li class="panel-item">
<span class="item-title">空白左右导航:</span>
<div class="item-container">
<label class="item-content">
<input type="radio" name="wsn" value="开启" checked><span class="item-content-commit">开启</span>
</label>
<label class="item-content">
<input type="radio" name="wsn" value="关闭"><span class="item-content-commit">关闭</span>
</label>
</div>
</li>
<li class="panel-item">
<button class="item-title">背景图像管理>></button>
</li>
</ol>
</div>
`;
document.body.append(configPanel);
}
setTimeout(() => {
configPanel.shadowRoot.getElementById('xclose').addEventListener('click', function (e) {
e.preventDefault();
configPanel.remove();
configPanel = null;
})
})
return configPanel;
}
}
let userPrivateConfigs = Object.create(userGlobalConfigs);
let tagStyle, fixedStyleObj, lineStyle, adRules, obs4DOM, pEvents;
if (userGlobalConfigs.baidu && location.href.match(/^https?:\/\/w{0,3}\.baidu\.com/i) && ['/', '/s', '/more/'].includes(location.pathname.toLowerCase())) {
// 百度私有用户配置
Object.assign(userPrivateConfigs, {
// 此处书写可以覆盖全局的配置
netThrottlMode: 'low',
});
// 1、Tag 配置
// #head 用于兼容谷歌浏览器渲染过慢导致的空白显示问题,firefox可以移除此项;
tagStyle = `
#s_top_wrap{
background-color: #ffffff53;
backdrop-filter: blur(2px);
}
#s-top-more{
background-color:#ffffff4d;
}
#head{
background-color:rgba(255, 255, 255, 0.3)!important;box-shadow:rgba(0, 0, 0, 0.5) 2px 0px 5px 2px;backdrop-filter:blur(2px);
}
#wrapper_wrapper{
cursor:pointer;
}
#content_left{
margin-left: unset !important;
padding-left: unset !important;
}
#s_tab>div.s_tab_inner:hover{
opacity:0.8;
}
#s_tab>div.s_tab_inner{
opacity:0;
}
#head_wrapper input#kw,#form .bdsug,#form .s_ipt_wr{
background-color:#ffffffa0;
}
#form .s_ipt_wr,#kw{
border-top-left-radius: 1rem;
border-bottom-left-radius: 1rem;
}
#form .bdsug li:hover{
background-color:#ffffffa0;
}
#foot div.foot-inner {
background-color:#fff0 !important;
}
.wrapper_new #head.peak-down.s_down{
background-color:#fff0 !important;
}
#wrapper #s_tab div a,wrapper #s_tab div .s-tab-item::before,#wrapper #s_tab div b,#wrapper #s_tab div .s-tab-item::before,#wrapper #s_tab div .cur-tab::before{
color:black !important;
}
#wrapper #s_tab div a:hover,wrapper #s_tab div .s-tab-item::before:hover,#wrapper #s_tab div b:hover,#wrapper #s_tab div .s-tab-item::before:hover,#wrapper #s_tab div .cur-tab::before:hover{
color:blue !important;
}
.foot{
background-color:#ffffff4d;
}
#foot{
opacity:0 !important;
}
#head_wrapper #s-hotsearch-wrapper,div#bottom_layer,div#s_side_wrapper,#head_wrapper #m,#searchTag{
display:none !important;
}
#head_wrapper.s-ps-islite{
padding-bottom:0px !important;
}
#s_tab a, #s_tab b{
margin-right:0px !important;
}
`;
// 2、Line 配置 --- 固有样式 ---
fixedStyleObj = {
"#head": function (ele) {
let urlPt = new URL(ele.baseURI);
if (urlPt.pathname === '/' || (urlPt.pathname === '/s' && (!urlPt.search))) {
return { cur: 'background-color:unset!important;backdrop-filter:none;', delay: '' }
} else {
return { cur: '', delay: '' };
}
},
// 侧边栏居中
"#s_tab>div.s_tab_inner": function (ele) {
setTimeout(() => {
ele.style.cssText += `margin-top:${ele.getBoundingClientRect().height / -2}px;`;
});
setTimeout(() => {
ele.style.opacity = null;
ele.style.left = '0px';
}, 50);
return { cur: 'opacity:0;', delay: `margin-top:${ele.getBoundingClientRect().height / -2}px;left:0px;` };
},
};
// 3、Line 配置 --- 标准样式 ---
lineStyle = {
// #page bottom:-100px;用于解决底部栏闪烁问题;在固有样式中异步恢复
// #s_tab>div.s_tab_inner left:-100px;用于解决左侧栏闪烁问题;在固有样式中异步恢复
standardStyle: `
body{
background-repeat:no-repeat;background-position:center;background-attachment:fixed;background-size:cover;
}
#page{
background-color:#ffffff4d;box-shadow:rgba(0, 0, 0, 0.5) 0px -1px 5px 2px;backdrop-filter:blur(2px);
position:fixed;bottom:0px;left:50%;border-radius:12px;z-index:1400;transform:translateX(-50%);margin-top: 0px;
}
#foot{
background-color:#ffffff00;
}
#page>div{
padding:14px;
width:min-content;
}
#page>div>*:last-child{
margin-right:0px;
}
#container{
background-image:linear-gradient(to right,rgba(255, 255, 255, 0.75) 25%,rgba(255, 255, 255, 0.15));box-sizing:border-box;cursor:default;
box-shadow:rgba(0, 0, 0, 0.5) 0px 0px 5px 2px,inset 0px 0px 5px 1px white;margin-bottom:0.5rem;padding:2.5em;border-radius:12px;
}
#head{
background-color:rgba(255, 255, 255, 0.3)!important;box-shadow:rgba(0, 0, 0, 0.5) 2px 0px 5px 2px;backdrop-filter:blur(2px);
}
#s_tab>div.s_tab_inner{
position:fixed;left:-100px;z-index:302;top:50%;border-top-right-radius:16px;border-bottom-right-radius:16px;width:60px;
transition:all 0.3s;background-color:#fff;
}
`,
// Line 配置 --- 计算样式(选择) ---
variableStyleSelector: {
'body': ['background-image'],
}
};
// 4、Ad 配置
adRules = {
// 4、Ad (soft)配置
adSoft: [
'.s-hotsearch-wrapper.s-isindex-wrap', '.s-isindex-wrap.s-bottom-layer', '[tpl=recommend_list]',
'#rs_new', '#s_side_wrapper', '#content_right div.hint_right_middle', '#help', '#searchTag',
],
// 4.1 Ad (hard配置)
adHard: [],
adDynamic: [],
};
// 5、监测配置(Line)
obs4DOM = [
{ 'selector': '#head', "target": null, 'eyeSlector': 'body', 'observer': null, 'options': { childList: true }, specialJudge: ['background-color'] },
//{'selector':'#s_tab>div.s_tab_inner',"target":null,'eyeSlector':'#s_tab','observer':null,'options':{childList:true},specialJudge:['opacity']},
//{'selector':'#wrapper',"target":null,'eyeSlector':'body','observer':null,'options':{childList:true},specialJudge:[]},
{ 'selector': '#page', "target": null, 'eyeSlector': '#wrapper_wrapper', 'observer': null, 'options': { childList: true, subtree: true }, specialJudge: [] },
];
// 6、私有事件配置(写在外部便于配置)
pEvents = {
'#container': function () {
// 设置背景
if (this.configs.centerColor) this.setStyle(`#container{background-color:${this.configs.centerColor.trim()};}`);
},
'#page .n': function () {
// 设置键盘左右建翻页导航
let that = this;
function navi() {
if (that.configs.keyBoardNav) {
document.addEventListener('keydown', function (e) {
if (!(e.key === 'ArrowLeft' || e.key === 'ArrowRight')) return;
if (document.activeElement.tagName.toUpperCase() === 'INPUT') return;
let nodes = document.querySelectorAll('#page .n');
if (nodes.length && e.key === 'ArrowLeft') {
for (let item of nodes) {
if (item.textContent.split('').includes('上')) item.click();
}
} else if (nodes.length && e.key === 'ArrowRight') {
for (let item of nodes) {
if (item.textContent.split('').includes('下')) item.click();
}
} else {
logger.log('nodes条件不足,略过本次')
}
})
}
}
window.addEventListener('load', navi);
document.addEventListener('abort', navi);
},
'#wrapper_wrapper': function () {
// 设置左右点击空白翻页导航
let that = this;
function navi() {
if (that.configs.whiteNav) {
document.addEventListener('click', function (e) {
let targetFlag = 'wrapper_wrapper', selectorFlag = '#page .n';
let targetNode = document.getElementById(targetFlag);
let nodes = document.querySelectorAll(selectorFlag);
if (e.target === targetNode && nodes.length) {
if (e.clientX < parseInt(getComputedStyle(targetNode).width) / 2) {
for (let item of nodes) {
if (item.textContent.split('').includes('上')) item.click();
}
} else {
for (let item of nodes) {
if (item.textContent.split('').includes('下')) item.click();
}
}
}
// else{
// logger.warn('点击导航异常');
// logger.dir(targetNode,nodes.length,e.target);
// }
})
}
}
window.addEventListener('DOMContentLoaded', navi);
document.addEventListener('abort', navi);
},
'quickAccessSearchBox': function () {
// 快捷键 ctrl + y 搜索框聚焦
if (this.configs.allowSearchBox) {
document.addEventListener('keydown', function (e) {
if (e.key.toLowerCase() === 'y' && e.ctrlKey) {
e.preventDefault();
let kw = document.querySelector('#kw');
kw ? kw.focus() : console.log('SearchAccess cant found searchBox');
}
})
}
}
};
} else if (userGlobalConfigs.baidu && location.href.match(/^https?:\/\/w{0,3}\.baidu\.com/i) && location.pathname.toLowerCase() === '/sf/vsearch') {
// 百度视频页面
Object.assign(userPrivateConfigs, {
// 此处书写可以覆盖全局的配置
netThrottlMode: 'low',
});
// 1、Tag 配置
tagStyle = `
#s_tab>div.s_tab_inner:hover{
opacity:0.8;
}
#s_tab>div.s_tab_inner{
opacity:0;
}
#wrapper #s_tab div a,wrapper #s_tab div .s-tab-item::before,#wrapper #s_tab div b,#wrapper #s_tab div .s-tab-item::before,#wrapper #s_tab div .cur-tab::before{
color:black !important;
}
#wrapper #s_tab div a:hover,wrapper #s_tab div .s-tab-item::before:hover,#wrapper #s_tab div b:hover,#wrapper #s_tab div .s-tab-item::before:hover,#wrapper #s_tab div .cur-tab::before:hover{
color:blue !important;
}
#s_tab a, #s_tab b{
margin-right:0px !important;
}
`;
fixedStyleObj = {
"#s_tab>div.s_tab_inner": function (ele) {
setTimeout(() => {
ele.style.cssText += `margin-top:${ele.getBoundingClientRect().height / -2}px;`;
});
setTimeout(() => {
ele.style.opacity = null;
ele.style.left = '0px';
}, 50);
return { cur: 'opacity:0;', delay: `margin-top:${ele.getBoundingClientRect().height / -2}px;left:0px;` };
},
};
lineStyle = {
standardStyle: `
body{
background-repeat:no-repeat;background-position:center;background-attachment:fixed;background-size:cover;
}
#s_kw_wrap{
background-color: #ffffffad;
}
#container {
background-image: linear-gradient(
to right,
rgba(255, 255, 255, 0.75) 25%,
rgba(255, 255, 255, 0.15)
);
box-sizing: border-box;
box-shadow: rgba(0, 0, 0, 0.5) 0px 0px 5px 2px, white 0px 0px 5px 1px inset;
padding: 2.5em;
border-radius: 12px;
display: flex;
flex-direction: row-reverse;
width: min-content;
margin: auto;
}
#content_left {
padding-right: 140px;
padding-left: 0px;
}
#head{
background-color:rgba(255, 255, 255, 0.3)!important;box-shadow:rgba(0, 0, 0, 0.5) 2px 0px 5px 2px;backdrop-filter:blur(2px);
}
#s_tab>div.s_tab_inner{
position:fixed;left:-100px;z-index:302;top:50%;border-top-right-radius:16px;border-bottom-right-radius:16px;width:60px;
transition:all 0.3s;background-color:#fff;
}
.s_side_wrapper{
display: none;
}`,
variableStyleSelector: {},
};
adRules = {
// Ad (soft)配置
adSoft: [],
// Ad 硬处理(只能删除节点的类型);给选择器即可
adHard: [],
// Ad 专用动态规则集
adDynamic: []
};
obs4DOM = [
{ 'selector': '#head', "target": null, 'eyeSlector': 'body', 'observer': null, 'options': { childList: true }, specialJudge: ['background-color'] },
];
pEvents = {
'#container': function () {
if (this.configs.centerColor) this.setStyle(`#container{background-color:${this.configs.centerColor.trim()};}`);
},
};
} else if (userGlobalConfigs.sou360 && location.href.match(/^https?:\/\/w{0,3}\.so\.com/i) && location.pathname.toLowerCase() === '/') {
//360首页
Object.assign(userPrivateConfigs, {
// 此处书写可以覆盖全局的配置
netThrottlMode: 'low',
});
// 1、Tag 配置
tagStyle = `
#footer{
display:none;
}
#header{
background-color:#ffffff6b;
backdrop-filter:blur(2px);
}
fieldset#input-container{
background-color:rgba(255, 255, 255, 0.75);
}
div#suggest-align{
background-color:#ffffff00;
}
div#goto-top,#main .gold-wrap{
display:none!important;
}
`;
// 2、Line 配置 --- 固有样式 ---
fixedStyleObj = {
//selector:function(ele){}
};
// 3、Line 配置 --- 标准样式 ---
lineStyle = {
standardStyle: `
body{
background-repeat:no-repeat;background-position:center;background-attachment:fixed;background-size:cover;
}
#skin_bg{
background-color:rgba(255, 255, 255, 0);
}
#card_container{
display:none;
}
#bd_search .fixed{
background-color:rgba(255, 255, 255, 0);
}
`,
// Line 配置 --- 计算样式(选择) ---
variableStyleSelector: {
'#skin_bg': ['background-color'],
// 都要有的背景
'body': ['background-image'],
}
}
// 4、Ad 配置
adRules = {
// Ad (soft)配置
adSoft: [],
// Ad 硬处理(只能删除节点的类型);给选择器即可
adHard: [
'#goto-top', '#often_so'
],
// Ad 专用动态规则集
adDynamic: [
//{ref:'#lawnfooter-samll__btne',fire:(eles)=>{return [eles[0]]},options:{childList:true}},
{
ref: '#lawnfooter-samll__btn', eyeSlector: 'body', options: { childList: true }, fire: (eles) => {
//eles 是重找规则里面的元素集
let pE = eles[0];
for (; ;) {
if (pE.parentElement === document.body) {
return [pE];
} else {
pE = pE.parentElement;
}
}
},
},
],
}
// 5、监测配置(Line)
obs4DOM = [
{ 'selector': '#card_container', "target": null, 'eyeSlector': '#main', 'observer': null, 'options': { childList: true }, specialJudge: ['display'] },
];
pEvents = {};
} else if (userGlobalConfigs.sou360 && location.href.match(/^https?:\/\/w{0,3}\.so\.com/i) && location.pathname.toLowerCase() === '/s') {
//360搜索结果页面
Object.assign(userPrivateConfigs, {
// 此处书写可以覆盖全局的配置
netThrottlMode: 'low',
});
// 1、Tag 配置
tagStyle = `
html{
background-color: unset !important;
}
div#header{
background-color: transparent;
}
#header div.inner{
background: #ffffff70;
box-shadow: black 0px 0px 5px 1px;
backdrop-filter: blur(2px);
}
ul.g-menu{
background-color: #ffffffe8;
}
#g-hd #head .round{
background-color: #ffffffb0;
}
#head #keyword{
background-color: #fff0;
}
div#tabs-wrap{
border:none;
}
ul#g-hd-tabs{
position: fixed;
width: min-content;
left: 0px;
top:50%;
height:min-content;
transform:translateY(-50%);
background-color: #eee;
z-index:3000;
border-top-right-radius:16px;
border-bottom-right-radius:16px;
opacity:0;
transition:opacity 0.3s;
}
ul#g-hd-tabs:hover{
opacity:0.85;
}
#tabs-wrap ul#g-hd-tabs li.g-hd-cur a {
color: #0fb264 !important;
}
#tabs-wrap ul#g-hd-tabs li a{
color:black !important;
}
div#container{
margin: 0px auto 6rem;
padding: 2rem;
border-radius: 12px;
background-image: linear-gradient(to right,rgba(255, 255, 255, 0.75) 25%,rgba(255, 255, 255, 0.15));
box-shadow: rgba(0, 0, 0, 0.5) 0px 0px 5px 2px,inset 0px 0px 5px 1px white;
border-radius: 12px;
display:-webkit-flex;
display:-moz-flex;
display:flex;
justify-content:center;
width:min-content;
cursor:default;
}
div#warper{
cursor:pointer;
}
div#main{
padding-left: 2rem;
margin-right:6%;
box-sizing: border-box;
min-width:608px;
}
div#container>div:last-child{
display:none!important;
}
#warper div.mod-relation,#main .inline-recommend{
display: none !important;
}
#warper #page{
position: fixed;
bottom: 0px;
margin: 0px;
left:0px;
right:0px;
z-index: 3000;
padding-left: 10px;
background-color: #ffffff4d;
box-shadow: rgba(0, 0, 0, 0.5) 0px -1px 5px 2px;
backdrop-filter: blur(2px);
border-radius: 10px;
padding-top: 14px;
padding-bottom: 14px;
margin-left: auto;
margin-right: auto;
width: min-content;
}
#page span.nums{
display: none !important;
}
div#page a, div#page strong{
background: #fff;
border: 1px solid #eee;
border-radius: 8px;
color: #666;
display: inline-block;
font-size: 14px;
height: 34px;
line-height: 34px;
margin-right: 12px;
text-align: center;
text-decoration: none;
vertical-align: middle;
width: 34px;
}
div#page strong{
background: #0fb264;
border-color: #0fb264;
box-shadow: 0 1px 3px rgba(188,188,188,0.2);
color: #ffffff;
}
div#footer{
display:none;
}
#warper #side{
left:0px;
box-sizing:border-box;
padding-right: 1rem;
min-width: 400px;
}
#mohe-hotnews_right .mh-small-box,#side #adwarnTip,#side #so_kw-ad,#side #lm-rightbottom,div.lianmeng-ad,dl#head_rs_top,#rs-top,.kzx-news-rec-info,
.lianmeng-ad,#side div.res-mediav-right,dl#head_pdr_guide,div#so_top,div.res-recommend-tag,.double-eleven,div#goto-top{
display:none!important;
}
#warper #side:hover{
background-color:unset!important;
}
div#side_wrap.fixed{
background-color:rgba(255, 255, 255, 0)!important;
}
`;
/*
.so-rich-official-recom .inter .inter-ul{
width:auto !important;
}
*/
// 2、Line 配置 --- 固有样式 ---
fixedStyleObj = {
// 左侧面板居中
// 底部栏
};
// 3、Line 配置 --- 标准样式 ---
lineStyle = {
standardStyle: `
body{
background-repeat:no-repeat;background-position:center;background-attachment:fixed;background-size:cover;
}
`,
// Line 配置 --- 计算样式(选择) ---
variableStyleSelector: {
// 都要有的背景
'body': ['background-image'],
}
}
// 4、Ad 配置
adRules = {
// Ad (soft)配置
adSoft: [],
// Ad 硬处理
adHard: [
'#so_kw-ad'
],
// Ad 专用动态规则集
adDynamic: [
{ ref: '#so_top', eyeSlector: 'body', options: { childList: true, subtree: true }, fire: (eles) => { return [...eles] } },
{ ref: '#head_pdr_guide', eyeSlector: '#tabs-wrap', options: { childList: true, subtree: true }, fire: (eles) => { return [...eles] } },
],
}
// 5、监测配置(Line)
obs4DOM = [
{ 'selector': 'body', "target": null, 'eyeSlector': 'body', 'observer': null, 'options': { attributeFilter: ['style'] }, specialJudge: ['backgound-image'] },
{ 'selector': 'div#container', "target": null, 'eyeSlector': 'div#warper', 'observer': null, 'options': { childList: true }, specialJudge: ['background-color'] },
];
pEvents = {
'div#container': function () {
// 添加中央内容区护眼色等。
if (this.configs.centerColor) this.setStyle(`div#container{background-color:${this.configs.centerColor.trim()};}`);
},
'#spre#snext': function () {
// 给键盘添加左右导航
if (this.configs.keyBoardNav) {
document.addEventListener('keydown', e => {
if ((e.key === 'ArrowLeft' || e.key === 'ArrowRight') && document.activeElement.tagName.toUpperCase() !== 'INPUT') {
if (e.key === 'ArrowLeft') {
let ele = document.getElementById('spre');
ele && ele.click();
} else {
let ele = document.getElementById('snext');
ele && ele.click();
}
}
});
}
},
'#warper': function () {
// 添加左右空白导航
function nav() {
document.addEventListener('click', function (e) {
if (e.target.id === 'warper') {
if (e.clientX < parseInt(getComputedStyle(e.target).width) / 2) {
let ele = document.getElementById('spre');
ele && ele.click();
} else {
let ele = document.getElementById('snext');
ele && ele.click();
}
}
});
}
if (this.configs.whiteNav) {
window.addEventListener('DOMContentLoaded', nav);
document.addEventListener('abort', nav);
}
},
'quickAccessSearchBox': function () {
// 快捷键 ctrl + y 搜索框聚焦
if (this.configs.allowSearchBox) {
document.addEventListener('keydown', function (e) {
if (e.key.toLowerCase() === 'y' && e.ctrlKey) {
e.preventDefault();
let kw = document.querySelector('#keyword');
kw ? kw.focus() : console.log('SearchAccess cant found searchBox');
}
})
}
}
};
} else {
console.warn('Some Website cant match.');
return;
}
function check(it) {
// Math is known to exist as a global in every environment.
return it && it.Math === Math && it;
}
const globalObject =
check(typeof window === "object" && window) ||
check(typeof self === "object" && self) ||
check(typeof global === "object" && global) ||
check(typeof globalThis === 'object' && globalThis) ||
// This returns undefined when running in strict mode
(function () {
return this;
})() ||
Function("return this")();
globalObject.$_SM = new Context(tagStyle, lineStyle, adRules, fixedStyleObj, obs4DOM, userPrivateConfigs, pEvents);
globalObject.$_SM.go();
logger.log('running');
let userUISetting = new ConfiguraturesByUserModify;
// eslint-disable-next-line no-undef
GM_registerMenuCommand('打开配置面板', userUISetting.getPanel.bind(userUISetting));
})()