csdn_optimizer

make csdn better

Från och med 2022-09-03. Se den senaste versionen.

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==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();
})();