// ==UserScript==
// @name NGA优化摸鱼体验
// @namespace https://www.hldww.com/
// @version 1.7
// @require https://cdn.staticfile.org/jquery/3.4.0/jquery.min.js
// @description NGA论坛显示优化,功能增强,防止突然蹦出一对??而导致的突然性的社会死亡
// @author HLD
// @match *://bbs.nga.cn/*
// @match *://ngabbs.com/*
// @match *://nga.178.com/*
// @downloadURL none
// ==/UserScript==
(function() {
'use strict';
let setting = {
hideAvatar: true,
hideSmile: true,
hideImage: false,
hideSign: true,
hideHeader: false,
linkTargetBlank: true,
imgResize: true,
authorMark: true,
markAndBan: true,
banMode: 'SIMPLE'
}
let post_author = []
let ban_list = []
let mark_list = []
//同步配置
if(window.localStorage.getItem('hld__NGA_setting')){
let local_setting = JSON.parse(window.localStorage.getItem('hld__NGA_setting'))
for(let k in setting) {
!local_setting.hasOwnProperty(k) && (local_setting[k] = setting[k])
}
for(let k in local_setting) {
!setting.hasOwnProperty(k) && delete local_setting[k]
}
setting = local_setting
}
//注册按键
$('body').keyup(function(event){
if (/textarea|select|input/i.test(event.target.nodeName)
|| /text|password|number|email|url|range|date|month/i.test(event.target.type)) {
return;
}
//切换显示头像
if(event.keyCode == 81){
$('.avatar').toggle()
}
//切换显示表情
if(event.keyCode == 87){
$('img').each(function(){
const classs = $(this).attr('class');
if(classs && classs.includes('smile')) $(this).toggle()
})
$('.smile_alt_text').toggle()
}
//切换显示图片
if(event.keyCode == 69){
$('.postcontent img').each(function(){
const classs = $(this).attr('class');
if(!classs && $(this).width() > 24) {
if($(this).is(":hidden")) {
$(this).show()
$('.switch-img').hide()
}else {
$('.switch-img').css('display', 'inline')
$(this).hide()
}
}
})
}
})
//查找楼主
if(setting.authorMark) {
const local_post_author = window.localStorage.getItem('hld__NGA_post_author')
local_post_author && (post_author = local_post_author.split(','))
const tid = GetQueryString('tid')
if($('#postauthor0').length > 0 && tid) {
const author_str = `${tid}:${$('#postauthor0').text()}`
if(!post_author.includes(author_str))
post_author.unshift(author_str) > 10 && post_author.pop()
window.localStorage.setItem('hld__NGA_post_author', post_author.join(','))
}
for(let pa of post_author) {
const t = pa.split(':')
if(t[0] == tid) {
$('body').append(``)
break
}
}
}
//拉黑备注
if(setting.markAndBan) {
const local_ban_list = window.localStorage.getItem('hld__NGA_ban_list')
local_ban_list && (ban_list = local_ban_list.split(','))
const local_mark_list = window.localStorage.getItem('hld__NGA_mark_list')
local_mark_list && (mark_list = local_mark_list.split(','))
//绑定事件
$('body').on('click', '.hld__extra-icon', function(){
const type = $(this).data('type')
const user = $(this).data('user')
if(type == 'ban') {
let ban_name = window.prompt('是否拉黑此用户?\n请检查用户名称,可能会出现解析异常', user)
ban_name = $.trim(ban_name)
if(ban_name) {
!ban_list.includes(ban_name) && ban_list.push(ban_name)
window.localStorage.setItem('hld__NGA_ban_list', ban_list.join(','))
reRender()
}
}
if(type == 'mark') {
const exists_remark = mark_list.find(v => v.startsWith(user))
let current_remark = exists_remark ? exists_remark.split(':')[1] : ''
let remark = window.prompt('请输入要备注的名称,此后以此备注高亮显示代替原ID\n留空则清除备注(清空备注重载生效)', current_remark)
remark = $.trim(remark)
if(remark.includes(':')) {
alert('备注不能包含“:”为脚本保留符号')
} else{
const r = `${user}:${remark}`
const check = mark_list.findIndex(v => v.startsWith(user))
remark == '' ? check > -1 && mark_list.splice(check, 1) : check > -1 ? mark_list[check] = r : mark_list.push(r)
window.localStorage.setItem('hld__NGA_mark_list', mark_list.join(','))
reRender()
}
}
})
//隐藏版头
if(setting.hideHeader) {
$('#toppedtopic, #sub_forums').hide()
let $toggle_header_btn = $('')
$toggle_header_btn.click(()=>$('#toppedtopic, #sub_forums').toggle())
$('#toptopics > div > h3').append($toggle_header_btn)
}
//名单管理
$('body').on('click', '#hld__list_manage', function(){
$('body').append(`
×
备注名单
一行一条,格式为<用户名>:<备注> 如“abc123:菜鸡”
`)
$('#hld__ban_list_textarea').val(ban_list.join('\n'))
$('#hld__mark_list_textarea').val(mark_list.join('\n'))
})
$('body').on('click', '#hld__banlist_panel_close', function(){
$('#hld__banlist_panel').remove()
})
$('body').on('click', '#hld__save_list', function(){
ban_list = $('#hld__ban_list_textarea').val().split('\n')
ban_list = RemoveBlank(ban_list)
ban_list = Uniq(ban_list)
mark_list = $('#hld__mark_list_textarea').val().split('\n')
mark_list = RemoveBlank(mark_list)
mark_list = Uniq(mark_list)
window.localStorage.setItem('hld__NGA_ban_list', ban_list.join(','))
window.localStorage.setItem('hld__NGA_mark_list', mark_list.join(','))
$('#hld__banlist_panel').remove()
})
}
//动态检测
setInterval(()=>{
$('.forumbox.postbox[hld-render!=ok]').length > 0 && runDom()
if(setting.markAndBan && $('.topicrow .author[hld-render!=ok]').length > 0) runMark()
$('#hld__setting').length == 0 && $('#startmenu > tbody > tr > td.last').append('')
}, 100)
//大图
const resizeImg = (el) => {
if($('#hld__img_full').length > 0) return
let url_list = []
let current_index = el.parent().find('[hld__imglist=ready]').index(el)
el.parent().find('[hld__imglist=ready]').each(function(){
url_list.push($(this).data('srcorg') || $(this).data('srclazy') || $(this).attr('src'))
})
let $imgBox = $('')
let $imgContainer = $('')
let $img = $('
')
const renderImg = (index) => {
let timer = null
$('#loader').show()
$imgContainer.css({
'top': $(window).height() * 0.03 + 'px',
'left': (($(window).width() - ($(window).height()) * 0.85) / 2) + 'px',
'width': $(window).height() * 0.85 + 'px',
'height': $(window).height() * 0.85 + 'px'
})
$img.css({'width': '', 'height': ''}).attr('src', url_list[index]).hide()
timer = setInterval(()=>{
const w = $img.width()
const h = $img.height()
if(w > 0) {
w > h ? $img.css({'width': '100%', 'height': 'auto'}) : $img.css({'height': '100%', 'width': 'auto'})
$img.show()
$('#loader').hide()
clearInterval(timer)
}
}, 1)
}
//当前图片
renderImg(current_index)
$img.mousedown(function (e) {
let endx = 0;
let endy = 0;
let left = parseInt($imgContainer.css("left"))
let top = parseInt($imgContainer.css("top"))
let downx = e.pageX
let downy = e.pageY
e.preventDefault()
$(document).on("mousemove", function (es) {
let endx = es.pageX - downx + left
let endy = es.pageY - downy + top
$imgContainer.css("left", endx + "px").css("top", endy + "px")
return false
});
})
$img.mouseup(function () {$(document).unbind("mousemove")})
$imgContainer.append($img)
$imgBox.append($imgContainer)
$imgBox.click(function(e){!$(e.target).hasClass('hld__img') && $(this).remove()})
$imgBox.append(``)
$imgBox.on('click', '.change', function(){
if($(this).hasClass('prev-img') && current_index - 1 >= 0)
renderImg(--current_index)
if($(this).hasClass('next-img') && current_index + 1 < url_list.length)
renderImg(++current_index)
if($(this).hasClass('rotate-right') || $(this).hasClass('rotate-left')) {
let deg = ($img.data('rotate-deg') || 0) - ($(this).hasClass('rotate-right') ? 90 : -90)
if(deg >= 360 || deg <= -360) deg = 0
$img.css('transform', `rotate(${deg}deg)`)
$img.data('rotate-deg', deg)
}else {
$img.css('transform', '')
$img.data('rotate-deg', 0)
}
window.getSelection ? window.getSelection().removeAllRanges() : document.selection.empty()
return false;
})
$imgBox.on("mousewheel DOMMouseScroll", function (e) {
const delta = (e.originalEvent.wheelDelta && (e.originalEvent.wheelDelta > 0 ? 1 : -1))||
(e.originalEvent.detail && (e.originalEvent.detail > 0 ? -1 : 1));
if($imgContainer.width() > 50) {
const offset_y = $imgContainer.height() * 0.2
const offset_x = $imgContainer.width() * 0.2
let offset_top = offset_y / 2
let offset_left = offset_x / 2
if($(e.target).hasClass('hld__zoom-target')) {
const target_offset_x = Math.round(e.clientX - $imgContainer.position().left)
const target_offset_y = Math.round(e.clientY - $imgContainer.position().top)
offset_left = (target_offset_x / ($imgContainer.height() / 2)) * offset_left
offset_top = (target_offset_y / ($imgContainer.height() / 2)) * offset_top
}
if(delta > 0) {
$imgContainer.css({
'width': ($imgContainer.height() + offset_y) + 'px',
'height': ($imgContainer.height() + offset_y) + 'px',
'top': ($imgContainer.position().top - offset_top) + 'px',
'left': ($imgContainer.position().left - offset_left) + 'px'
})
}
if(delta < 0) {
$imgContainer.css({
'width': ($imgContainer.height() - offset_y) + 'px',
'height': ($imgContainer.height() - offset_y) + 'px',
'top': ($imgContainer.position().top + offset_top) + 'px',
'left': ($imgContainer.position().left + offset_left) + 'px'
})
}
}
e.stopPropagation()
return false
})
$('body').append($imgBox)
}
//新页面打开连接
setting.linkTargetBlank && $('.topic').attr('target', '_blank')
const runMark = () => {
$('.topicrow .author[hld-render!=ok]').each(function(){
ban_list.includes($(this).text()) && $(this).parents('tbody').remove()
for(let m of mark_list) {
const t = m.split(':')
if(t[0] == $(this).text()) {
$(this).html(`${t[1]}(${$(this).text()})`)
}
}
//添加标志位
$(this).attr('hld-render', 'ok')
})
}
const runDom = () => {
//楼内
$('.forumbox.postbox[hld-render!=ok]').each(function(){
//隐藏头像
setting.hideAvatar && $(this).find('.avatar').css('display', 'none')
//隐藏表情
$(this).find('img').each(function(){
const classs = $(this).attr('class');
if(classs && classs.includes('smile')) {
const alt = $(this).attr('alt')
const $alt = $('[' + alt + ']')
setting.hideSmile ? $(this).hide() : $alt.hide()
$(this).after($alt)
}else if(!classs && $(this).attr('onload')) {
$(this).attr('hld__imglist', 'ready')
if(setting.imgResize) {
$(this).width() > 200 && $(this).css({'outline': '', 'outline-offset': '', 'cursor': 'pointer', 'min-width': '200px', 'min-height': 'auto', 'width': '200px', 'height': 'auto', 'margin:': '5px'})
}
let $imgB = $('')
$imgB.on('click', function(){
$(this).prev('img').toggle()
$(this).text($(this).prev('img').is(':hidden') ? '图' : '隐藏')
})
$(this).removeAttr('onload')
if(setting.hideImage) {
$(this).hide();
$imgB.show()
}
$(this).after($imgB)
}
})
//隐藏签名
setting.hideSign && $(this).find('.sign, .sigline').css('display', 'none')
//添加拉黑标记菜单及功能
if(setting.markAndBan) {
$(this).find('.small_colored_text_btn.block_txt_c2.stxt').each(function(){
let current_user = ''
if($(this).parents('td').prev('td').html() == '') {
current_user = $(this).parents('table').prev('.posterinfo').children('.author').text()
}else {
current_user = $(this).parents('td').prev('td').find('.author').text()
}
$(this).append(``)
})
}
//标记拉黑备注
setting.authorMark && markDom($(this))
//添加标志位
$(this).attr('hld-render', 'ok')
})
}
const reRender = () => {
$('.forumbox.postbox').each(function(){
markDom($(this))
})
}
const markDom = $el => {
$el.find('.b').each(function(){
$(this).find('span.hld__post-author, span.hld__remark').remove()
let name = $(this).attr('hld-mark-before-name') || $(this).text().replace('[', '').replace(']', '')
if(setting.markAndBan) {
if(ban_list.includes(name)) {
if(setting.banMode == 'STRICT') {
if($(this).parents('div.comment_c').length > 0) $(this).parents('div.comment_c').remove()
else $(this).parents('.forumbox.postbox').remove()
}else {
if($(this).hasClass('author')) {
if($(this).parents('div.comment_c').length > 0) $(this).parents('div.comment_c').remove()
else $(this).parents('.forumbox.postbox').remove()
}else {
$(this).parent().html('此用户在你的黑名单中,已屏蔽其言论')
}
}
}
for(let m of mark_list) {
const t = m.split(':')
if(t[0] == name) {
$(this).attr('hld-mark-before-name', name).append(``)
}
}
}
if(name == $('#hld__post-author').val() && $(this).find('span.hld__post-author').length == 0)
$(this).append('[楼主]')
})
}
if(setting.imgResize) {
$('#m_posts').on('click', '.postcontent img[hld__imglist=ready]', function(){
resizeImg($(this))
})
}
//设置面板
let $panel_dom = $(``)
$('body').append($panel_dom)
//本地恢复设置
for(let k in setting) {
if($('#hld__cb_' + k).length > 0) {
$('#hld__cb_' + k)[0].checked = setting[k]
const enable_dom_id = $('#hld__cb_' + k).attr('enable')
if(enable_dom_id) {
setting[k] ? $('#'+enable_dom_id).show() : $('#'+enable_dom_id).hide()
$('#'+enable_dom_id).find('input').each(function(){
$(this).val() == setting[$(this).attr('name').substr(8)] && ($(this)[0].checked = true)
})
$('#hld__cb_' + k).on('click', function(){
$(this)[0].checked ? $('#'+enable_dom_id).slideDown() : $('#'+enable_dom_id).slideUp()
})
}
}
}
//导出设置
$('body').on('click', '#hld__export__data', function(){
let obj = {
name: 'NGA-BBS',
setting: setting,
ban_list: ban_list,
mark_list: mark_list
}
window.prompt('导出成功,请复制以下代码以备份', Base64.encode(JSON.stringify(obj)))
})
//导入
$('body').on('click', '#hld__import__data', function(){
let base_str = window.prompt('导入字符串\n注意,导入会覆盖你当前所有的设置以及名单列表!', '')
base_str = $.trim(base_str)
if(base_str) {
let str = Base64.decode(base_str)
if(str) {
let obj
try {
obj = JSON.parse(str)
setting = obj.setting
ban_list = obj.ban_list
mark_list = obj.mark_list
window.localStorage.setItem('hld__NGA_setting', JSON.stringify(setting))
window.localStorage.setItem('hld__NGA_ban_list', ban_list.join(','))
window.localStorage.setItem('hld__NGA_mark_list', mark_list.join(','))
$panel_dom.hide()
alert('导入成功,刷新生效')
}catch(err){
alert('配置有误,导入失败')
}
}
}
})
//保存
$('body').on('click', '#hld__save__data', function(){
for(let k in setting) {
$('input#hld__cb_' + k).length > 0 && (setting[k] = $('input#hld__cb_' + k)[0].checked)
$(`input[name="hld__rb_${k}"]`).length > 0 && (setting[k] = $(`input[name="hld__rb_${k}"]:checked`).val())
}
window.localStorage.setItem('hld__NGA_setting', JSON.stringify(setting))
$panel_dom.hide()
popMsg('保存成功,刷新页面生效')
})
//消息
const popMsg = (msg) => {
alert(msg)
}
function GetQueryString(name) {
var url = decodeURI(window.location.search.replace(/&/g, "&"));
var reg = new RegExp("(^|&)" + name + "=([^&]*)(&|$)");
var r = url.substr(1).match(reg);
if(r != null) return unescape(r[2]);
return null;
}
function Uniq(array){
return [...new Set(array)]
}
function RemoveBlank(array) {
let r = [];
array.map(function(val, index) {
if (val !== '' && val != undefined) {
r.push(val);
}
});
return r;
}
const Base64 = {
encode: (str) => {
return window.btoa(unescape(encodeURIComponent(str)))
},
decode: (str) => {
try {
return decodeURIComponent(escape(window.atob(str)))
}catch(err){
alert('字符串有误,导入失败')
}
}
}
//样式
let style = document.createElement("style")
style.type = "text/css"
style.appendChild(document.createTextNode(`
.postcontent img {
margin: 0 5px 5px 0;
}
#hld__img_full{
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: rgba(0, 0, 0, 0.6);
z-index: 99999;
/*display: flex;*/
/*align-items: center;*/
/*justify-content: center;*/
}
.hld__img_container {
position: absolute;
display: flex;
justify-content: center;
align-items: center;
}
#hld__img_full img{
cursor: move;
transition: transform .2s ease;
}
#hld__img_full .hld__imgcenter{
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
}
.hld__if_control {
position: absolute;
display:flex;
left:50%;
bottom: 15px;
width: 160px;
margin-left:-80px;
height: 40px;
background: rgba(0, 0, 0, 0.6);
z-index:9999999;
}
#hld__img_full .change {
width: 40px;
height: 40px;
cursor:pointer;
}
#hld__img_full .rotate-right,
#hld__img_full .rotate-left{
background:url() center no-repeat;
background-size: 25px;
}
#hld__img_full .rotate-right {
transform: rotateY(180deg);
}
#hld__img_full .rotate-left:hover {
transform: scale(1.2);
}
#hld__img_full .rotate-right:hover {
transform: scale(1.2) rotateY(180deg);
}
#hld__img_full .next-img:hover {
transform: scale(1.2) rotate(180deg);
}
#hld__img_full .prev-img,
#hld__img_full .next-img{
background:url() center no-repeat;
}
#hld__img_full .next-img {
transform: rotate(180deg);
}
#hld__img_full .prev-img:hover {
transform: scale(1.2);
}
#hld__img_full .next-img:hover {
transform: scale(1.2) rotate(180deg);
}
#hld__setting {
color:#6666CC;
}
#hld__banlist_panel {
position:fixed;
top:150px;
left:50%;
transform: translateX(-50%);
background:#fff8e7;
width:370px;
height:300px;
padding: 15px 20px;
border-radius: 10px;
box-shadow: 0 0 10px #666;
border: 1px solid #591804;
}
#hld__banlist_panel > div{
display:flex;
justify-content: space-between;
}
#hld__banlist_panel .hld__list-c{
width: 45%;
}
#hld__banlist_panel .hld__list-c textarea{
height:200px;
width:100%;
resize: none;
}
#hld__banlist_panel .hld__list-desc {
font-size:9px;
color:#666;
}
#hld__banlist_panel .hld__list-c > p:first-child{
weight:bold;
font-size:14px;
margin-bottom:10px;
}
#hld__setting_panel {
display:none;
position:fixed;
top:70px;
left:50%;
transform: translateX(-50%);
background:#fff8e7;
width:242px;
padding: 15px 20px;
border-radius: 10px;
box-shadow: 0 0 10px #666;
border: 1px solid #591804;
}
#hld__setting_panel p{
margin-bottom:10px;
}
#hld__setting_panel .hld__sp-title {
font-size: 15px;
font-weight: bold;
text-align: center;
}
#hld__setting_panel .hld__sp-section{
font-weight: bold;
margin-top: 20px;
}
.hld__setting-close{
position: absolute;
top: 5px;
right: 5px;
padding: 3px 6px;
background: #fff0cd;
color: #591804;
transition: all .2s ease;
cursor:pointer;
border-radius: 4px;
text-decoration: none;
}
.hld__setting-close:hover{
background: #591804;
color: #fff0cd;
text-decoration: none;
}
#hld__setting_panel button {
transition: all .2s ease;
cursor:pointer;
}
button.hld__btn {
padding: 3px 8px;
border: 1px solid #591804;
background: #fff8e7;
color: #591804;
}
button.hld__btn:hover {
background: #591804;
color: #fff0cd;
}
button.hld__save-btn {
margin-top:10px;
margin-left: 50%;
transform: translateX(-50%);
}
.hld__post-author {
color:#F00;
font-weight:bold;
}
.hld__extra-icon {
padding: 0 2px;
cursor:pointer;
background-repeat: no-repeat;
background-position: center;
}
.hld__extra-icon svg{
width:10px;
height:10px;
vertical-align: -0.15em;
fill: currentColor;
overflow: hidden;
}
.hld__extra-icon:hover{
text-decoration:none;
}
span.hld__remark{
color:#666;
font-size:0.8em;
}
span.hld__banned {
color:#ba2026;
}
.hld__sp-fold {
padding-left:23px;
}
.hld__sp-fold .hld__f-title {
font-weight:bold;
}
.hld__buttons {
display:flex;
justify-content: space-between;
}
#loader{
display:none;
position: absolute;
top: 50%;
left: 50%;
margin-top:-10px;
margin-left:-10px;
width: 20px;
height: 20px;
border: 6px dotted #FFF;
border-radius: 50%;
-webkit-animation: 1s loader linear infinite;
animation: 1s loader linear infinite;
}
@keyframes loader {
0% {
-webkit-transform: rotate(0deg);
transform: rotate(0deg);
}
100% {
-webkit-transform: rotate(360deg);
transform: rotate(360deg);
}
}
`))
document.getElementsByTagName("head")[0].appendChild(style)
})();