// ==UserScript==
// @name 收藏控 · 链接清洗
// @version 1.2.6
// @author 极品小猫
// @description 把链接地址缩减至最短可用状态,并复制到剪切板,以方便分享。【在每个页面的底部中间,有一个小小的按钮,用来呼出面板】
// @namespace https://gf.qytechs.cn/users/3128
// @homepage https://gf.qytechs.cn/users/3128
// @icon https://s1.ax1x.com/2020/08/13/dSUEsU.png
// @supportURL
// @updateURL
// @downloadURL
// @include *://*/*
// @require https://cdn.staticfile.org/jquery/2.1.4/jquery.min.js
// @grant GM_setClipboard
// @grant GM_notification
// @grant GM_addStyle
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_registerMenuCommand
// @grant GM_info
// @license MIT
// @noframes
// ==/UserScript==
/****
*
2021-11-13
1、改进 Param 参数获取通过 URLSearchParams 获得
2、改进 Rules 规则表,设定保留参数
3、改进 自定义规则生成逻辑,应用保留参数设定
4、改进 自定义规则配置应用方法,通过 URLSearchParams 函数删除参数
*
*/
/** 主功能函数 **/
(function(){
let webUrl=location.href,
webOrigin=location.origin;
webHost=location.host.toLowerCase(),
webSearch=location.search,
webPathname=location.pathname;
console.log(GM_info);
const rules = { //预定义规则
'mp.weixin.qq.com':{
//testReg: /^http(?:s)?:\/\/.+/i, // //'', /s?__biz=MzUxMjMxOTE0Nw==&mid=2247531927&idx=1&sn=ba1327de710ee8116f567f5d20f393cc
//replace: '',
reserveQuery: ['biz','mid','idx','sn'], //保留的参数
hash: false
},
'www.bilibili.com': {/* Blibili */
testReg: /^http(?:s)?:\/\/www\.bilibili\.com\/video\/(av\d+).*$/i,
replace: 'https://www.bilibili.com/$1',
reserveQuery: ['p'],
hash: true
},
'itunes.apple.com': {/* Apple Stroe */
testReg: /^http(?:s)?:\/\/itunes\.apple\.com\/(?:\w{2}\/)?([^\/]+)\/(?:[^\/]+\/)?((?:id)\d+).*$/i,
replace: 'https://itunes.apple.com/cn/$1/$2',
},
'chrome.google.com/webstore': {/* Chrome Store */
testReg: /^http(?:s)?:\/\/chrome\.google\.com\/webstore\/detail\/[^\/]+\/([a-z]{32}).*/i,
replace: 'https://chrome.google.com/webstore/detail/$1',
},
's.taobao.com': {/* Taobao Search */
testReg: /^http(?:s)?:\/\/s\.taobao\.com\/search.*$/i,
replace: 'https://s.taobao.com/search',
reserveQuery: ['q'],
},
'list.tmall.com': {/* Tmall Search */
testReg: /^http(?:s)?:\/\/list\.tmall\.com\/search_product\.htm.*$/i,
replace: 'https://list.tmall.com/search_product.htm',
reserveQuery: ['q'],
},
'item.taobao.com': {/* Taobao item */
testReg: /^http(?:s)?:\/\/item\.taobao\.com\/item\.htm.*$/i,
replace: 'https://item.taobao.com/item.htm',
reserveQuery: ['id'],
},
'detail.tmall.com': {/* Tmall item */
testReg: /^http(?:s)?:\/\/detail\.tmall\.com\/item\.htm.*$/i,
replace: 'https://detail.tmall.com/item.htm',
reserveQuery: ['id'],
},
'taobao/tmall.com/shop': {/* Taobao/Tmall Shop */
testReg: /^http(?:s)?:\/\/(\w+)\.(taobao|tmall)\.com\/shop\/view_shop\.htm.*$/i,
replace: 'https://$1.$2.com/',
},
'c.pc.qq.com': {/* Open Taobao share link from QQ */
testReg: /^http(?:s)?:\/\/c\.pc\.qq\.com\/middle.html\?.*pfurl=([^&]*)(?:&.*$|$)/i,
replace: '$1',
reserveQuery: [],
methods: ['decodeUrl'],
},
'item.m.jd.com': {/* JD mobile to PC */
testReg: /^http(?:s)?:\/\/item\.m\.jd\.com\/product\/(\d+)\.html(\?.*)?$/i,
replace: 'https://item.jd.com/$1.html',
},
'item.m.jd.com/ware/': {/* JD mobile to PC */
testReg: /^http(?:s)?:\/\/item\.m\.jd\.com\/ware\/view\.action\?.*wareId=(\d+).*$/i,
replace: 'https://item.jd.com/$1.html',
},
'search.jd.com': {/* JD Search */
testReg: /^http(?:s)?:\/\/search\.jd\.com\/Search\?.*$/i,
reserveQuery: ['keyword', 'enc'],
},
're.jd.com': {/* JD hot sell */
testReg: /^http(?:s)?:\/\/re\.jd\.com\/cps\/item\/(\d+)\.html.*$/i,
replace: 'https://item.jd.com/$1.html',
},
'weibo.com/u': {/* Weibo personal homepage to mobile */
testReg: /^http(?:s)?:\/\/(?:www\.)?weibo\.com\/u\/(\d+)(\?.*)?$/i,
replace: 'https://m.weibo.cn/$1',
},
'weibo.com': {/* Weibo article page to mobile */
testReg: /^http(?:s)?:\/\/(?:www\.)?weibo\.com\/(?:\d+)\/(\w+)(\?.*)?$/i,
replace: 'https://m.weibo.cn/status/$1',
},
'gf.qytechs.cn': {/* Greasyfork Script */
testReg: /^http(?:s)?:\/\/(?:www\.)?greasyfork\.org\/(?:[\w-]*\/)?scripts\/(\d+)-.*$/i,
replace: 'https://gf.qytechs.cn/zh-CN/scripts/$1',
},
'store.steampowered.com|steamcommunity.com': {/* Steam */
testReg: /^http(?:s)?:\/\/(store\.steampowered|steamcommunity)\.com\/app\/(\d+).*$/i,
replace: 'https://$1.com/app/$2',
},
'meta.appinn.com': {/* Appinn BBS */
testReg: /^http(?:s)?:\/\/meta\.appinn\.net\/t(?:\/[^/]*)*?\/(\d+)(\/.*$|$)/i,
replace: 'https://meta.appinn.net/t/$1',
},
'amazon.co.jp': {/* amazon.co.jp */
testReg: /^http(?:s)?:\/\/(?:www\.)?amazon\.co\.jp\/([^\/]+)\/dp\/(\w+)\/.*$/i,
replace: 'https://www.amazon.co.jp/$1/dp/$2',
},
'yangkeduo.com': {/* Pin Duo Duo product Page */
testReg: /^http(?:s)?:\/\/mobile\.yangkeduo\.com\/goods.html\?.*$/i,
reserveQuery: ['goods_id'],
},
'other': {/* All url */
testReg: /^(http(?:s)?:\/\/[^?#]*)[?#].*$/i,
reserveQuery: ['id', 'tid', 'uid', 'q', 'wd', 'query', 'keyword'],
}
}
/** 必须函数 */
let localConfig=localStorage['Cat_LC_Config']?JSON.parse(localStorage['Cat_LC_Config']):{"param":[], testReg:'', OpenMode:'history'};
const LinkCleaner={
webRule : rules[webHost],
theUrlParam : new URLSearchParams(webSearch),
theUrlParamKeys : [],
reserveQueryJSON : {},
param_Count : function(search, len){ //分割参数,获取参数key
let Params=new URLSearchParams(search||location.search),
theUrlParam=[];
for(let key of Params.keys()) {
theUrlParam.push(key);
}
//const theUrlParam=search.replace(/^\?/,'').split('&');
if(len) return theUrlParam.length;
return theUrlParam;
},
PureUrl : function(url){
switch(localConfig.OpenMode) {
case 'location':
if(window.location.href !== url) window.location.href = url;
default:
case 'history':
history.pushState('', '', url);
break;
}
},
Cat_get_pure_url : function(url){
url = url||window.location.href;
const hash = url.replace(/^[^#]*(#.*)?$/, '$1')
const base = url.replace(/(\?|#).*$/, '')
let pureUrl = url
const getQueryString = function(key) {
let ret = url.match(new RegExp('(?:\\?|&)(' + key + '=[^?#&]*)', 'i'))
return ret === null ? '' : ret[1]
}
/* 链接处理方法 */
const methods = {
decodeUrl: function(url){return decodeURIComponent(url) }
}
for(let i in rules){
let rule = rules[i]
let reg = rule.testReg
let replace = rule.replace
if (reg.test(url)){
let newQuerys = ''
if(typeof(rule.reserveQuery)!=='undefined' && rule.reserveQuery.length>0){
rule.reserveQuery.map((reserveQuery) => {
const ret = getQueryString(reserveQuery)
if(ret !== '') newQuerys += (newQuerys.length ? '&' : '?') + ret;
})
}
newQuerys += typeof(rule.hash)!=='undefined' && rule.hash ? hash : '';
pureUrl = (typeof(replace)==='undefined'?base:url.replace(reg, replace) ) + newQuerys;
if(typeof(rule.methods)!=='undefined' && rule.methods.length>0){
rule.methods.map((methodName)=>{
pureUrl = methods[methodName](pureUrl)
})
}
break;
}
}
return pureUrl;
},
/* 复制净化后的链接和标题 */
getCleanUrlAndTitle : function(){
const pureUrl = this.Cat_get_pure_url();
const ttileAndUrl = document.title + ' \n' + pureUrl;
GM_setClipboard(ttileAndUrl);
CatCLNotification('链接地址已净化,并和网站标题一起复制到剪切板中~');
PureUrl(pureUrl);
},
/* 复制净化后的链接 */
getCleanUrl : function(){
const pureUrl = this.Cat_get_pure_url();
GM_setClipboard(pureUrl);
CatCLNotification('链接地址已净化并复制到剪切板中~');
PureUrl(pureUrl);
},
/* 直接复制页面链接和标题 */
getUrlAndTitle : function(){
const theUrl = document.title + ' \n' + window.location.href;
GM_setClipboard(theUrl);
CatCLNotification('网站标题 & 链接地址已复制到剪切板中~');
this.CatLCToggleEl();
},
/*** 面板切换 ***/
CatLCToggleEl : function(el){
el = document.querySelector('#Cat_lc_panel');
if(!el) return false;
$(this).toggleClass('show');
$(this).find('span').toggleClass('show');
$('#Cat_lc_panel').slideToggle('fast', 'linear');
},
/* 清理整个页面 */
cleanAllPage : function(){
console.log(this);
const aTagEles = document.getElementsByTagName('a');
for (let i = 0; i < aTagEles.length; i++) {
let theLink = aTagEles[i].href;
if (theLink.match(/^(http:\/\/|https:\/\/|\/\/)/) !== null) {
theLink = theLink.replace(/^\/\//, 'https://');
aTagEles[i].href = Cat_get_pure_url(theLink);
}
}
//panel.style.display = '';
CatCLNotification('页面中所有链接已净化~\n可能导致部分链接无法使用,刷新后恢复。');
this.CatLCToggleEl();
},
CustomRule : function(search){
const CR=$('#Cat_lc_panel_CustomRule');
if(CR.css('display')=='none') {
const theUrlParam=this.param_Count();
const p=$('<p>'),
UrlRule=new RegExp(location.href.replace(/\/[^/]+(\.\w+)?\/?\?.+/, '\/.+$1').replace(/\/%[^/]+/,'').replace(/\//g,'\\/'), 'i'),
SaveBtn=$('<input>').attr({type: 'button', id:'lc_CustomRule_SaveBtn'}).val('应用配置'),
CloseBtn=$('<input>').attr({type: 'button', id:'lc_CustomRule_CloseBtn'}).val('关闭'),
UrlText=$('<input>').attr({type: 'text', id:'lc_CustomRule_UrlRule', placeholder:'网址匹配采用正则表达式', title:'网址匹配采用正则表达式'}).val(localConfig['testReg']||UrlRule),
Option_Auto=$('<label for="lc_CustomRule_AutoCleaner">').text('自动净化链接').append($('<input>').attr({type: 'checkbox', id: 'lc_CustomRule_AutoCleaner'})),
Option_OpenMode=$('<select></select>').attr({id: 'lc_CustomRule_OpenMode'}).append('<option value="history">免刷新·history</option><option value="location">刷新·location</option>');
//history 通过变更历史记录净化链接,有较好的浏览体验,适合个人访问使用。但该方法无法检验净化后的访问结果,有可能因为有用的参数被净化而导致分享链接出错。
//location 通过地址跳转净化链接,会刷新网页,影响浏览体验,适合分享地址使用,及时检验净化后的可访问性。
console.log(theUrlParam);
if(theUrlParam.length==0) {
CR.html('<span class="lc_panle_CustomRule_title">没有可选择的参数</span>');
} else {
//CR.html('<span class="lc_panle_CustomRule_title">选择需要过滤的参数</span><div id="panel_CustomRule_param_new"></div>');
CR.html('<fieldset><legend class="lc_panle_CustomRule_title"> 选择需要过滤的参数 </legend><div id="panel_CustomRule_param_new"></div></fieldset>');
if(this.webRule) {
for(let key of this.webRule.reserveQuery) {
this.reserveQueryJSON[key]=true;
}
}
//生成自定义规则参数选项
theUrlParam.map(function(item, index, arr) {
let label=$('<label>').attr({for:'lc_CustomRule_param_' + index}).text(item),
checkbox=$('<input>').attr({type: 'checkbox', id:'lc_CustomRule_param_' + index}).val(item)
//必须参数检查
if(LinkCleaner.reserveQueryJSON[item.replace(/_/g,'')]) {
checkbox.prop('disabled', true);
label.attr('title', '该参数为必须参数');
label.css({'background':'red'});
}
$('#panel_CustomRule_param_new').append(label.append(checkbox));
}, this.theUrlParam);
}
//CR.append('<span class="lc_panle_CustomRule_title">其它已存在的规则</span><div id="panel_CustomRule_param_exist"></div>');
CR.append('<fieldset><legend class="lc_panle_CustomRule_title"> 其它已存在的规则 </legend><div id="panel_CustomRule_param_exist"></div></fieldset>');
$('#Cat_lc_panel_CustomRule').append(p, '<span class="lc_panle_CustomRule_InputTitle">匹配网址:</span>', UrlText, Option_Auto, '<span class="lc_panle_CustomRule_InputTitle">净化模式:</span>', Option_OpenMode, SaveBtn, CloseBtn, '<br>');
//配置加载
for(let key of localConfig['param']) {
let e=$('#panel_CustomRule_param_new>label>input[type="checkbox"][id^="lc_CustomRule_param_"][value="'+key+'"]'),
label=$('<label>').attr({for:'lc_CustomRule_param_exist_' + key}).text(key),
checkbox=$('<input>').attr({type: 'checkbox', id:'lc_CustomRule_param_exist_' + key}).val(key);
if(e.length>0) e.prop('checked', true);
else $('#panel_CustomRule_param_exist').append(label.append(checkbox.prop('checked', true)));
}
if(localConfig.AutoCleaner) $('#lc_CustomRule_AutoCleaner').prop('checked', true);
//绑定事件
SaveBtn.on('click', function(){
let theSearch=location.search;
localConfig['param']=[];
$('#Cat_lc_panel_CustomRule label>input[type="checkbox"][id^="lc_CustomRule_param_"]').each(function(i, e){
if(e.checked) localConfig['param'].push(e.value);
let rule=new RegExp('[?&]'+e.value+'='+'[^&]*','i');
console.log(e, rule);
theSearch=theSearch.replace(rule,'');
});
//localConfig.testReg=new RegExp($('#Cat_lc_panel_CustomRule>#lc_CustomRule_UrlRule').val().replace(/\\/g, "\\\\"));
let testReg=$('#Cat_lc_panel_CustomRule>#lc_CustomRule_UrlRule').val();
localConfig.testReg=testReg;
console.log(theSearch);
localConfig.AutoCleaner=$('#lc_CustomRule_AutoCleaner').is(':checked');
StorageDB_GM('Cat_LC_Config','', 'localStorage').insert(localConfig);
//localStorage['Cat_LC_Config']=JSON.stringify(localConfig);
console.log(this);
LinkCleaner.AutoCleaner();
});
CloseBtn.on('click', function(){
CR.slideToggle('fast', 'linear');
});
CR.slideToggle('fast', 'linear');
if(!eval(localConfig.testReg).test(location.href)) {
UrlText.css({'background-color':'#ffc8ff'});
p.append('<span style="color:red">当前网址与规则不匹配</span>');
}
}
else
CR.slideToggle('fast', 'linear');
},
AutoCleaner: function(){
console.log(localConfig.testReg);
let newUrl, testReg=eval(localConfig.testReg);
console.log(testReg);
if(testReg.test(location.href)) {
localConfig['param'].forEach((e)=>{
this.theUrlParam.delete(e); //删除掉不要的参数
});
if(this.theUrlParam.toString().length==0) newUrl=webUrl.replace(/\?.+/i,'');
else newUrl=webOrigin+webPathname+"?"+this.theUrlParam;
this.PureUrl(newUrl);
$('div#Cat_lc_button').addClass('work');
} else {
console.log('%c 地址不匹配,停止链接清洗', 'background: yellow');
//CatCLNotification('地址不匹配,停止链接清洗');
}
},
UI : function(){
/** 添加界面 **/
const CatLCPopPanel = document.createElement('div');
CatLCPopPanel.id = 'Cat_link_cleaner';
CatLCPopPanel.innerHTML = `
<div id="Cat_lc_button"><span>︽</span></div>
<div id="Cat_lc_panel">
<div id="Cat_lc_panel_CustomRule" style="display:none;">
</div>
<div id="Cat_lc_panel_content">
<div class="Cat_lc_button" id="Cat_CL_TitleBtn" data-tip="Copy pure link with title">复制网页标题和净化后的链接</div>
<div class="Cat_lc_button" id="CatCLButtonPure" data-tip="Copy pure link only">复制净化后的链接</div>
<div class="Cat_lc_button" id="CatCLButtonCleanAll" data-tip="Clean all links in this page">净化网页中所有链接</div>
<div class="Cat_lc_hr"></div>
<div class="Cat_lc_button" id="Cat_CustomRule" data-tip="Custom Rule">自定义规则</div>
<div class="Cat_lc_button" id="Cat_AutoCleaner" data-tip="Auto Cleaner" title="需要搭配自定义规则使用">自动清洗</div>
<div class="Cat_lc_hr"></div>
</div>
</div>`;
/**/
document.body.insertBefore(CatLCPopPanel, document.body.lastChild.nextSibling);
/** 事件响应函数 **/
/** 添加监听器 **/
$('#Cat_lc_panel>#Cat_lc_panel_content>.Cat_lc_button').each(function(i, eve){
$(eve).on('click', function(e){
//console.log(this, e, this.id, e.id);
switch(this.id) {
case 'Cat_CL_TitleBtn':
/* 净化并复制标题和链接 */
LinkCleaner.getCleanUrlAndTitle();
break;
case 'CatCLButtonPure':
/* 净化并复制链接 */
LinkCleaner.getCleanUrl();
break;
case 'buttonCopyT':
/* 复制当前链接和标题 */
LinkCleaner.getUrlAndTitle();
break;
/* 清理整个页面 */
case 'CatCLButtonCleanAll':
LinkCleaner.cleanAllPage();
break;
case 'Cat_CustomRule':
LinkCleaner.CustomRule();
break;
case 'Cat_AutoCleaner':
localConfig.AutoCleaner=!localConfig.AutoCleaner?true:false;
CatCLNotification('【已'+(localConfig.AutoCleaner?'开启':'关闭')+'】自动净化功能,将在新打开页面生效', localConfig.AutoCleaner?LinkCleaner.AutoCleaner:'');
StorageDB_GM('Cat_LC_Config','', 'localStorage').add('AutoCleaner', localConfig.AutoCleaner);
}
});
});
console.log(this);
/* 面板切换按钮 */
$('#Cat_lc_button').on('click', this.CatLCToggleEl);
UI_CSS();
}
}
/** 获取是否显示页面工具栏 **/
let isShowPageBar = GM_getValue('SHow_page_bar', true);
if (isShowPageBar) LinkCleaner.UI();
if(localConfig.AutoCleaner) {
if(!localConfig.param) {
CatCLNotification('不存在配置,请先添加自定义规则');
return false;
}
else LinkCleaner.AutoCleaner(); //自动清理
}
function UI_CSS(){
/** 添加样式 **/
GM_addStyle(`
#Cat_link_cleaner {
width: 100%;
position: fixed;
left: 0;
bottom: 0;
z-index: 99999999;
pointer-events: none;
font-family: 'Source Sans Pro', arial;
text-align: left;
}
#Cat_link_cleaner * {
pointer-events: auto;
}
#Cat_lc_button {
position: relative;
margin: 0 auto;
width: 36px;
height: 18px;
color: #333;
font-size: 16px;
line-height: 10px;
cursor: pointer;
text-align: center;
border: 1px solid #333;
border-radius: 18px 18px 0 0;
background-color: #555;
box-shadow: 0 0 5px rgba(0, 0, 0, .1);
transition: 0.5s cubic-bezier(0.18, 0.89, 0.32, 1.28);
opacity: 0.3;
}
#Cat_lc_button.work {
background: red;
}
#Cat_lc_button:hover {
height: 24px;
opacity: 1;
}
#Cat_lc_button > span{
color: #fff;
display: inline-block;
transition: 0.5s;
line-height: 20px;
}
#Cat_lc_button.show{
opacity: 1;
}
#Cat_lc_button > span.show{
transform: rotate(180deg);
line-height: 24px;
}
#Cat_lc_panel {
color: #000;
display: none;
border-top: 5px solid #65adff;
background-color: #FFF;
box-shadow: 0 0 5px rgba(0, 0, 0, .1);
}
#Cat_lc_panel > #Cat_lc_panel_CustomRule {
border: 2px solid #ccc;
padding: 10px 20px;
font-size: 18px;
}
#Cat_lc_panel > #Cat_lc_panel_CustomRule > fieldset {
border: 2px threedface groove;
margin: 0 20px 20px 20px;
padding: 10px 5px;
}
#Cat_lc_panel > #Cat_lc_panel_CustomRule > fieldset > legend.lc_panle_CustomRule_title {
font-size: 30px;
line-height: 1.5;
font-weight: bold;
padding: 5px;
display: block;
padding: 0 20px;
margin: 0 20px;
}
#Cat_lc_panel > #Cat_lc_panel_CustomRule > fieldset > [id^="panel_CustomRule_param_"] > label {
margin: 0 5px;
padding: 5px 10px;
border: 2px solid #ccc;
border-radius: 15px;
background: #ccc;
display: inline-flex;
cursor: pointer;
}
#Cat_lc_panel > #Cat_lc_panel_CustomRule > fieldset > [id^="panel_CustomRule_param_"] > label > input[type="checkbox"] {
margin: auto;
width: 24px;
height: 24px;
}
#Cat_lc_panel > #Cat_lc_panel_CustomRule > .lc_panle_CustomRule_InputTitle {
line-height: 36px;
border: 2px solid #333;
display: inline-block;
background: #ccc;
box-sizing: border-box;
padding: 0 5px;
}
#Cat_lc_panel > #Cat_lc_panel_CustomRule > input[type="button"] {
padding: 0 5px;
height: 40px;
font-size: 20px;
line-height: 1;
box-sizing: border-box;
border: 2px solid #333;
}
#Cat_lc_panel > #Cat_lc_panel_CustomRule > input[type="text"] {
margin: 5px 0;
padding: 2px;
width: 700px;
height: 40px;
font-size: 20px;
border: 2px solid #333;
box-sizing: border-box;
}
#Cat_lc_panel > #Cat_lc_panel_CustomRule > label {
padding: 0 5px;
display: inline-flex;
border: 2px solid #333;
box-sizing: border-box;
background: deepskyblue;
line-height: 36px;
}
#Cat_lc_panel > #Cat_lc_panel_CustomRule > label > input[type="checkbox"][id^="lc_CustomRule_AutoCleaner"] {
margin: auto;
width: 24px;
height: 24px;
}
#Cat_lc_panel > #Cat_lc_panel_CustomRule > #lc_CustomRule_OpenMode {
height: 40px;
font-size: 18px;
border: 2px solid #333;
box-sizing: border-box;
}
#Cat_lc_panel > #Cat_lc_panel_content {
display: flex;
justify-content: center;
align-items: center;
flex: 1 1 none;
flex-wrap: wrap;
width: 100%;
max-width: 900px;
margin: 0 auto;
padding: 16px 0;
text-align: center;
position: relative;
}
#Cat_lc_panel > #Cat_lc_panel_content > .Cat_lc_button {
position: relative;
padding: 8px 16px;
margin: 0 8px 0 0;
font-size: 16px;
line-height: 1.2em;
font-weight: lighter;
border: 1px solid #65adff;
border-radius: 8px;
cursor: pointer;
}
#Cat_lc_panel > #Cat_lc_panel_content > .Cat_lc_button:hover {
border: 1px solid #0062d1;
background-color: #0062d1;
color: #FFF;
font-weight: normal;
}
#Cat_lc_panel > #Cat_lc_panel_content > .Cat_lc_button:hover::before {
content: attr(data-tip);
border-radius:3px;
color: #fff;
left: 50%;
padding: 10px;
white-space: pre;
position: absolute;
text-align: center;
width: calc(100% + 20px);
bottom: calc(100% + 10px);
margin-left: calc(-50% - 20px);
background-color: rgba(0, 0, 0, .9);
}
#Cat_lc_panel > #Cat_lc_panel_content > .Cat_lc_button:hover::after {
content: "";
position: absolute;
width: 0;
height: 0;
left: calc(50% - 8px);
top: -10px;
border-top: 8px solid rgba(0, 0, 0, .8);
border-right: 8px solid transparent;
border-left: 8px solid transparent;
}
#Cat_lc_panel > #Cat_lc_panel_content > .Cat_lc_hr {
width: 100%;
margin: 5px 0;
}
#Cat_lc_panel > #Cat_lc_panel_content > #CatCLButtonCoffee {
padding: 0;
margin: 0;
}
#Cat_lc_panel > #Cat_lc_panel_content > #CatCLButtonCoffee > svg {
width: 35px;
height: 35px;
}
`);
}
/* 弹出通知 */
function CatCLNotification(text, click) {
GM_notification(text, '清洗链接 Success! ', "", click);
};
function getUrlParam(name, url, option, newVal) {//筛选参数,url 参数为数字时
var search = url ? url.replace(/^.+\?/,'') : location.search;
//网址传递的参数提取,如果传入了url参数则使用传入的参数,否则使用当前页面的网址参数
var reg = new RegExp("(?:^|&)(" + name + ")=([^&]*)(?:&|$)", "i"); //正则筛选参数
var str = search.replace(/^\?/,'').match(reg);
if (str !== null) {
switch(option) {
case 0:
return unescape(str[0]); //所筛选的完整参数串
case 1:
return unescape(str[1]); //所筛选的参数名
case 2:
return unescape(str[2]); //所筛选的参数值
case 'new':
return url.replace(str[1]+'='+str[2], str[1]+'='+newVal);
default:
return unescape(str[2]); //默认返回参数值
}
} else {
return null;
}
}
function StorageDB_GM(collectionName, key, local) { //collectionName = 对应的 GM_value 表,key = 表中的键名,localDB = 本地配置名(启用时同步修改GM_DB与本地变量DB)
let DB;
if(local) DB=localStorage[collectionName]?JSON.parse(localStorage[collectionName]):{};
else DB=GM_getValue(collectionName) ? GM_getValue(collectionName) : {};
if(key && !DB[key]) {
DB[key]={};
DB.length++;
}
//console.log('StorageDB_GM start', collectionName, key, DB[key]);
return {
add : function(name, value) {
if(key)
DB[key][name]=value;
else {
DB[name]=value;
console.error('缺乏 key ,无法添加数据');
//return false;
}
//console.log(collectionName, key, DB, DB[key]);
local ? localStorage[collectionName]=JSON.stringify(DB) : GM_setValue(collectionName, DB);
return DB;
},
del:function(name) {
if(name) {
console.log(DB, DB[name]);
delete DB[name];
Storage.setItem(collectionName, JSON.stringify(DB)); //回写 localStorage
} else {
//删除整个 localStorage 数据
Storage.removeItem(name);
}
},
insert: function(obj){
if(key) {
//console.log('StorageDB_GM insert: ', key, obj);
DB[key]=obj;
} else {
DB=obj;
console.error('缺乏 key ,无法插入数据');
}
console.log(local, collectionName, DB)
local ? localStorage[collectionName]=JSON.stringify(DB) : GM_setValue(collectionName, DB);
return DB;
},
Updata : function(name, obj, value){
DB[obj]=DB[obj]||{};
DB[obj][name]=value;
Storage.setItem(collectionName, JSON.stringify(DB)); //回写 localStorage
},
Query : function(obj,name){
return DB[obj]?name?(DB[obj][name]?DB[obj][name]:null):DB[obj]:null;
},
find : function(name) {
if(!collectionName) return false;
if(DB&&key&&DB[key]) return DB[key][name];
else if(DB&&DB[name]) return DB[name];
else return false;
},
read : function(name){
if(key) return $.isEmptyObject(DB[key])?null:DB[key];//如果为空,则返回 null
return $.isEmptyObject(DB)?null:DB;//如果为空,则返回 null
}
}
}
/* 注册(不可用)菜单项 */
GM_registerMenuCommand('复制【净化】链接和标题', LinkCleaner.getCleanUrlAndTitle);
GM_registerMenuCommand('复制【净化】链接', LinkCleaner.getCleanUrl);
GM_registerMenuCommand('【净化】所有链接', LinkCleaner.cleanAllPage);
GM_registerMenuCommand('显示/隐藏页面工具条', () => {
GM_setValue('SHow_page_bar', !isShowPageBar);
isShowPageBar = GM_getValue('SHow_page_bar', true);
alert('页面工具条已被设置为【' + (isShowPageBar ? '显示' : '隐藏') + '】,仅在此后新打开页面中生效。');
});
})();