Sleazy Fork is available in English.

绅士之庭透明界面

让绅士之庭的界面和文章背景半透明

このスクリプトの質問や評価の投稿はこちら通報はこちらへお寄せください。
// ==UserScript==
// @name         绅士之庭透明界面
// @namespace    moe.moekai.gmgardtransparentpost
// @version      3.6.79
// @description  让绅士之庭的界面和文章背景半透明
// @author       YIU
// @match        http*://gmgard.com/*
// @match        http*://hggard.com/*
// @match        http*://gmgard.moe/*
// @match        http*://lzone.moe/*
// @icon         https://hggard.com/favicon.ico
// @run-at       document-end
// @grant        GM_addStyle
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_addValueChangeListener
// @grant        unsafeWindow
// ==/UserScript==

(function() {
	const $ = unsafeWindow.jQuery;
	const addStyleDoms = {};
	const settingsKeys = {
		panelBgAlpha: 'panel-bg-alpha',
		panelBgBlur: 'panel-bg-blur'
	};

	//============ < STYLE > =============================
	const hartIcon = '';
	const strs = `#body{background-color:rgba(255,255,255,0.75);backdrop-filter:blur(0px)}
.categories{background:#ffffff99}
#blog{background-color:#fff0}
#blog .uimg{background:#fff9}
hr{border-top-color:#eee8!important;border-bottom-color:#eee8!important}
.well{background-color:#f5f5f5b8}
input.tagsearch{background-color:#f5f5f599!important;}
span[style='background:#000000'],span[style='background: #000000'],span[style='background:rgb(0, 0, 0)'],
 span[style='background-color:#000000'],span[style='background-color: #000000'],span[style='background-color:rgb(0, 0, 0)']{background:#000000a5}
span[style='background:#ffffff'],span[style='background: #ffffff'],span[style='background:rgb(255, 255, 255)'],
 span[style='background-color:#ffffff'],span[style='background-color: #ffffff'],span[style='background-color:rgb(255, 255, 255)']{background:#fffb}
.reply-section{background:#efeeef90!important}
.home-list{background-color:#f7f7de50}
.home-settings{background:#dddddd50}
#rankContent{background-color:#daebf470}
.author-info{background:#f3f3f380}
.author-sign{background:#fffa;border:1px outset #8888}
.spoiler-content{background:none repeat scroll 0 0 #f5f5f590}
.spoiler-content span[style*='background-color']{background-color:#000000a5}
.navbar-inner{background-color:#fff0;background-image:linear-gradient(to bottom,#ffffffa0,#f2f2f280)}
.btn-info{background:transparent linear-gradient(to bottom,#5bc0dea0,#2f96b480)}
.btn:hover,.btn:focus{background-position:unset;-webkit-transition:unset;-moz-transition:unset;-o-transition:unset;transition:unset}
#multiview table{background-color:#fff6}
pre{background-color:#f5f5f599}
.expbar{background-image:linear-gradient(to bottom,#f5f5f500,#f9f9f999)}
.progress-success .bar{background-image:linear-gradient(to bottom,#00ff004a,#00ff008c)}
#jumppage,#postjumppage{background-color:#fffc}
.pager .btn,.ajax-pager .btn{background:#fffc;background-image:none}
.main-content .label,.main-content .badge{background-color:#9999}
.btn{background-image:linear-gradient(to bottom,transparent,transparent);background-color:#f5f5f5b0}
.btn-warning{background-image:linear-gradient(to bottom,#fbb45099,#f8940699)!important}
.btn-danger{background-image:linear-gradient(to bottom,#ee5f5bc0,#bd362fc0)!important}
.btn-inverse{background-image:linear-gradient(to bottom,#4449,#222c)}
.listview li div.badge,.hanintro .label{background-color:#2d89efbd}
.label-info[href],.badge-info[href]{background-color:#007ebda0!important}
.label-important,.badge-important{background-color:#b94a48d6!important}
.label-info,.badge-info,#createform .label-info{background-color:#3a87adb0!important}
.label-warning,.badge-warning{background-color:#f89406b0!important}
.label-success,.badge-success{background-color:#468847b0!important}
.label-inverse,.badge-inverse{background-color:#0008!important}
.lolly,.lvl{background-color:#f5ab36b0!important}
.cbadge{background-color:#FF677D!important}
.scate-1 .sactive,.scate-1 .sinde,.scate-1.sactive,.scate-1.sinde{background:#e63333}
.scate-2 .sactive,.scate-2 .sinde,.scate-2.sactive,.scate-2.sinde{background:#2f75b5}
.scate-3 .sactive,.scate-3 .sinde,.scate-3.sactive,.scate-3.sinde{background:#60f}
.scate-4 .sactive,.scate-4 .sinde,.scate-4.sactive,.scate-4.sinde{background:#70ad47}
.scate-5 .sactive,.scate-5 .sinde,.scate-5.sactive,.scate-5.sinde{background:#bf8f00}
.scate-6 .sactive,.scate-6 .sinde,.scate-6.sactive,.scate-6.sinde{background:#ff92db}
.scate-7 .sactive,.scate-7 .sinde,.scate-7.sactive,.scate-7.sinde{background:#630}
.scate-8 .sactive,.scate-8 .sinde,.scate-8.sactive,.scate-8.sinde{background:#3cf}
.scate-9 .sactive,.scate-9 .sinde,.scate-9.sactive,.scate-9.sinde{background:#d9e51e}
.listview li div.badge.info{background-color:#007ebda0}
.alert-info{background-color:#d9edf7b0}
.btn-primary{background-image:linear-gradient(to bottom,transparent,transparent)!important;background-color:#006dcc99!important}
.btn-primary:hover,.btn-primary:focus,.btn-primary:active,.btn-primary.active,.btn-primary.disabled,.btn-primary[disabled]{background-color:#04c9!important}
.btn-warning:hover,.btn-warning:focus,.btn-warning:active,.btn-warning.active,.btn-warning.disabled,.btn-warning[disabled]{background:#f89406a0!important}
.nav-pills>.active>a,.nav-pills>.active>a:hover,.nav-pills>.active>a:focus{background-color:#08ca}
.navbar .nav>.active>a, .navbar .nav>.active>a:hover, .navbar .nav>.active>a:focus{background-color:#e5e5e595}
.rankflag::before{background:linear-gradient(210deg,rgba(218,235,244,.4) 0,rgba(218,235,244,.4) 50%,transparent 51%,transparent 100%)}
.rankflag::after{background:linear-gradient(150deg,rgba(218,235,244,.4) 0,rgba(218,235,244,.4) 50%,transparent 51%,transparent 100%)}
.nav-tabs .active a,.nav-tabs .active a:hover,.nav-tabs .active a:focus,.user-comment,#multiview input[type='text'],#multiview textarea{background-color:#fffb}
.feed-item,select{background-color:#fff9}
textarea,input[type='text'],input[type='password'],input[type='datetime'],input[type='datetime-local'],input[type='date'],input[type='month'],input[type='time'],
input[type='week'],input[type='number'],input[type='email'],input[type='url'],input[type='search'],input[type='tel'],input[type='color'],.uneditable-input{background-color:#fffa}
footer{background-color:#e2e2e2e0}
.feed-footer{background:#f7f6f980}
.tabcontent{background-color:#fff4}
.tooltip-inner{background-color:#000b}
#flinkdiv img{opacity:.5;transition: opacity ease-in-out .5s}
#flinkdiv img:hover{opacity:.9}
#donatediv #msgmenu .active a,#donatediv .btn-warning,.expbar,.progress-success .bar{background-color:transparent}
.msglist-item:hover{background-color:#c0c0c090}
.dropdown-menu{background-color:#fffd;backdrop-filter:blur(3px)}
.nav-pills .open .dropdown-toggle{background-color:#999a}
.nav>li>a:hover,.nav>li>a:focus{background-color:#eeeb}
.post .post-title a:hover{background-color:#e6e6e6cc}
.totop{margin-left:1000px;background-repeat:no-repeat;background-position-x:0px;-webkit-transition:background-position-x ease-out .6s;transition:background-position-x ease-out .6s}
.totop-wrapper.hidden>.totop{background-position-x:-250px}
.user-bg>h2,.user-bg>.user-stats{padding:2px 10px}
.user-bg>h2,.user-bg>.user-stats,.user-bg>.user-numbers-div,.userinfo h3{background-color:#fffa;border-radius:5px}
.bubble{background:#ffffff90}
.bubble:after{border-width:10px 15px 10px 0;border-color:transparent #ffffffe8}
.btn-info{background:transparent linear-gradient(to bottom,#5bc0dea0,#2f96b480)}
.well{background-color:#f5f5f5b8}
#main:has(.monthly-container){background:linear-gradient(135deg,#cebe29aa 0,#9b1f50aa 33%,#2989d8aa 71%,#89b4ffaa 91%)!important}
.btn-toolbar .btn,.searchbtn,.favbtn,#blog a[onclick*='rptBlog'],#newtag .btn,.author-info .btn:not(.disabled){background:#f5f5f5b0;background-image:none}
#tag-input{background-color:#fffc}
.tag-info{background:#f3f3f3a0}
.tm-tag-info{color:#4594b5;background-color:#c5eefaa0}
.tm-tag:not(.tm-tag-info){color:#368000;background-color:#b1ff6b80}
.cke_wysiwyg_frame,.cke_wysiwyg_div,a.cke_combo_button{background:transparent!important}
.cke_top,.cke_bottom{background:transparent linear-gradient(to bottom,#f5f5f555,#019ed555)!important}
.cke_inner{background:transparent linear-gradient(to right, rgba(255,255,255,.5),rgba(255,255,255,.5),rgba(255,255,255,.5),rgba(255,255,255,1),rgba(255,255,255,1))!important}
.cke_toolgroup,.cke_combo_button{background:transparent linear-gradient(to bottom,#fff9,#e4e4e444)!important}
::-webkit-scrollbar{width:15px;height:15px}
::-webkit-scrollbar-thumb{border:solid 1px #0001;border-radius:8px;box-shadow:0 0 6px 2px #fff8;background-color:rgba(153,153,153,.5)}
::-webkit-scrollbar-thumb:hover{background-color:rgba(100,100,100,.5);border-color:#0004}
::-webkit-scrollbar-thumb:active{background-color:rgba(50,50,50,.5);border-color:#0006}
::-webkit-scrollbar-track{border-radius:8px;-webkit-box-shadow:inset 0 0 4px rgba(0,0,0,.3);background-color:#fff8}
::-webkit-scrollbar-track:hover{background-color:#fffa}
::-webkit-scrollbar-track:active{background-color:#fffd}
::-webkit-scrollbar-corner{display:block}
::-webkit-scrollbar-button{border-radius:8px;background-color:#fff8}
::-webkit-scrollbar-button:start:increment,::-webkit-scrollbar-button:end:decrement{visibility:hidden}
::-webkit-scrollbar-button:start,::-webkit-scrollbar-button:end{width:12px;height:12px;border-width:8px;border-style:solid}
::-webkit-scrollbar-button:start{border-color:transparent transparent rgba(0,0,0,.3) transparent}
::-webkit-scrollbar-button:start:hover{border-color:transparent transparent rgba(0,0,0,.5) transparent;background-color:#fffa}
::-webkit-scrollbar-button:start:active{border-color:transparent transparent rgba(0,0,0,.8) transparent;background-color:#fffd}
::-webkit-scrollbar-button:end{border-color:rgba(0,0,0,.3) transparent transparent transparent}
::-webkit-scrollbar-button:end:hover{border-color:rgba(0,0,0,.5) transparent transparent transparent;background-color:#fffa}
::-webkit-scrollbar-button:end:active{border-color:rgba(0,0,0,.8) transparent transparent transparent;background-color:#fffd}
.gm_grid-container{display:grid;grid-template-columns:auto 1fr;gap:10px;align-items:center}
.gm_grid-item{padding:10px;border:none}
.gm_sliderBar{position:relative;width:100%}
.gm-bgalpha-slider,.gm-bgblur-slider{position:relative;width:50%}
.gm_slot{position:absolute;top:50%;left:0;transform:translateY(-50%);width:100%;height:10px;background-color:#bbb8;z-index:10;border-radius:10px}
.gm_process{background-color:#ff8fbc;background:linear-gradient(to right,#ffb0b0,#ff8fbc);z-index:20;width:0}
.gm_slider{position:absolute;top:50%;left:0;transform:translate(-50%,-50%);width:36px;height:31px;z-index:30;cursor:move;background-image:url('${hartIcon}');background-size:auto 100%;background-repeat:no-repeat;background-position:center}
.gm_percent{position:absolute;top:50%;right:-60px;transform:translateY(-50%);color:#666}
.gm_non-selectable{-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}`;


	// ===================== < JS > =======================================
	GM_addStyle(strs);


	function setPanelBackground(alpha = 0.75){
		const bubbleAlphaDiff = 0.37 - alpha;
		const bubbleAfterAlpha =calculateTransparency(alpha, 0.3, 1, 0.7, 0.97);
		const newcss = `#body,input.tagsearch{background-color:rgba(255,255,255,${alpha})!important}
		.categories{background:rgba(255,255,255,${alpha - 0.11 > 0 ? alpha - 0.11 : 0 })}
		.dropdown-menu{background-color:rgba(255,255,255,${alpha + 0.13 < 1 ? alpha + 0.13 : 1 })}
		.btn-toolbar .btn,.searchbtn,.favbtn,#blog a[onclick*='rptBlog'],#newtag .btn,
		.HGswitch .btn:not(.disabled),.author-info .btn:not(.disabled){background:rgba(245,245,245,${alpha - 0.05 > 0 ? alpha - 0.05 : 0 })}
		.nav-tabs .active a,.nav-tabs .active a:hover,.nav-tabs .active a:focus,.user-comment,#multiview input[type='text'],
		#multiview textarea{background-color:rgba(255,255,255,${alpha - 0.03 > 0 ? alpha - 0.03 : 0 })}
		.bubble{background:rgba(255,255,255,${alpha + bubbleAlphaDiff < 1 ? alpha + bubbleAlphaDiff : 1 })}
		.bubble:after{border-color:transparent rgba(255,255,255,${bubbleAfterAlpha})}
		.well{background-color:rgba(245,245,245,${alpha + 0.01 < 1 ? alpha + 0.01 : 1 })}
		.navbar-inner{background-image:linear-gradient(to bottom,rgba(255,255,255,${alpha + 0.07 < 1 ? alpha + 0.07 : 1 }),rgba(242,242,242,${alpha + 0.15 < 1 ? alpha + 0.15 : 1}))}`;

		if (addStyleDoms.panelbg) {
			addStyleDoms.panelbg.remove();
		}
		addStyleDoms.panelbg = GM_addStyle(newcss);
	}


	function setPanelBlur(blur = 0){
		$("#body").css("backdrop-filter",`blur(${blur}px)`);
	}


	function initial(){
		setPanelBackground(GM_getValue(settingsKeys.panelBgAlpha, undefined));
		setPanelBlur(GM_getValue(settingsKeys.panelBgBlur, undefined));
	}


	function calculateTransparency(input, inputMin, inputMax, outputMin, outputMax) {
		const output = outputMin + (outputMax - outputMin) * ((input - inputMin) / (inputMax - inputMin));
		return output;
	}


	const observer = new MutationObserver((mutationsList, observer) => {
		for (const mutation of mutationsList) {
			if (mutation.type === 'childList') {
				if (unsafeWindow.CKEDITOR && unsafeWindow.CKEDITOR.tools && unsafeWindow.CKEDITOR.tools.callFunction) {
					hookCKEditorToolsCallFunction();
					observer.disconnect();
					break;
				}
			}
		}
	});
	observer.observe(document.body, { childList: true, subtree: true });


	function hookCKEditorToolsCallFunction() {
		const originalCallFunction = unsafeWindow.CKEDITOR.tools.callFunction;

		unsafeWindow.CKEDITOR.tools.callFunction = new Proxy(originalCallFunction, {
			apply(target, thisArg, argumentsList) {
				if(argumentsList.length > 0){
					const elem = argumentsList[1];
					if(elem instanceof Element || elem instanceof HTMLDocument){
						if($(elem).hasClass('cke_button__maximize')){
							$("header").css('visibility',elem.title.includes('全屏') ? 'hidden' : 'visible');
						}
					}
				}
				return Reflect.apply(target, thisArg, argumentsList);
			}
		});
	}


	/**
	 * 将小数范围映射到百分比范围,并取整
	 * @param {number} value - 要映射的值
	 * @param {number} min - 原始范围的最小值
	 * @param {number} max - 原始范围的最大值
	 * @param {number} newMin - 新范围的最小值
	 * @param {number} newMax - 新范围的最大值
	 * @returns {number} - 映射后的取整值
	 */
	function mapValue(value, min, max, newMin = 0, newMax = 100) {
		const mappedValue = ((value - min) / (max - min)) * (newMax - newMin) + newMin;
		return Math.floor(mappedValue);
	}


	/**
	* 将百分比范围映射回小数范围,并保留两位小数
	* @param {number} mappedValue - 要映射的值(在新范围内)
	* @param {number} newMin - 新范围的最小值
	* @param {number} newMax - 新范围的最大值
	* @param {number} min - 原始范围的最小值
	* @param {number} max - 原始范围的最大值
	* @returns {number} - 映射回来的原始值,保留两位小数
	*/
	function reverseMapValue(mappedValue, newMin, newMax, min, max) {
		const originalValue = ((mappedValue - newMin) / (newMax - newMin)) * (max - min) + min;
		return parseFloat(originalValue.toFixed(2));
	}


	/** 创建滑块条
	* @param {string} id - 滑块条id
	* @param {string} parent - 要插入到的父级节点
	* @param {function} moveCallback - 滑块条改变时触发回调
	*/
	function createGMSliderBar(id, parent, moveCallback) {
		const template = `<div id="${id}" class="gm_sliderBar gm_non-selectable"><div class="gm_slot"></div><div class="gm_slot gm_process"></div>
        <div class="gm_slider"></div><span class="gm_percent">0%</span>
			</div>`;

		const gmSbar = $(template);
		$(parent).append(gmSbar);
		let process = $(`#${id} .gm_process`)[0];
		let slider = $(`#${id} .gm_slider`)[0];
		let percent = $(`#${id} .gm_percent`)[0];
		let max = $(`#${id}`)[0].offsetWidth;
		let startX = 0,
			moveX = 0,
			currentX = 0,
			isDrag = false;

		const handleStart = function(e){
			e.stopPropagation();
			if (e.button === 0 || e.touches) {
				const cx = !e.touches ? e.clientX : e.touches[0].clientX;
				isDrag = true;
				startX = cx;
				document.addEventListener('mousemove', handleMove);
				document.addEventListener('mouseup', handleEnd);
				document.addEventListener('touchmove', handleMove);
				document.addEventListener('touchend', handleEnd);
			}
		};

		const handleMove = function(e){
			e.stopPropagation();
			if(isDrag){
				const cx = !e.touches ? e.clientX : e.touches[0].clientX;
				let diffX = cx - startX;
				moveX = diffX + currentX;
				if(moveX < 0) moveX = 0;
				if(moveX > max) moveX = max;
				let pre = Math.floor(moveX / max * 100);
				percent.innerText = pre + '%';
				slider.style.left = moveX + 'px';
				process.style.width = (moveX + 10) + 'px';
				if(moveCallback) moveCallback(pre);
			}
		}

		const handleEnd = function(e){
			e.stopPropagation();
			isDrag = false;
			currentX = moveX;
			document.removeEventListener('mousemove', handleMove);
			document.removeEventListener('mouseup', handleEnd);
			document.removeEventListener('touchmove', handleMove);
			document.removeEventListener('touchend', handleEnd);
		}

		gmSbar[0].setval = v =>{
			moveX = Math.floor(v / 100 * max);
			if(moveX < 0) moveX = 0;
			if(moveX > max) moveX = max;
			startX = moveX;
			currentX = moveX;
			gmSbar[0].clientX = moveX;
			percent.innerText = v + '%';
			slider.style.left = moveX + 'px';
			process.style.width = (moveX + 10) + 'px';
		};

		slider.addEventListener('mousedown', handleStart);
		slider.addEventListener('touchstart', handleStart);
		return gmSbar[0];
	}


	// 创建设置页面
	function createSettingsPage() {
		function addSettingsGroup(id, title, content) {
			const accordion = $('.accordion');
			if (!accordion){
				console.error('Accordion settings list not found');
				return;
			}
			if ($(`#${id}`).length > 0) {
				$(`#${id}`).append(content);
				return;
			}
			const groupTemplate = `<div class="accordion-group">
			<div class="accordion-heading"><a class="accordion-toggle collapsed" data-toggle="collapse" data-parent="#account-accordion" href="#${id}"><h3>${title}</h3></a></div>
			<div class="accordion-body collapse" id="${id}" style="height:0px">${content}</div>
			</div>`;
			return accordion.prepend(groupTemplate);
		}

		// ------------- 背景透明度调整 --------------
		const backgroundAlphaPanel = `<div class="gm_grid-container">
		<div class="gm_grid-item"><strong>内容背景透明度</strong></div>
		<div class="gm_grid-item gm-bgalpha-slider"></div><br>
		</div>`;

		addSettingsGroup('gm-sg-panel', '界面显示设置', backgroundAlphaPanel);
		const gmAlphaSlider = createGMSliderBar('gm-s-background-alpha-slider', '.gm-bgalpha-slider', function(pre) {
			const bgAlpha = reverseMapValue(pre, 0, 100, 0.3, 1);
			GM_setValue(settingsKeys.panelBgAlpha, bgAlpha);
		});

		gmAlphaSlider.setval(mapValue(GM_getValue(settingsKeys.panelBgAlpha, 0.75), 0.3, 1));

		GM_addValueChangeListener(settingsKeys.panelBgAlpha, function (name, old_value, new_value, remote) {
			setPanelBackground(new_value);
		});

		// ------------- 背景模糊度调整 --------------
		const backgroundBlurPanel = `<div class="gm_grid-container">
		<div class="gm_grid-item"><strong>内容背景模糊度</strong></div>
		<div class="gm_grid-item gm-bgblur-slider"></div><br>
		</div>`;

		addSettingsGroup('gm-sg-panel', undefined, backgroundBlurPanel);
		const gmBlurSlider = createGMSliderBar('gm-s-background-blur-slider', '.gm-bgblur-slider', function(pre) {
			const bgBlur = reverseMapValue(pre, 0, 100, 0, 8);
			GM_setValue(settingsKeys.panelBgBlur, bgBlur);
		});

		gmBlurSlider.setval(mapValue(GM_getValue(settingsKeys.panelBgBlur, 0), 0, 8));

		GM_addValueChangeListener(settingsKeys.panelBgBlur, function (name, old_value, new_value, remote) {
			setPanelBlur(new_value);
		});
	}

	// ==========================================
	initial();

	if (location.href.match(/Account\/Manage/gi)){
		createSettingsPage();
	}

})();