// ==UserScript==
// @name DeepFlood X
// @namespace http://www.deepflood.com/
// @version 0.4.1
// @description 【原DeepFlood增强】自动签到、无缝翻页帖子评论、快捷回复、代码高亮、屏蔽用户、屏蔽帖子、楼主低等级提醒
// @author dabao
// @match *://*.deepflood.com/*
// @icon 
// @require https://s4.zstatic.net/ajax/libs/layui/2.9.9/layui.min.js
// @resource highlightStyle https://s4.zstatic.net/ajax/libs/highlight.js/11.9.0/styles/atom-one-light.min.css
// @resource highlightStyle_dark https://s4.zstatic.net/ajax/libs/highlight.js/11.9.0/styles/atom-one-dark.min.css
// @grant GM_xmlhttpRequest
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_deleteValue
// @grant GM_notification
// @grant GM_registerMenuCommand
// @grant GM_unregisterMenuCommand
// @grant GM_getResourceURL
// @grant GM_addElement
// @grant GM_addStyle
// @grant GM_openInTab
// @grant unsafeWindow
// @run-at document-end
// @license GPL-3.0 License
// @supportURL https://www.deepflood.com/
// @homepageURL https://www.deepflood.com/
// ==/UserScript==
(function () {
'use strict';
const { version, author, name, icon } = GM_info.script;
// 适配新站点
const BASE_URL = "https://www.deepflood.com";
const util = {
clog:(c) => {
console.group(`%c %c [${name}]-v${version} by ${author}`, `background:url(${icon}) center/12px no-repeat;padding:3px`, "");
console.log(c);
console.groupEnd();
},
getValue: (name, defaultValue) => GM_getValue(name, defaultValue),
setValue: (name, value) => GM_setValue(name, value),
sleep: (ms) => new Promise((resolve) => setTimeout(resolve, ms)),
addStyle(id, tag, css) {
tag = tag || 'style';
let doc = document, styleDom = doc.head.querySelector(`#${id}`);
if (styleDom) return;
let style = doc.createElement(tag);
style.rel = 'stylesheet';
style.id = id;
tag === 'style' ? style.innerHTML = css : style.href = css;
doc.head.appendChild(style);
},
removeStyle(id,tag){
tag = tag || 'style';
let doc = document, styleDom = doc.head.querySelector(`#${id}`);
if (styleDom) { doc.head.removeChild(styleDom) };
},
getAttrsByPrefix(element, prefix) {
return Array.from(element.attributes).reduce((acc, { name, value }) => {
if (name.startsWith(prefix)) acc[name] = value;
return acc;
}, {});
},
data(element, key, value) {
if (arguments.length < 2) return undefined;
if (value !== undefined) element.dataset[key] = value;
return element.dataset[key];
},
async post(url, data, headers, responseType = 'json') {
return this.fetchData(url, 'POST', data, headers, responseType);
},
async get(url, headers, responseType = 'json') {
return this.fetchData(url, 'GET', null, headers, responseType);
},
async fetchData(url, method='GET', data=null, headers={}, responseType='json') {
const options = {
method,
headers: { 'Content-Type':'application/json',...headers},
body: data ? JSON.stringify(data) : undefined
};
const response = await fetch(url.startsWith("http") ? url : BASE_URL + url, options);
const result = await response[responseType]().catch(() => null);
return response.ok ? result : Promise.reject(result);
},
getCurrentDate() {
const localTimezoneOffset = (new Date()).getTimezoneOffset();
const beijingOffset = 8 * 60;
const beijingTime = new Date(Date.now() + (localTimezoneOffset + beijingOffset) * 60 * 1000);
const timeNow = `${beijingTime.getFullYear()}/${(beijingTime.getMonth() + 1)}/${beijingTime.getDate()}`;
return timeNow;
},
createElement(tagName, options = {}, childrens = [], doc = document, namespace = null) {
if (Array.isArray(options)) {
if (childrens.length !== 0) {
throw new Error("If options is an array, childrens should not be provided.");
}
childrens = options;
options = {};
}
const { staticClass = '', dynamicClass = '', attrs = {}, on = {} } = options;
const ele = namespace ? doc.createElementNS(namespace, tagName) : doc.createElement(tagName);
if (staticClass) {
staticClass.split(' ').forEach(cls => ele.classList.add(cls.trim()));
}
if (dynamicClass) {
dynamicClass.split(' ').forEach(cls => ele.classList.add(cls.trim()));
}
Object.entries(attrs).forEach(([key, value]) => {
if (key === 'style' && typeof value === 'object') {
Object.entries(value).forEach(([styleKey, styleValue]) => {
ele.style[styleKey] = styleValue;
});
} else {
if (value !== undefined) ele.setAttribute(key, value);
}
});
Object.entries(on).forEach(([event, handler]) => {
ele.addEventListener(event, handler);
});
childrens.forEach(child => {
if (typeof child === 'string') {
child = doc.createTextNode(child);
}
ele.appendChild(child);
});
return ele;
},
b64DecodeUnicode(str) {
return decodeURIComponent(atob(str).split('').map(function (c) {
return '%' + ('00' + c.charCodeAt(0).toString(16)).slice(-2);
}).join(''));
}
};
const opts = {
post: {
pathPattern: /^\/(categories\/|page|award|search|$)/,
scrollThreshold: 1500,
nextPagerSelector: '.nsk-pager a.pager-next',
postListSelector: 'ul.post-list:not(.topic-carousel-panel)',
topPagerSelector: 'div.nsk-pager.pager-top',
bottomPagerSelector: 'div.nsk-pager.pager-bottom',
},
comment: {
pathPattern: /^\/post-/,
scrollThreshold: 690,
nextPagerSelector: '.nsk-pager a.pager-next',
postListSelector: 'ul.comments',
topPagerSelector: 'div.nsk-pager.post-top-pager',
bottomPagerSelector: 'div.nsk-pager.post-bottom-pager',
},
setting: {
SETTING_SIGN_IN_STATUS: 'setting_sign_in_status',
SETTING_SIGN_IN_LAST_DATE: 'setting_sign_in_last_date',
SETTING_SIGN_IN_IGNORE_DATE: 'setting_sign_in_ignore_date',
SETTING_AUTO_LOADING_STATUS: 'setting_auto_loading_status'
},
settings:{
"version": version,
"sign_in": { "enabled": true, "method": 0, "last_date": "", "ignore_date": "" },
"signin_tips": { "enabled": true },
"re_signin": { "enabled": true },
"auto_jump_external_links": { "enabled": true },
"loading_post": { "enabled": true },
"loading_comment": { "enabled": true },
"quick_comment": { "enabled": true },
"open_post_in_new_tab": { "enabled": false },
"block_members": { "enabled": true },
"block_posts": { "enabled": true,"keywords":[] },
"level_tag": { "enabled": true, "low_lv_alarm":false, "low_lv_max_days":30 },
"code_highlight": { "enabled": true },
"image_slide":{ "enabled":true },
"visited_links":{ "enabled": true, "link_color":"","visited_color":"","dark_link_color":"","dark_visited_color":"" },
"user_card_ext": { "enabled":true }
}
};
layui.use(function () {
const layer = layui.layer;
const dropdown = layui.dropdown;
const message = {
info: (text) => message.__msg(text, { "background-color": "#4D82D6" }),
success: (text) => message.__msg(text, { "background-color": "#57BF57" }),
warning: (text) => message.__msg(text, { "background-color": "#D6A14D" }),
error: (text) => message.__msg(text, { "background-color": "#E1715B" }),
__msg: (text, style) => { let index = layer.msg(text, { offset: 't', area: ['100%', 'auto'], anim: 'slideDown' }); layer.style(index, Object.assign({ opacity: 0.9 }, style)); }
};
const Config = {
// 初始化配置数据
initValue() {
const value = [
{ name: opts.setting.SETTING_SIGN_IN_STATUS, defaultValue: 0 },
{ name: opts.setting.SETTING_SIGN_IN_LAST_DATE, defaultValue: '1753/1/1' },
{ name: opts.setting.SETTING_SIGN_IN_IGNORE_DATE, defaultValue: '1753/1/1' },
{ name: opts.setting.SETTING_AUTO_LOADING_STATUS, defaultValue: 1 },
{ name: 'open_post_in_new_tab', defaultValue: 0 },
{ name: 'feedback', defaultValue: 0 }
];
this.upgradeConfig();
value.forEach((v) => util.getValue(v.name) === undefined && util.setValue(v.name, v.defaultValue));
},
// 升级配置项
upgradeConfig() {
const upgradeConfItem = (oldConfKey, newConfKey) => {
if (util.getValue(oldConfKey) && util.getValue(newConfKey) === undefined) {
util.clog(`升级配置项 ${oldConfKey} 为 ${newConfKey}`);
util.setValue(newConfKey, util.getValue(oldConfKey));
GM_deleteValue(oldConfKey);
}
};
upgradeConfItem('menu_signInTime', opts.setting.SETTING_SIGN_IN_LAST_DATE);
},
initializeConfig() {
const defaultConfig = opts.settings;
if (!util.getValue('settings')) {
util.setValue('settings', defaultConfig);
return;
}
if(this.getConfig('version')===version) return;
let storedConfig = util.getValue('settings');
const cleanDefaults = (stored, defaults) => {
Object.keys(stored).forEach(key => {
if (defaults[key] === undefined) {
delete stored[key];
} else if (typeof stored[key] === 'object' && stored[key] !== null && !(stored[key] instanceof Array)) {
cleanDefaults(stored[key], defaults[key]);
}
});
};
const mergeDefaults = (stored, defaults) => {
Object.keys(defaults).forEach(key => {
if (typeof defaults[key] === 'object' && defaults[key] !== null && !(defaults[key] instanceof Array)) {
if (!stored[key]) stored[key] = {};
mergeDefaults(stored[key], defaults[key]);
} else {
if (stored[key] === undefined) {
stored[key] = defaults[key];
}
}
});
};
mergeDefaults(storedConfig, defaultConfig);
cleanDefaults(storedConfig, defaultConfig);
storedConfig.version = version;
util.setValue('settings',storedConfig);
},updateConfig(path, value) {
let config = util.getValue('settings');
let keys = path.split('.');
let lastKey = keys.pop();
let lastObj = keys.reduce((obj, key) => obj[key], config);
lastObj[lastKey] = value;
util.setValue('settings', config);
},getConfig(path) {
let config = GM_getValue('settings');
let keys = path.split('.');
return keys.reduce((obj, key) => obj[key], config);
}
};
const FeatureFlags={
isEnabled(featureName) {
if (Config.getConfig(featureName)) {
return Config.getConfig(`${featureName}.enabled`);
} else {
console.error(`Feature '${featureName}' does not exist.`);
return false;
}
}
};
const main = {
loginStatus: false,
//检查是否登陆
checkLogin() {
if (unsafeWindow.__config__ && unsafeWindow.__config__.user) {
this.loginStatus = true;
util.clog(`当前登录用户 ${unsafeWindow.__config__.user.member_name} (ID ${unsafeWindow.__config__.user.member_id})`);
}
},
// 自动签到
autoSignIn(rand) {
if(!FeatureFlags.isEnabled('sign_in')) return;
if (!this.loginStatus) return
if (util.getValue(opts.setting.SETTING_SIGN_IN_STATUS) === 0) return;
rand = rand || (util.getValue(opts.setting.SETTING_SIGN_IN_STATUS) === 1);
let timeNow = util.getCurrentDate(),
timeOld = util.getValue(opts.setting.SETTING_SIGN_IN_LAST_DATE);
if (!timeOld || timeOld != timeNow) {
util.setValue(opts.setting.SETTING_SIGN_IN_LAST_DATE, timeNow);
this.signInRequest(rand);
}
},
// 重新签到
reSignIn() {
if (!this.loginStatus) return;
if (util.getValue(opts.setting.SETTING_SIGN_IN_STATUS) === 0) {
unsafeWindow.mscAlert('提示', this.getMenuStateText(this._menus[0], 0) + ' 状态时不支持重新签到!');
return;
}
util.setValue(opts.setting.SETTING_SIGN_IN_LAST_DATE, '1753/1/1');
location.reload();
},
addSignTips() {
if(!FeatureFlags.isEnabled('signin_tips')) return;
if (!this.loginStatus) return
if (util.getValue(opts.setting.SETTING_SIGN_IN_STATUS) !== 0) return;
const timeNow = util.getCurrentDate();
const { SETTING_SIGN_IN_IGNORE_DATE, SETTING_SIGN_IN_LAST_DATE } = opts.setting;
const timeIgnore = util.getValue(SETTING_SIGN_IN_IGNORE_DATE);
const timeOld = util.getValue(SETTING_SIGN_IN_LAST_DATE);
if (timeNow === timeIgnore || timeNow === timeOld) return;
const _this = this;
let tip = util.createElement("div", { staticClass: 'nsplus-tip' });
let tip_p = util.createElement('p');
tip_p.innerHTML = '今天你还没有签到哦! 【<a class="sign_in_btn" data-rand="true" href="javascript:;">随机抽个鸡腿</a>】 【<a class="sign_in_btn" data-rand="false" href="javascript:;">只要5个鸡腿</a>】 【<a id="sign_in_ignore" href="javascript:;">今天不再提示</a>】';
tip.appendChild(tip_p);
tip.querySelectorAll('.sign_in_btn').forEach(function (item) {
item.addEventListener("click", function (e) {
const rand = util.data(this, 'rand');
_this.signInRequest(rand);
tip.remove();
util.setValue(SETTING_SIGN_IN_LAST_DATE, timeNow);
})
});
tip.querySelector('#sign_in_ignore').addEventListener("click", function (e) {
tip.remove();
util.setValue(SETTING_SIGN_IN_IGNORE_DATE, timeNow);
});
document.querySelector('header').append(tip);
},
async signInRequest(rand) {
await util.post('/api/attendance?random=' + (rand || false), {}, { "Content-Type": "application/json" }).then(json => {
if (json.success) {
message.success(`签到成功!今天午饭+${json.gain}个鸡腿; 积攒了${json.current}个鸡腿了`);
}
else {
message.info(json.message);
}
}).catch(error => {
message.info(error.message || "发生未知错误");
util.clog(error);
});
util.clog(`[${name}] 签到完成`);
},
is_show_quick_comment: false,
quickComment() {
if (!this.loginStatus || !opts.comment.pathPattern.test(location.pathname)) return;
if (util.getValue(opts.setting.SETTING_AUTO_LOADING_STATUS) === 0) return;
const _this = this;
const onClick = (e) => {
if (_this.is_show_quick_comment) {
return;
}
e.preventDefault();
const mdEditor = document.querySelector('.md-editor');
const clientHeight = document.documentElement.clientHeight, clientWidth = document.documentElement.clientWidth;
const mdHeight = mdEditor.clientHeight, mdWidth = mdEditor.clientWidth;
const top = (clientHeight / 2) - (mdHeight / 2), left = (clientWidth / 2) - (mdWidth / 2);
mdEditor.style.cssText = `position: fixed; top: ${top}px; left: ${left}px; margin: 30px 0px; width: 100%; max-width: ${mdWidth}px; z-index: 999;`;
const moveEl = mdEditor.querySelector('.tab-select.window_header');
moveEl.style.cursor = "move";
moveEl.addEventListener('mousedown', startDrag);
addEditorCloseButton();
_this.is_show_quick_comment = true;
};
const commentDiv = document.querySelector('#fast-nav-button-group #back-to-parent').cloneNode(true);
commentDiv.id = 'back-to-comment';
commentDiv.innerHTML = '<svg class="iconpark-icon" style="width: 24px; height: 24px;"><use href="#comments"></use></svg>';
commentDiv.addEventListener("click", onClick);
document.querySelector('#back-to-parent').before(commentDiv);
document.querySelectorAll('.nsk-post .comment-menu,.comment-container .comments').forEach(x=>x.addEventListener("click",(event) =>{ if(!["引用", "回复", "编辑"].includes(event.target.textContent)) return; onClick(event);},true));
function addEditorCloseButton() {
const fullScreenToolbar = document.querySelector('#editor-body .window_header > :last-child');
const cloneToolbar = fullScreenToolbar.cloneNode(true);
cloneToolbar.setAttribute('title', '关闭');
cloneToolbar.querySelector('span').classList.replace('i-icon-full-screen-one', 'i-icon-close');
cloneToolbar.querySelector('span').innerHTML = '<svg width="16" height="16" viewBox="0 0 48 48" fill="none"><path d="M8 8L40 40" stroke="currentColor" stroke-width="4" stroke-linecap="round" stroke-linejoin="round"></path><path d="M8 40L40 8" stroke="currentColor" stroke-width="4" stroke-linecap="round" stroke-linejoin="round"></path></svg>';
cloneToolbar.addEventListener("click", function (e) {
const mdEditor = document.querySelector('.md-editor');
mdEditor.style = "";
const moveEl = mdEditor.querySelector('.tab-select.window_header');
moveEl.style.cursor = "";
moveEl.removeEventListener('mousedown', startDrag);
this.remove();
_this.is_show_quick_comment = false;
});
fullScreenToolbar.after(cloneToolbar);
}
function startDrag(event) {
if (event.button !== 0) return;
const draggableElement = document.querySelector('.md-editor');
const parentMarginTop = parseInt(window.getComputedStyle(draggableElement).marginTop);
const initialX = event.clientX - draggableElement.offsetLeft;
const initialY = event.clientY - draggableElement.offsetTop + parentMarginTop;
document.onmousemove = function (event) {
const newX = event.clientX - initialX;
const newY = event.clientY - initialY;
draggableElement.style.left = newX + 'px';
draggableElement.style.top = newY + 'px';
};
document.onmouseup = function () {
document.onmousemove = null;
document.onmouseup = null;
};
}
}
}
main.init();
});
})();