// ==UserScript==
// @name csdn_optimizer
// @namespace https://github.com/Kyouichirou
// @version 1.4.1
// @description make csdn better
// @author HLA
// @include /https:\/\/(\w+\.)?\w+\.csdn\.net.*/
// @grant unsafeWindow
// @grant GM_setValue
// @grant GM_getValue
// @grant GM_addStyle
// @grant GM_registerMenuCommand
// @grant GM_unregisterMenuCommand
// @grant GM_setClipboard
// @grant GM_notification
// @grant GM_openInTab
// @run-at document-start
// @license MIT
// @compatiable chrome; just test on chrome 80+
// @noframes
// @icon https://blog.csdn.net/favicon.ico
// ==/UserScript==
(() => {
"use strict";
const Notification = (
content = "",
title = "",
duration = 2500,
cfunc,
ofunc
) => {
GM_notification({
text: content,
title: title,
timeout: duration,
onclick: cfunc,
ondone: ofunc,
});
};
const escapeHTML_Blank = (html) => html.replace(/\s/g, " ");
const CSDN = {
input: {
get article() {
const bgp = ``;
return `
div#article_content{
height: auto !important;
}
.article-header-box {background-color: transparent !important;}
.blog-content-box {background-image: url(${bgp}) !important;}
.comment-box .comment-list-container .comment-list-box {
overflow: auto !important;
max-height: none !important;
}
#content_views pre code,
#content_views pre{user-select: unset !important;}
#content_views span {background-color: transparent !important;}
.toolbar-advert,
div.hide-article-box,
#csdn-redpack,
div#recommendAdBox,
.hljs-button.signin,
#blogColumnPayAdvert,
.opt-box.text-center,
img.comment-sofa-flag,
.recommend-item-box.recommend-download-box.clearfix,
.recommend-item-box.type_download.clearfix,
.recommend-item-box.type_course.clearfix,
.recommend-item-box.recommend-other-item-box.type_download,
.recommend-item-box.recommend-other-item-box.type_discussion_topic,
.recommend-item-box.recommend-other-item-box.type_course,
.recommend-item-box.type_download.clearfix,
.recommend-item-box.baiduSearch,`;
},
placeholder(mode) {
GM_addStyle(`
input::-webkit-input-placeholder {
font-size: 0px;
text-align: right;
}
${mode ? this.article : ""}
.login-box,
.login-mark,
.passport-login-container,
.csdn-side-toolbar,
.toolbar-search-hot {
display: none !important;
}
`);
},
dropmenu() {
let input = document.getElementsByClassName(
"toolbar-search onlySearch"
);
if (input.length > 0) {
const m = new MutationObserver((records) => {
for (const e of records) {
if (e.addedNodes.length > 0) {
for (const a of e.addedNodes) {
if (
a.className ===
"toolbar-search-drop-menu "
) {
a.remove();
break;
}
}
break;
}
}
});
m.observe(input[0], { childList: true });
unsafeWindow.addEventListener(
"visibilitychange",
(e) => {
e.preventDefault();
e.stopPropagation();
},
true
);
let boxs = input[0].getElementsByTagName("input");
let buttons = input[0].getElementsByTagName("button");
if (boxs.length > 0 && buttons.length > 0) {
const box = boxs[0];
buttons[0].addEventListener(
"click",
(e) => {
if (!box.value) {
e.preventDefault();
e.stopImmediatePropagation();
}
},
true
);
box.addEventListener(
"keydown",
(e) => {
if (!box.value) {
e.preventDefault();
e.stopImmediatePropagation();
}
},
true
);
buttons = null;
boxs = null;
}
input = null;
}
},
},
antiRedirect() {
const links = Object.getOwnPropertyDescriptors(
HTMLAnchorElement.prototype
).href;
const trackids = ["?utm_", "utm_source", "?ops_request"];
Object.defineProperty(HTMLAnchorElement.prototype, "href", {
...links,
get() {
const href = decodeURIComponent(links.get.call(this));
for (const id of trackids) {
const tmp = href.split(id);
if (tmp.length > 1) {
this.href = tmp[1];
return tmp[1];
}
}
return href;
},
});
},
antiLeech() {
// if those pictures are not from csdn
const content = document.getElementById("content_views");
if (!content) return;
const imgs = content.getElementsByTagName("img");
const hosts = ["csdnimg.cn", "csdn.net"];
for (const img of imgs) {
const src = img.src;
if (src && !hosts.some((e) => src.includes(e))) {
img.setAttribute("referrerPolicy", "no-referrer");
img.src = img.src + "?";
}
}
},
comment: {
// covert the link of text to href
textTolink(text) {
const reg =
/((http|https):\/\/[-A-Za-z0-9+&@#/%?=~_|!:,.;]+[-A-Za-z0-9+&@#/%=~_|])/g;
return reg.test(text)
? text.replace(
reg,
"<a href='$1' target='_blank' style='color: #00a0e9;width: 360px;display: inline;'>$1</a>"
)
: null;
},
text_convertor() {
const snc = document.getElementsByClassName("new-comment");
for (const e of snc) {
if (!e.innerHTML.includes("href")) {
const s = this.textTolink(e.textContent);
s && (e.innerHTML = s);
}
}
},
},
menus: {
s_id: null,
r_id: null,
n_id: null,
reader_style: null,
simple_style: null,
create_simple_style(m = true) {
if (this.simple_style) return;
this.reset_reader();
let i = 7;
let t = "";
const none_display = `
div#asideCustom,
div#asideNewComments,
div#asideNewNps{display: none !important;}`;
const sider_bar = `
aside.blog_container_aside,
div#rightAside{opacity: 0.1;}
aside.blog_container_aside:hover{
opacity: 1;
transition: opacity 3s;
}
div#rightAside:hover{
opacity: 1;
transition: opacity 3s;
}`;
for (i; i > 1; i--)
t += `.csdn-toolbar-fl.toolbar-menus > li:nth-of-type(${i}),`;
this.simple_style = GM_addStyle(
t +
".toolbar-btn.toolbar-btn-vip.csdn-toolbar-fl {visibility: hidden !important;}" +
none_display +
(m ? sider_bar : "")
);
},
create_reader_style() {
if (this.reader_style) return;
this.reset_simple();
const sty = `
#csdn-toolbar,
aside.blog_container_aside,
.aside-box.kind_person.d-flex.flex-column,
.comment-box,
.first-recommend-box,
.second-recommend-box,
.recommend-box.insert-baidu-box,
.recommend-tit-mod {display: none ! important;}
`;
this.reader_style = GM_addStyle(sty);
},
simple_action() {
if (this.s_id) {
GM_unregisterMenuCommand(this.s_id);
this.s_id = null;
}
GM_setValue("mode", "s");
this.create_simple_style();
this.reader();
this.normal();
},
reader_action() {
if (this.r_id) {
GM_unregisterMenuCommand(this.r_id);
this.r_id = null;
}
GM_setValue("mode", "r");
this.create_reader_style();
this.simple();
this.normal();
},
reset_simple() {
if (this.simple_style) {
this.simple_style.remove();
this.simple_style = null;
}
},
reset_reader() {
if (this.reader_style) {
this.reader_style.remove();
this.reader_style = null;
}
},
normal_action() {
if (this.n_id) {
GM_unregisterMenuCommand(this.n_id);
this.n_id = null;
}
GM_setValue("mode", "");
this.reset_reader();
this.reset_simple();
this.simple();
this.reader();
},
simple() {
!this.s_id &&
(this.s_id = GM_registerMenuCommand(
"Simple Mode",
this.simple_action.bind(this)
));
},
reader() {
!this.r_id &&
(this.r_id = GM_registerMenuCommand(
"Reader Mode",
this.reader_action.bind(this)
));
},
normal() {
!this.n_id &&
(this.n_id = GM_registerMenuCommand(
"Normal Mode",
this.normal_action.bind(this)
));
},
create(m) {
const mode = GM_getValue("mode");
if (!m && mode) {
this.create_simple_style(m);
return;
}
if (mode) {
if (mode === "r") {
this.simple();
this.create_reader_style();
} else {
this.reader();
this.create_simple_style();
}
this.normal();
} else {
this.simple();
this.reader();
}
},
},
code: {
// ctrl + right mouse => copy the code directly
clipboard(text) {
if (!text) {
Notification("failed to get the content of code");
return;
}
try {
try {
navigator.clipboard.writeText(text);
} catch (err) {
console.log(err);
GM_setClipboard(text, "text");
}
Notification("the code has been copied to clipboard");
} catch (err) {
console.log(err);
Notification("some error on copy the code");
}
},
get_code(node) {
const num = node.getElementsByClassName("pre-numbering");
if (num.length > 0) {
let text = "";
for (const e of node.childNodes) {
if (e.className === "pre-numbering") continue;
text += e.innerText;
}
return text;
}
return node.innerText;
},
event() {
const pres = document.getElementsByTagName("pre");
for (const pre of pres)
pre.title = "Ctrl + Right mouse button to copy this code";
pres.length > 0 &&
document.addEventListener("contextmenu", (e) => {
if (e.ctrlKey) {
for (const p of e.path) {
if (p.localName === "pre") {
e.preventDefault();
e.stopImmediatePropagation();
const text = this.get_code(p);
this.clipboard(text);
break;
}
}
}
});
},
},
anti_prevent_copy() {
// disable initialization of csdn copy event(this feature will add copyrigth content to the data of copy)
unsafeWindow.csdn = {};
Object.defineProperty(unsafeWindow.csdn, "copyright", {
value: null,
});
},
anti_click_redirect() {
// anti track what you click
document.addEventListener(
"click",
(e) => {
let ic = 0;
for (const a of e.path) {
if (a.localName === "a") {
e.preventDefault();
e.stopImmediatePropagation();
const href = a.href;
if (href) {
const trackids = [
"?utm_",
"utm_source",
"?ops_request",
];
const index = trackids.findIndex((e) =>
href.includes(e)
);
GM_openInTab(
index < 0 ? href : href.slice(0, index),
{
insert: true,
active: true,
}
);
}
break;
} else if (ic > 2) break;
ic++;
}
},
true
);
},
clear_bottom() {
// clear the bottom zone that displays some relative articles
const b = document.getElementsByClassName(
"recommend-box insert-baidu-box"
);
const rubbish = [
"\u534e\u4e3a",
"CSDN\u8d44\u8baf",
"CSDNnew",
"\u5927\u5b66\u751f",
"\u5e94\u5c4a\u751f",
"\u6bd5\u4e1a\u751f",
"\u79c1\u6d3b",
"\u6708\u85aa",
"\u5e74\u85aa",
"\u8df3\u69fd",
"\u5de5\u8d44",
"\u4e8b\u4e1a",
"\u5de5\u7a0b\u5e08",
"\u85aa\u8d44",
"\u517c\u804c",
"\u7c89\u4e1d",
"\u627e\u5de5\u4f5c",
"\u7b80\u5386",
"\u9762\u8bd5",
"\u540c\u4e8b",
"\u4f17\u5305",
"HR",
"\u85aa\u6c34",
"\u5916\u5305",
"\u79bb\u804c",
"\u8fbe\u5185",
];
if (b.length > 0) {
const items = b[0].getElementsByClassName(
"recommend-item-box type_blog clearfix"
);
let i = items.length;
const trackids = ["?utm_", "utm_source", "?ops_request"];
for (i; i--; ) {
const text = items[i].innerText;
if (text && rubbish.some((e) => text.includes(e))) {
items[i].remove();
continue;
}
const links = items[i].getElementsByTagName("a");
let fr = false;
for (const a of links) {
let href = a.href;
if (href && href.startsWith("http")) {
const index = trackids.findIndex((e) =>
href.includes(e)
);
index > 0 && (href = href.slice(0, index));
if (href.includes("/article/") || !this.blackLists)
continue;
const i = href.lastIndexOf("/");
if (i > 0) {
const id = href.slice(i + 1);
if (this.blackLists.includes(id)) {
fr = true;
break;
}
}
index > 0 && (a.href = href);
}
}
fr && items[i].remove();
}
}
this.blackLists = null;
},
user_manage: {
// block some rubbish author
create_button(mode) {
const postion = document.getElementsByClassName(
"user-profile-operate-btn"
);
const [color, name, title] = mode
? ["black", "unBlock", "unblock this author"]
: ["red", "Block", "block this author"];
if (postion.length === 0) {
const button = `
<button
style="
margin-left: 158px;
width: 65px;
opacity: 0.95;
font-size: 14px;
line-height: 20px;
text-align: center;
cursor: pointer;
color: ${color};
background: none;
border: 1px solid;
border-radius: 3px;
"
title=${escapeHTML_Blank(title)}
>
${name}
</button>`;
const user = document.getElementsByClassName(
"user-info d-flex flex-column profile-intro-name-box"
);
if (user.length === 0)
console.log("failed to get the postion of button");
else {
user[0].insertAdjacentHTML("beforeend", button);
setTimeout(
() =>
(user[0].lastElementChild.onclick = (e) =>
this.click_event(e.target)),
250
);
}
} else {
const button = `
<button class="block" style="
width: 92px;
height: 32px;
color: ${color};
font-size: 15px;
cursor: pointer;
border-radius: 20px;
opacity: 0.95;
border: 1px solid #999aaa;
" title=${escapeHTML_Blank(title)}>${name}</button>`;
postion[0].insertAdjacentHTML("afterbegin", button);
setTimeout(
() =>
(postion[0].firstElementChild.onclick = (e) =>
this.click_event(e.target)),
350
);
}
},
click_event(target) {
debugger;
let blackLists = GM_getValue("block");
if (target.innerText === "Block") {
if (blackLists) {
if (!blackLists.includes(this.id)) {
blackLists.push(this.id);
GM_setValue("block", blackLists);
}
} else {
blackLists = [];
blackLists.push(this.id);
GM_setValue("block", blackLists);
}
target.innerText = "unBlock";
target.title = "unblock this author";
target.style.color = "black";
} else {
if (blackLists) {
const index = blackLists.indexOf(this.id);
if (index > -1) {
blackLists.splice(index, 1);
GM_setValue("block", blackLists);
}
}
target.innerText = "Block";
target.title = "block this author";
target.style.color = "red";
}
},
id: null,
main(mode, id) {
this.create_button(mode);
this.id = id;
},
},
blackLists: null,
get_block() {
this.blackLists = GM_getValue("block");
},
start() {
const href = location.href;
const f = href.includes("/article/");
this.antiRedirect();
this.input.placeholder(f);
this.anti_prevent_copy();
//unsafewindow, some page can not capture the event.
window.onload = () => {
if (f) {
this.get_block();
this.antiLeech();
this.code.event();
this.anti_click_redirect();
this.comment.text_convertor();
setTimeout(() => this.clear_bottom(), 330);
} else {
const regs = [
/https:\/\/blog\.csdn\.net\/(\w+)($|\/|\/category\w+\.html|\?type=[a-z]+)/,
/https:\/\/(\w+)\.blog\.csdn\.net(\/|$)/,
];
let id = null;
for (const reg of regs) {
const ms = href.match(reg);
if (ms && ms.length > 1) {
id = ms[1];
break;
}
}
if (id) {
this.get_block();
const mode =
this.blackLists && this.blackLists.includes(id);
this.user_manage.main(mode, id);
this.blackLists = null;
this.anti_click_redirect();
} else if (href.endsWith(".net/"))
this.anti_click_redirect();
}
this.input.dropmenu();
};
this.menus.create(f);
},
};
CSDN.start();
})();