zhihu optimizer

now, I can say this is the best GM script for zhihu!

Инсталирай този скрипт?
Препоръчано от автора

Може да харесате и csdn_optimizer.

Инсталирай този скрипт
// ==UserScript==
// @name         zhihu optimizer
// @namespace    https://github.com/Kyouichirou
// @version      3.5.4.3
// @description  now, I can say this is the best GM script for zhihu!
// @author       HLA
// @run-at       document-start
// @match        https://*.zhihu.com/*
// @grant        GM_addStyle
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_listValues
// @grant        unsafeWindow
// @grant        GM_xmlhttpRequest
// @grant        GM_deleteValue
// @grant        GM_unregisterMenuCommand
// @grant        GM_addValueChangeListener
// @grant        GM_removeValueChangeListener
// @grant        GM_registerMenuCommand
// @grant        GM_notification
// @grant        GM_openInTab
// @grant        GM_getTab
// @grant        GM_getTabs
// @grant        GM_download
// @grant        GM_saveTab
// @grant        GM_info
// @grant        window.onurlchange
// @grant        window.close
// @connect      www.zhihu.com
// @connect      lens.zhihu.com
// @connect      api.zhihu.com
// @connect      cn.bing.com
// @connect      img.meituan.net
// @connect      www.cnblogs.com
// @require      https://cdn.staticfile.org/xlsx/0.10.0/xlsx.core.min.js
// @icon         https://static.zhihu.com/heifetz/favicon.ico
// @compatible   chrome 80+
// @license      MIT
// @noframes
// @note         cent browser(https://www.centbrowser.com/) is recommended, and edeg browser is not recommended, too bloated and jumbled
// @note         just test on chrome 80+
// ==/UserScript==

(() => {
    "use strict";
    const Assist_info_URL = {
        shortcuts:
            "https://img.meituan.net/csc/df2540f418efadc25e0562df5924bb8b193354.png",
        usermanual:
            "https://github.com/Kyouichirou/D7E1293/blob/main/Tmapermonkey/zhihu_optimizer_manual.md",
        feedback:
            "https://greasyfork.org/zh-CN/scripts/420005-zhihu-optimizer/feedback",
        github: "https://github.com/Kyouichirou",
        greasyfork:
            "https://greasyfork.org/zh-CN/scripts/420005-zhihu-optimizer",
        cmd_help:
            "https://img.meituan.net/csc/5409e56911b74b0fa3e8e0e3fc40c62587055.png",
        search_help:
            "https://img.meituan.net/csc/29bae0a159923ec0c3f196326b6e3a2816319.png",
        Overview:
            "https://img.meituan.net/csc/083a417e5e990b04248baf5912a24ca2333972.png",
    };
    const blackKey = [
        "\u5171\u9752\u56e2",
        "\u4e60\u4e3b\u5e2d",
        "\u6bdb\u4e3b\u5e2d",
        "\u8096\u6218",
        "\u7559\u5b66\u4e2d\u4ecb",
        "\u65b0\u534e\u793e",
        "\u4eba\u6c11\u65e5\u62a5",
        "\u5149\u660e\u7f51",
        "\u5fb7\u4e91\u793e",
        "\u6597\u7f57\u5927\u9646",
        "\u592e\u89c6\u65b0\u95fb",
        "\u4eba\u6c11\u7684\u540d\u4e49",
        "\u5171\u4ea7\u515a",
        "\u5f20\u827a\u5174",
        "\u6c88\u9038",
        "\u6731\u4e00\u9f99",
        "\u8fea\u4e3d\u70ed\u5df4",
        "\u738b\u4e00\u535a",
        "\u6613\u70ca\u5343\u73ba",
        "\u91d1\u707f\u8363",
        "\u66fe\u4ed5\u5f3a",
        "\u738b\u4fca\u51ef",
        "\u9a81\u8bdd\u4e00\u4e0b",
        "\u90fd\u5e02\u5c0f\u8bf4",
        "\u8a00\u60c5\u5c0f\u8bf4",
        "\u803d\u7f8e\u5c0f\u8bf4",
        "\u6bdb\u6cfd\u4e1c",
    ];
    let blackName = null;
    let blackTopicAndQuestion = null;
    let collect_Answers = null;
    const Notification = (
        content = "",
        title = "",
        duration = 2500,
        cfunc,
        ofunc
    ) => {
        GM_notification({
            text: content,
            title: title,
            timeout: duration,
            onclick: cfunc,
            ondone: ofunc,
        });
    };
    const colorful_Console = {
        //this function is adpated from jianshu.com
        colors: {
            warning: "#F73E3E",
            Tips: "#327662",
            info: "#1475b2",
        },
        main(info, bc) {
            const t = info.title,
                c = info.content,
                a = [
                    "%c ".concat(t, " %c ").concat(c, " "),
                    "padding: 1px; border-radius: 3px 0 0 3px; color: #fff; font-size: 12px; background: ".concat(
                        "#606060",
                        ";"
                    ),
                    "padding: 1px; border-radius: 0 3px 3px 0; color: #fff; font-size: 12px; background: ".concat(
                        bc,
                        ";"
                    ),
                ];
            (function () {
                let e;
                window.console &&
                    "function" === typeof window.console.log &&
                    (e = console).log.apply(e, arguments);
            }.apply(null, a),
                a);
        },
    };
    const installTips = {
        //first time run, open the usermanual webpage
        e(mode) {
            Notification(
                mode
                    ? "important update"
                    : "thanks for installing, please read user manual carefully",
                "Tips",
                6000
            );
            GM_setValue("initial", "3.5");
            GM_setValue("installeddate", Date.now());
            GM_openInTab(Assist_info_URL.usermanual, { insert: true });
            setTimeout(
                () => GM_openInTab(Assist_info_URL.shortcuts, { insert: true }),
                300
            );
        },
        main() {
            const i = GM_getValue("initial");
            if (i === true) this.e(true);
            else if (!i) this.e();
        },
    };
    const getSelection = () => {
        const select = window.getSelection();
        return select ? select.toString().trim() : null;
    };
    //clear zero-width character
    const clear_zero_width = (text) =>
        text.trim().replace(/[\u200B-\u200D\uFEFF]/g, "");
    const escapeBlank = (target) => {
        const type = typeof target;
        if (type === "object") {
            const entries = Object.entries(target);
            for (const [key, value] of entries)
                target[key] = value.replace(/\s/g, " ");
        } else {
            target = target.replace(/\s/g, " ");
        }
        return target;
    };
    const escapeHTML = (s) => {
        const reg = /“|&|’|<|>|[\x00-\x20]|[\x7F-\xFF]|[\u0100-\u2700]/g;
        return typeof s !== "string"
            ? s
            : s.replace(reg, ($0) => {
                  let c = $0.charCodeAt(0),
                      r = ["&#"];
                  c = c == 0x20 ? 0xa0 : c;
                  r.push(c);
                  r.push(";");
                  return r.join("");
              });
    };
    //cut out part of title with specified length, take care of chinese character & english character
    const titleSlice = (str) => {
        let length = 0;
        let newstr = "";
        for (const e of str) {
            length += e.charCodeAt(0).toString(16).length === 4 ? 2 : 1;
            newstr += e;
            if (length > 27) return `${newstr}...`;
        }
        return newstr;
    };
    const createButton = (name, title, otherButton = "", position = "left") => {
        title = escapeBlank(title);
        const html = `
            <div
                id = "assist-button-container"
                >
                <style>
                    button.assist-button {
                        border-radius: 0 1px 1px 0;
                        border: rgb(247, 232, 176) solid 1.2px;
                        display: inline-block;
                        margin-top: 4px;
                        font-size: 14px;
                        height: 28px;
                        width: 75px;
                        box-shadow: 3px 4px 1px #888888;
                        justify-content: center;
                    }
                    div#assist-button-container {
                        opacity: 0.15;
                        ${position}: 4%;
                        width: 60px;
                        flex-direction: column;
                        position: fixed;
                        bottom: 7%;
                    }
                    div#assist-button-container:hover {
                        opacity: 1;
                        transition: opacity 2s;
                    }
                </style>
                ${otherButton}
                <button class="assist-button block" style="color: black;" title=${title}>${name}</button>
            </div>`;
        document.documentElement.insertAdjacentHTML("beforeend", html);
    };
    const createPopup = (wtime = 3) => {
        const html = `
        <div id="autoscroll-tips">
            <style>
                div#autoscroll-tips {
                    position: fixed;
                    top: 0;
                    right: 0;
                    bottom: 0;
                    left: 0;
                    z-index: 10000;
                    overflow: hidden;
                    -webkit-transition: background-color 0.2s ease-in-out;
                    transition: background-color 0.2s ease-in-out;
                }
                .autoscroll-tips_content {
                    position: fixed;
                    top: 45%;
                    left: 50%;
                    transform: translate(-50%, -50%);
                    background: lightgray;
                    width: 240px;
                    height: 120px;
                    border: 1px solid #b9d5ff;
                }
            </style>
            <div class="autoscroll-tips_content">
                <div class="autotips_content" style="margin: 5px 5px 5px 5px">
                    <h2 style="text-align: center">Auto Scroll Mode</h2>
                    <hr />
                    <p style="text-align: center">${wtime}s, auto load next page</p>
                    <div
                        class="auto_button"
                        style="
                            text-align: center;
                            letter-spacing: 25px;
                            margin-top: 12px;
                        "
                    >
                        <style>
                            button {
                                width: 60px;
                                height: 24px;
                                box-shadow: 1px 2px 1px #888888;
                            }
                        </style>
                        <button title="load next page immediately">OK</button>
                        <button title="cancel load next page">Cancel</button>
                    </div>
                </div>
            </div>
        </div>`;
        document.body.insertAdjacentHTML("beforeend", html);
    };
    /*
    unfinished project
    directly block some ad information url requests
    loaded_article, control the quantity of loaded articles
    salt_article, zhihu's paid_content
    class, extends, super, class Inheritance, subclass
    */
    unsafeWindow.XMLHttpRequest = class extends unsafeWindow.XMLHttpRequest {
        open(...args) {
            // keep the js of init.js from accessing continuously the url: zhihu-web-analytics.zhihu.com
            if (args.length > 1 && args[1].includes("/logs/batch"))
                (args[1] = "http://127.0.0.1"), (args[2] = true);
            return super.open(...args);
        }
    };
    const fetch_intercept = {
        intercept() {
            const fetch = unsafeWindow.fetch;
            unsafeWindow.fetch = (...args) =>
                (async (args) => {
                    const url = args[0];
                    const ad_list = [
                        "/commercial",
                        "include=ad_type",
                        "/read_count_statistics",
                        "/events/r",
                        "/top_search",
                        "/preset_words",
                        "/scan_info",
                        "/logs/batch",
                        "/paid_content",
                        "/market/rhea/questions",
                    ];
                    const article_api = "/api/v4/questions/";
                    url.includes(article_api) && (this.loaded_articles += 1);
                    return ad_list.some((e) => url.includes(e))
                        ? {
                              // keep the js of init.js from accessing continuously the object
                              status: 204,
                              headers: {
                                  get() {
                                      return null;
                                  },
                              },
                              text() {
                                  throw new SyntaxError("some error");
                              },
                          }
                        : await fetch(...args);
                })(args);
        },
        // response, response.clone().json(), copy the response to new obj, the response can only be read once
        // check if the answer is paid content(salt)
        check_salt(response) {
            response
                .then((result) => this.check_content(result))
                .catch(() => console.log("error, no content"));
        },
        check_content(result) {
            for (const a in result) {
                const obj = result[a];
                const name = Object.prototype.toString.call(obj);
                if (name === "[object Array]") {
                    if (a === "data") {
                        for (const e of obj) {
                            if (e.type === "knowledge_result") {
                                const item = e["answer_obj"];
                                if (item) {
                                    const id = item["id"];
                                    id &&
                                        !this.salt_articles.includes(id) &&
                                        this.salt_articles.push(id);
                                }
                            }
                        }
                        return;
                    }
                } else if (name === "[object Object]") this.check_content(obj);
            }
        },
        loaded_articles: 0,
        salt_articles: null,
        initial(index = 0) {
            index > 0 && (this.salt_articles = []);
            this.intercept();
        },
    };
    /*
    convert image to base64 code
    just support a little type of image, chrome, don't support image/gif type
    support type, test on this site: https://kangax.github.io/jstests/toDataUrl_mime_type_test/
    */
    const imageConvertor = {
        canvas(image, format, quality) {
            const canvas = document.createElement("canvas");
            const context2D = canvas.getContext("2d");
            context2D.clearRect(0, 0, canvas.width, canvas.height);
            canvas.width = image.naturalWidth;
            canvas.height = image.naturalHeight;
            context2D.drawImage(image, 0, 0);
            return canvas.toDataURL("image/" + format, quality);
            //0-1, default: 0.92, the quality of pic
        },
        main(imgURL, format = "webp", quality = 0.92) {
            return new Promise((resolve, reject) => {
                const img = new Image();
                img.src = imgURL;
                img.crossOrigin = "";
                img.onload = () => resolve(this.canvas(img, format, quality));
                img.onerror = (err) => reject(err);
            });
        },
    };
    const Download_module = (url, filename, timeout = 3000) => {
        return new Promise((resolve, reject) => {
            GM_download({
                url: url,
                name: filename,
                timeout: timeout,
                onload: () => resolve(true),
                onerror: (err) => {
                    console.log(err);
                    reject(null);
                },
                ontimeout: () => reject("timeout error"),
            });
        });
    };
    const image_base64_download = (url, filename) => {
        let a = document.createElement("a");
        a.href = url;
        a.download = filename;
        a.click();
        a.remove();
        a = null;
    };
    const xmlHTTPRequest = (
        url,
        time = 2500,
        responeType = "json",
        rType = false
    ) => {
        return new Promise(function (resolve, reject) {
            GM_xmlhttpRequest({
                method: "GET",
                url: url,
                timeout: time,
                responseType: responeType,
                onload: (response) => {
                    if (response.status == 200) {
                        if (rType) {
                            //after redirect, get the final URL
                            resolve(response.finalUrl);
                        } else {
                            resolve(response.response);
                        }
                    } else {
                        console.log(`err: code ${response.status}`);
                        reject("request data error");
                    }
                },
                onerror: (e) => {
                    console.log(e);
                    reject("something error");
                },
                ontimeout: (e) => {
                    console.log(e);
                    reject("timeout error");
                },
            });
        });
    };
    const column_Home = {
        item_index: 0,
        single_Content_request(type, id, node, info) {
            const types = {
                0: "answers",
                1: "questions",
                2: "articles",
            };
            const name = types[type];
            if (!name) return;
            const args = `include=data[*].is_normal,admin_closed_comment,reward_info,is_collapsed,annotation_action,annotation_detail,collapse_reason,is_sticky,collapsed_by,suggest_edit,comment_count,can_comment,content,editable_content,attachment,voteup_count,reshipment_settings,comment_permission,created_time,updated_time,review_info,relevant_info,question,excerpt,is_labeled,paid_info,paid_info_content,relationship.is_authorized,is_author,voting,is_thanked,is_nothelp,is_recognized;data[*].mark_infos[*].url;data[*].author.follower_count,badge[*].topics;data[*].settings.table_of_content.enabled`;
            const api = `https://www.zhihu.com/api/v4/${name}/${id}?${args}`;
            xmlHTTPRequest(api).then(
                (json) => {
                    typeof json === "string" && (json = JSON.parse(json));
                    node.insertAdjacentHTML(
                        "afterbegin",
                        this.item_Raw(json, this.item_index, info)
                    );
                    this.item_index += 1;
                },
                (err) => {
                    const index =
                        zhihu.Column.home_Module.loaded_list.indexOf(id);
                    index > -1 &&
                        zhihu.Column.home_Module.loaded_list.splice(index, 1);
                    colorful_Console.main(
                        { title: id, content: "failed to request data" },
                        colorful_Console.colors.warning
                    );
                    console.log(err);
                }
            );
        },
        time_Format(date) {
            return (
                `0${date.getHours()}`.slice(-2) +
                ":" +
                `0${date.getMinutes()}`.slice(-2)
            );
        },
        item_Raw(json, index, info) {
            const author = json.author;
            const ct = (json.created_time || json.created) * 1000;
            const mt = (json.updated_time || json.created) * 1000;
            const c = new Date(ct);
            const m = new Date(mt);
            const html = `
            <div class="List-item" tabindex="0">
                <div
                    class="ContentItem AnswerItem"
                    data-za-index=${index}
                    name=${json.id}
                    itemprop="suggestedAnswer"
                    itemtype="http://schema.org/Answer"
                    itemscope=""
                    data-za-detail-view-path-module=${json.type}
                    data-za-detail-view-path-index=${index}
                >
                    <div class="ContentItem-meta">
                        <h2 class="ContentItem-title">
                            <div
                                itemprop="zhihu:question"
                                itemtype="http://schema.org/Question"
                                itemscope=""
                            >
                                <meta
                                    itemprop="url"
                                    content=${info.url}
                                /><meta itemprop="name" content=${
                                    info.title
                                }/><a
                                    target="_blank"
                                    data-za-detail-view-element_name="Title"
                                    href=${info.url}
                                    >${info.title}</a
                                >
                            </div>
                        </h2>
                        <hr>
                        <br>
                        <div
                            class="AuthorInfo AnswerItem-authorInfo AnswerItem-authorInfo--related"
                            itemprop="author"
                            itemscope=""
                            itemtype="http://schema.org/Person"
                        >
                            <meta itemprop="name" content=${author.name} /><meta
                                itemprop="image"
                                content=${author.avatar_url}
                            /><meta
                                itemprop="url"
                                content=https://www.zhihu.com/${
                                    author.is_org ? "org" : "people"
                                }/${author.url_token}
                            /><span
                                class="UserLink AuthorInfo-avatarWrapper"
                                ><div class="Popover">
                                    <div
                                        id="Popover77-toggle"
                                        aria-haspopup="true"
                                        aria-expanded="false"
                                        aria-owns="Popover77-content"
                                    >
                                        <a
                                            class="UserLink-link"
                                            data-za-detail-view-element_name="User"
                                            target="_blank"
                                            href=https://www.zhihu.com/${
                                                author.is_org ? "org" : "people"
                                            }/${author.url_token}
                                            ><img
                                                class="Avatar AuthorInfo-avatar"
                                                width="38"
                                                height="38"
                                                src=${json.author.avatar_url}
                                                srcset="
                                                    ${json.author.avatar_url} 2x
                                                "
                                                alt=${json.author.name}
                                        /></a>
                                    </div></div
                            ></span>
                            <div class="AuthorInfo-content">
                                <div class="AuthorInfo-head">
                                    <span class="UserLink AuthorInfo-name"
                                        ><div class="Popover">
                                            <div
                                                id="Popover78-toggle"
                                                aria-haspopup="true"
                                                aria-expanded="false"
                                                aria-owns="Popover78-content"
                                            >
                                                <a
                                                    class="UserLink-link"
                                                    data-za-detail-view-element_name="User"
                                                    target="_blank"
                                                    href=https://www.zhihu.com/${
                                                        author.is_org
                                                            ? "org"
                                                            : "people"
                                                    }/${author.url_token}
                                                    >${json.author.name}</a
                                                >
                                            </div>
                                        </div></span
                                    >
                                </div>
                                <div class="AuthorInfo-detail">
                                    <div class="AuthorInfo-badge">
                                        <div class="ztext AuthorInfo-badgeText">
                                            ${json.author.headline}
                                        </div>
                                    </div>
                                </div>
                            </div>
                        </div>
                        <div class="LabelContainer-wrapper"></div>
                    </div>
                    <meta
                        itemprop="dateCreated"
                        content=${c.toISOString()}
                    /><meta
                        itemprop="dateModified"
                        content=${m.toISOString()}
                    />
                    <div class="RichContent is-collapsed RichContent--unescapable">
                        <div class="RichContent-inner" style="max-height: 400px">
                            <span
                                class="RichText ztext CopyrightRichText-richText"
                                itemprop="text"
                            >${json.content}</span>
                        </div>
                        <button
                            type="button"
                            class="Button ContentItem-rightButton ContentItem-expandButton Button--plain"
                        >
                            展开阅读全文<span
                                style="display: inline-flex; align-items: center"
                                >&#8203;<svg
                                    class="Zi Zi--ArrowDown ContentItem-arrowIcon"
                                    fill="currentColor"
                                    viewBox="0 0 24 24"
                                    width="24"
                                    height="24"
                                >
                                    <path
                                        d="M12 13L8.285 9.218a.758.758 0 0 0-1.064 0 .738.738 0 0 0 0 1.052l4.249 4.512a.758.758 0 0 0 1.064 0l4.246-4.512a.738.738 0 0 0 0-1.052.757.757 0 0 0-1.063 0L12.002 13z"
                                        fill-rule="evenodd"
                                    ></path></svg
                            ></span>
                        </button>
                        <div class="time_module">
                            <span data-tooltip="发布于 ${zhihu.Column.timeStampconvertor(
                                ct
                            )} ${this.time_Format(
                c
            )}">编辑于 ${zhihu.Column.timeStampconvertor(mt)}</span>
                        </div>
                    </div>
                </div>
            </div>`;
            return html;
        },
    };
    //hidden element = false
    const elementVisible = {
        getElementTop(obj) {
            let top = 0;
            while (obj) {
                top += obj.offsetTop;
                obj = obj.offsetParent;
            }
            return top;
        },
        check(wh, element, offset) {
            const tp = this.getElementTop(element);
            return tp + element.clientHeight > offset && offset + wh > tp;
        },
        main(fNode, args) {
            const offset = fNode.scrollTop;
            const wh = window.innerHeight;
            const type = Object.prototype.toString.call(args);
            if (type === "[object HTMLCollection]") {
                for (const e of args) if (this.check(wh, e, offset)) return e;
                return null;
            } else return this.check(wh, args, offset);
        },
    };
    const change_Title = (title) => {
        const chs = document.head.children;
        for (const c of chs) {
            if (c.localName === "title") {
                c.innerHTML = title;
                document.title = title;
                break;
            }
        }
    };
    const get_Title = () => {
        const title = document.title;
        return title.endsWith("- 知乎")
            ? title.slice(0, title.length - 5)
            : title;
    };
    class Database {
        /*
        dname: name of database;
        tname: name of table;
        mode: read or read&write;
        */
        constructor(dbname, tbname = "", rwmode = false, version = 1) {
            this.dbopen =
                version === 1
                    ? indexedDB.open(dbname)
                    : indexedDB.open(dbname, version);
            this.RWmode = rwmode ? "readwrite" : "readonly";
            this.tbname = tbname;
            const getIndex = (fieldname) => this.Table.index(fieldname);
        }
        Initialize() {
            return new Promise((resolve, reject) => {
                //if the db does not exist, this event will fired firstly;
                //adjust the version of db, which can trigger this event => create/delete table or create/delete index (must lauch from this event);
                this.dbopen.onupgradeneeded = (e) => {
                    this.store = e.target.result;
                    this.updateEvent = true;
                    this.storeEvent();
                    resolve(0);
                };
                this.dbopen.onsuccess = () => {
                    if (this.store) return;
                    this.updateEvent = false;
                    this.store = this.dbopen.result;
                    this.storeEvent();
                    resolve(1);
                };
                this.dbopen.onerror = (e) => {
                    console.log(e);
                    reject("error");
                };
                /*
                The event handler for the blocked event.
                This event is triggered when the upgradeneeded event should be triggered _
                because of a version change but the database is still in use (i.e. not closed) somewhere,
                even after the versionchange event was sent.
                */
                this.dbopen.onblocked = () => {
                    console.log("please close others tab to update database");
                    reject("conflict");
                };
                this.dbopen.onversionchange = (e) =>
                    console.log("The version of this database has changed");
            });
        }
        createTable(keyPath) {
            if (this.updateEvent) {
                const index = keyPath
                    ? { keyPath: keyPath }
                    : { autoIncrement: true };
                this.store.createObjectStore(this.tbname, index);
            }
        }
        createNewTable(keyPath) {
            return new Promise((resolve, reject) => {
                this.version = this.store.version + 1;
                this.store.close();
                if (this.storeErr) {
                    reject("database generates some unknow error");
                    return;
                }
                this.dbopen = indexedDB.open(this.store.name, this.version);
                this.Initialize().then(
                    () => {
                        this.createTable(keyPath);
                        resolve(true);
                    },
                    () => reject("database initial fail")
                );
            });
        }
        storeEvent() {
            this.store.onclose = () => console.log("closing...");
            this.store.onerror = () => (this.storeErr = true);
        }
        get checkTable() {
            return this.store.objectStoreNames.contains(this.tbname);
        }
        get Tablenames() {
            return this.store.objectStoreNames;
        }
        get DBname() {
            return this.store.name;
        }
        get Indexnames() {
            return this.Table.indexNames;
        }
        get DBversion() {
            return this.store.version;
        }
        get Datacount() {
            return new Promise((resolve, reject) => {
                const req = this.Table.count();
                req.onsuccess = (e) =>
                    resolve({
                        count: e.target.result,
                        name: e.target.source.name,
                    });
                req.onerror = (e) => reject(e);
            });
        }
        //take care the transaction, must make sure the transaction is alive when you need deal with something continually
        get Table() {
            const transaction = this.store.transaction(
                [this.tbname],
                this.RWmode
            );
            const table = transaction.objectStore(this.tbname);
            transaction.onerror = () =>
                console.log("warning, error on transaction");
            return table;
        }
        rollback() {
            this.transaction && this.transaction.abort();
        }
        read(keyPath) {
            return new Promise((resolve, reject) => {
                const request = this.Table.get(keyPath);
                request.onsuccess = () => resolve(request.result);
                request.onerror = () => reject("error");
            });
        }
        batchCheck(tables, keyPath) {
            return new Promise((resolve) => {
                const arr = [];
                for (const t of tables) {
                    const transaction = this.store.transaction(
                        this.store.objectStoreNames,
                        this.RWmode
                    );
                    const table = transaction.objectStore(t);
                    const rq = new Promise((reso, rej) => {
                        const req = table.get(keyPath);
                        req.onsuccess = () => reso(req.result);
                        req.onerror = () => rej("error");
                    });
                    arr.push(rq);
                }
                Promise.allSettled(arr).then((results) => {
                    resolve(
                        results.map((r) =>
                            r.status === "rejected" ? null : r.value
                        )
                    );
                });
            });
        }
        add(info) {
            return new Promise((resolve, reject) => {
                const op = this.Table.add(info);
                op.onsuccess = () => resolve(true);
                op.onerror = (e) => {
                    console.log(e);
                    reject("error");
                };
            });
        }
        getAll(tables) {
            return new Promise((resolve) => {
                const arr = [];
                for (const t of tables) {
                    const transaction = this.store.transaction(
                        this.store.objectStoreNames,
                        this.RWmode
                    );
                    const table = transaction.objectStore(t);
                    const rq = new Promise((reso, rej) => {
                        const req = table.getAll();
                        req.onsuccess = (e) =>
                            reso({
                                name: e.target.source.name,
                                data: e.target.result,
                            });
                        req.onerror = () => rej(t);
                    });
                    arr.push(rq);
                }
                Promise.allSettled(arr).then((results) => resolve(results));
            });
        }
        updateRecord(keyPath) {
            return new Promise((resolve, reject) => {
                this.read(keyPath).then(
                    (result) => {
                        if (result) {
                            result.visitTime = Date.now();
                            let times = result.visitTimes;
                            result.visitTimes = times ? ++times : 2;
                            this.update(result).then(
                                () => resolve(true),
                                (err) => reject(err)
                            );
                        } else resolve(true);
                    },
                    (err) => reject(err)
                );
            });
        }
        update(info, keyPath, mode = false) {
            //if db has contained the item, will update the info; if it does not, a new item is added
            return new Promise((resolve, reject) => {
                //keep cursor
                if (mode) {
                    this.read(info[keyPath]).then(
                        (result) => {
                            if (!result) {
                                this.add(info).then(
                                    () => resolve(true),
                                    (err) => reject(info)
                                );
                            } else {
                                const op = this.Table.put(
                                    Object.assign(result, info)
                                );
                                op.onsuccess = () => resolve(true);
                                op.onerror = (e) => {
                                    console.log(e);
                                    reject(info);
                                };
                            }
                        },
                        (err) => console.log(err)
                    );
                } else {
                    const op = this.Table.put(info);
                    op.onsuccess = () => resolve(true);
                    op.onerror = (e) => {
                        console.log(e);
                        reject(info);
                    };
                }
            });
        }
        clear() {
            this.Table.clear();
        }
        //must have primary key
        deleteiTems(keyPath) {
            return new Promise((resolve, reject) => {
                const op = this.Table.delete(keyPath);
                op.onsuccess = () => {
                    console.log("delete item successfully");
                    resolve(true);
                };
                op.onerror = (e) => {
                    console.log(e);
                    reject("error");
                };
            });
        }
        //note: create a index must lauch from onupgradeneeded event; we need triggle update event
        createIndex(indexName, keyPath, objectParameters) {
            if (!this.updateEvent) {
                console.log("this function must be through onupgradeneeded");
                return;
            }
            this.table.createIndex(indexName, keyPath, objectParameters);
        }
        deleTable() {
            if (!this.updateEvent) {
                console.log("this function must be through onupgradeneeded");
                return;
            }
            this.dbopen.deleteObjectStore(this.tbname);
        }
        deleIndex(indexName) {
            if (!this.updateEvent) {
                console.log("this function must be through onupgradeneeded");
                return;
            }
            this.table.deleteIndex(indexName);
        }
        close() {
            //The connection is not actually closed until all transactions created using this connection are complete
            this.store.close();
        }
        static deleDB(dbname) {
            return new Promise((resolve, reject) => {
                const DBDeleteRequest = window.indexedDB.deleteDatabase(dbname);
                DBDeleteRequest.onerror = (e) => {
                    console.log(e);
                    reject("error");
                };
                DBDeleteRequest.onsuccess = () => {
                    console.log("success");
                    resolve(true);
                };
            });
        }
    }
    const dataBaseInstance = {
        db: null,
        auto_tags() {
            const tags = document.getElementsByClassName("Tag-content");
            const arr = [];
            for (const tag of tags) {
                const text = tag.innerText.trim();
                text && arr.push(text);
            }
            return arr.length > 0 ? arr.join(" ") : null;
        },
        additem(columnID, node, pid) {
            const info = {};
            info.pid = pid || this.pid;
            info.update = Date.now();
            info.excerpt = "";
            info.visitTimes = 1;
            info.visitTime = info.update;
            const content_id = node
                ? "RichText ztext CopyrightRichText-richText"
                : "RichText ztext Post-RichText";
            node = node || document.body;
            const contentholder = node.getElementsByClassName(content_id);
            if (contentholder.length > 0) {
                const chs = contentholder[0].childNodes;
                let excerpt = "";
                //take some data as this article's digest
                let ic = 0;
                for (const node of chs) {
                    if (node.localName === "p") {
                        excerpt += node.innerText;
                        if (excerpt.length > 300) {
                            excerpt = excerpt.slice(0, 300);
                            break;
                        }
                    }
                    ic++;
                    if (ic > 5) break;
                }
                info.excerpt = excerpt;
            }
            info.userName = "";
            info.userID = "";
            const user = node.getElementsByClassName(
                "UserLink AuthorInfo-name"
            );
            if (user.length > 0) {
                const link = user[0].getElementsByTagName("a");
                if (link.length > 0) {
                    const p = link[0].pathname;
                    info.userID = p.slice(p.lastIndexOf("/") + 1);
                    info.userName = link[0].text;
                }
            }
            info.ColumnID = columnID || "";
            const defaultV = this.auto_tags() || "A B";
            const p = prompt(
                "please input some tags about this article, like: javascript python; multiple tags use blank space to isolate",
                defaultV
            );
            let tags = [];
            if (p && p !== defaultV && p.trim()) {
                const tmp = p.split(" ");
                for (let e of tmp) {
                    e = e.trim();
                    e && tags.push(e);
                }
            }
            const note = prompt(
                "you can input something to highlight this article, eg: this article is about advantage python usage"
            );
            info.tags = tags;
            info.note = note || "";
            info.title = get_Title();
            this.db.update(info, "pid", false).then(
                () =>
                    Notification(
                        "add this article to collection successfully",
                        "Tips"
                    ),
                () =>
                    Notification(
                        "add this aritcle to collection fail",
                        "Warning"
                    )
            );
        },
        fold(info) {
            this.db.update(info, "name", true).then(
                () => console.log("this answer has been folded"),
                () => console.log("add this anser to folded list fail")
            );
        },
        get DBname() {
            return this.db.DBname;
        },
        get Table() {
            return this.db.Table;
        },
        batchCheck(tableNames) {
            const pid = this.pid;
            return new Promise((resolve) =>
                this.db
                    .batchCheck(tableNames, pid)
                    .then((results) => resolve(results))
            );
        },
        updateRecord(keyPath) {
            const pid = keyPath || this.pid;
            this.db.updateRecord(pid).then(
                () => console.log("update record finished"),
                (err) => console.log(err)
            );
        },
        check(keyPath) {
            const pid = keyPath || this.pid;
            return new Promise((resolve, reject) => {
                this.db.read(pid).then(
                    (result) => resolve(result),
                    (err) => reject(err)
                );
            });
        },
        get pid() {
            return location.pathname.slice(3);
        },
        dele(mode, keyPath) {
            const pid = keyPath || this.pid;
            this.db.deleteiTems(pid).then(
                () =>
                    mode &&
                    Notification(
                        `the article of ${pid} has been deleted from collection successfully`,
                        "Tips"
                    ),
                () => mode && Notification("delete article fail")
            );
        },
        update(info) {
            return this.db.update(info);
        },
        /**
         * @param {String} name
         */
        set TableName(name) {
            this.db.tbname = name;
        },
        close() {
            this.db && this.db.close();
            this.db = null;
        },
        getdataCount(tables) {
            return new Promise((resolve) => {
                const arr = tables.map((t) => {
                    this.TableName = t;
                    return this.db.Datacount;
                });
                Promise.allSettled(arr).then((results) => resolve(results));
            });
        },
        getAll(tables) {
            return new Promise((resolve) =>
                this.db.getAll(tables).then((result) => resolve(result))
            );
        },
        initial(tableNames, mode = false, keyPath = "pid") {
            return new Promise((resolve, reject) => {
                if (!Array.isArray(tableNames)) {
                    reject("this parameter must be array");
                    return;
                }
                const dbname = "zhihuDatabase";
                const db = new Database(
                    dbname,
                    tableNames.length === 1 ? tableNames[0] : "",
                    mode
                );
                this.db = db;
                db.Initialize().then(
                    (result) => {
                        if (result === 0) {
                            for (const table of tableNames) {
                                db.tbname = table;
                                db.createTable(keyPath);
                            }
                        }
                        resolve(result);
                    },
                    (err) => reject(err)
                );
            });
        },
    };
    const data2Excel = {
        s2ab(s) {
            const buf = new ArrayBuffer(s.length);
            const view = new Uint8Array(buf);
            for (let i = 0; i !== s.length; ++i)
                view[i] = s.charCodeAt(i) & 0xff;
            return buf;
        },
        sheet2blob(sheet, sheetName) {
            sheetName = sheetName || "sheet1";
            const workbook = {
                SheetNames: [sheetName],
                Sheets: {},
            };
            workbook.Sheets[sheetName] = sheet;
            const wopts = {
                bookType: "xlsx",
                bookSST: false,
                type: "binary",
            };
            const wbout = XLSX.write(workbook, wopts);
            const blob = new Blob([this.s2ab(wbout)], {
                type: "application/octet-stream",
            });
            // string to ArrayBuffer
            return blob;
        },
        openDownloadDialog(url, saveName) {
            if (typeof url == "object" && url instanceof Blob)
                url = URL.createObjectURL(url);
            let aLink = document.createElement("a");
            aLink.href = url;
            aLink.download = saveName || "";
            let event;
            if (window.MouseEvent) event = new MouseEvent("click");
            else {
                event = document.createEvent("MouseEvents");
                event.initMouseEvent(
                    "click",
                    true,
                    false,
                    window,
                    0,
                    0,
                    0,
                    0,
                    0,
                    false,
                    false,
                    false,
                    false,
                    0,
                    null
                );
            }
            aLink.dispatchEvent(event);
            URL.revokeObjectURL(url);
        },
        json_toExcel(data, wbname, shname) {
            const sheet = XLSX.utils.json_to_sheet(data);
            this.openDownloadDialog(
                this.sheet2blob(sheet, shname),
                `${wbname}.xlsx`
            );
        },
        // dic => sheet, if set the header, the columns' name should been put the first place.
        dic_toExcel(data, wbname, shname = "data") {
            const sheet = XLSX.utils.aoa_to_sheet(Object.entries(data));
            this.openDownloadDialog(
                this.sheet2blob(sheet, shname),
                `${wbname}.xlsx`
            );
        },
    };
    const control_pannel = {
        collect() {
            console.log("wait a few seconds, data is retrieving...");
            dataBaseInstance.getAll(["collection"]).then((results) => {
                const result = results[0];
                if (!result) {
                    console.log("no data");
                    return;
                }
                if (result.status === "fulfilled") {
                    const data = result.value.data;
                    for (const e of data) {
                        const tags = e.tags;
                        if (tags && Array.isArray(tags))
                            e.tags = tags.join(";");
                    }
                    data2Excel.json_toExcel(data, "collection_" + Date.now());
                    console.log(
                        "successfully retrieved all collected articles"
                    );
                } else console.log("failed to retrieve data");
            });
        },
    };
    const control_pannel_init = () => {
        unsafeWindow.optimizer = {
            collect: null,
        };
        let time_id = null;
        Object.defineProperty(unsafeWindow.optimizer, "collect", {
            enumerable: true,
            configurable: true,
            get() {
                return "input: 'optimier.collect = true;' || download all collected articles as excel file";
            },
            set(value) {
                if (value === true) {
                    time_id && clearTimeout(time_id);
                    time_id = setTimeout(() => control_pannel.collect(), 300);
                }
            },
        });
    };
    const MangeData = {
        importData: {
            isRunning: false,
            create(mode) {
                const html = `
                <div
                    id="read_local_text"
                    style="
                        background: lightgray;
                        width: 360px;
                        position: fixed;
                        top: 50%;
                        left: 50%;
                        transform: translate(-50%, -50%);
                        height: 360px;
                        border: 1px solid #ccc !important;
                        box-shadow: 1px 1px 4px #888888;
                    "
                >
                    <h2 style="text-align: center; margin: 5px">IndexedDB Data Import</h2>
                    <hr />
                    <input
                        type="file"
                        class="read-local-txt-input"
                        id="readTxt"
                        accept=".txt"
                        ${mode ? 'multiple="multiple"' : ""}
                        style="margin-top: 35%; margin-left: 25%"
                    />
                </div>`;
                document.body.insertAdjacentHTML("beforeend", html);
                this.event(mode);
            },
            notice(filename) {
                colorful_Console.main(
                    {
                        tilte: "DB import data",
                        content: `this file(${filename}) does not match current DB`,
                    },
                    colorful_Console.colors.warning
                );
            },
            checkFile_format(json, filename, lists) {
                if (!json || !lists.includes(json.name)) {
                    this.notice(filename);
                    return null;
                }
                const data = json.data;
                if (!data || !Array.isArray(data) || data.length === 0) {
                    this.notice(filename);
                    return null;
                }
                return data;
            },
            remove() {
                this.timeID && clearTimeout(this.timeID, (this.timeID = null));
                this.node && (this.node.remove(), (this.node = null));
            },
            node: null,
            timeID: null,
            read_file(file, lists) {
                return new Promise((resolve, reject) => {
                    const r = new FileReader();
                    //default encode is UTF-8;
                    r.readAsText(file);
                    r.onload = (e) => {
                        const result = e.target.result;
                        if (!result) {
                            reject(file.name);
                            return;
                        }
                        try {
                            const json = JSON.parse(result);
                            const data = this.checkFile_format(
                                json,
                                file.name,
                                lists
                            );
                            if (!data) {
                                reject(file.name);
                                return;
                            }
                            const tname = json.name;
                            dataBaseInstance.TableName = tname;
                            const arr = data.map((e) =>
                                dataBaseInstance.update(e)
                            );
                            Promise.allSettled(arr).then((results) => {
                                results.forEach((e) => {
                                    if (e.status === "rejected") {
                                        const r = e.value;
                                        r &&
                                            colorful_Console.main(
                                                {
                                                    title: "DB put data",
                                                    content: `ID of ${r.name} has failed to import`,
                                                },
                                                colorful_Console.colors.warning
                                            );
                                    }
                                });
                                Notification(
                                    `the data(${tname}) import operation has completed`,
                                    "Tips",
                                    3500
                                );
                                resolve(file.name);
                            });
                        } catch (error) {
                            console.log(error);
                            reject(file.name);
                        }
                    };
                    r.onerror = (e) => {
                        console.log(e);
                        reject(file.name);
                    };
                });
            },
            loadFile(e, lists) {
                this.isRunning = true;
                const arr = [];
                for (const file of e.target.files)
                    arr.push(this.read_file(file, lists));
                Promise.allSettled(arr).then((results) => {
                    results.forEach((e) =>
                        console.log(
                            e.status === "rejected"
                                ? `failed to import file(${e.value}) to DB`
                                : `import file(${e.value}) to DB successfully`
                        )
                    );
                    this.timeID = setTimeout(
                        () => ((this.timeID = null), this.remove()),
                        1200
                    );
                    Notification(
                        "the operation import data to DB has finished",
                        "Tips"
                    );
                    this.isRunning = false;
                });
            },
            event(mode) {
                this.node = document.getElementById("read_local_text");
                let i = this.node.getElementsByTagName("input")[0];
                i.onchange = (e) => {
                    if (e.target.files.length === 0) return;
                    if (
                        !confirm(
                            "take care! are you sure to start importing data?"
                        )
                    )
                        return;
                    const lists = mode
                        ? ["collection", "preference"]
                        : ["foldedAnswer"];
                    this.loadFile(e, lists);
                };
                i = null;
            },
            main(mode) {
                !this.isRunning && this.node
                    ? this.remove()
                    : this.create(mode);
            },
        },
        exportData: {
            _download(text, filename) {
                const a = document.createElement("a");
                a.setAttribute(
                    "href",
                    "data:text/plain;charset=utf-8," + encodeURIComponent(text)
                );
                a.setAttribute("download", filename);
                a.style.display = "none";
                document.body.appendChild(a);
                a.click();
                document.body.removeChild(a);
            },
            _getData(tables) {
                dataBaseInstance.getAll(tables).then((results) => {
                    let i = 0;
                    for (const result of results) {
                        if (result.status === "rejected") {
                            Notification(
                                `the operation of backup DB(${result.value}) data fail`,
                                "Warning"
                            );
                            continue;
                        }
                        if (result.value.data.length === 0) {
                            Notification(
                                `the database(${result.value.name}) has not yet stored the data`,
                                "DB Tips"
                            );
                        }
                        setTimeout(
                            () =>
                                this._download(
                                    JSON.stringify(result.value),
                                    `DB_Backup_${
                                        result.value.name
                                    }_${Date.now()}.txt`
                                ),
                            i
                        );
                        i += 2000;
                    }
                });
            },
            _QAwebPage() {
                this._getData(["foldedAnswer"]);
            },
            _columnPage() {
                this._getData(["collection", "preference"]);
            },
            main(mode = true) {
                mode ? this._QAwebPage() : this._columnPage();
            },
        },
    };
    const zhihu = {
        /*
        these original functions of zhihu webpage will be failed in reader mode, so need to be rebuilt
        rebuild:
        1. gif player;
        2. video player;
        3. show raw picture
        add:
        1. picture viewer, continually open the raw picture in viewer mode;
        2. time clock
        note:
        the source of video come from v-list2, which has some problems, need escape ?
        */
        qaReader: {
            time_module: {
                /*
                1. adapted from https://zyjacya-in-love.github.io/flipclock-webpage/#
                2. html and css is adopted, and some codes have been reedited or cutted;
                3. rebuild js, the original js is too big, intricate or complicated;
                */
                get formated_Time() {
                    const time = this.Date_format;
                    const info = {};
                    info.hour = time.h;
                    this.time_arr = [...time.string];
                    info.before = this.time_arr.map((e) =>
                        e === "0" ? "9" : (parseInt(e) - 1).toString()
                    );
                    return info;
                },
                time_arr: null,
                create_Module(className, value) {
                    const html = `
                        <li class=${className}>
                            <a href="#"
                                ><div class="up">
                                    <div class="shadow"></div>
                                    <div class="inn">${value}</div>
                                </div>
                                <div class="down">
                                    <div class="shadow"></div>
                                    <div class="inn">${value}</div>
                                </div></a
                            >
                        </li>`;
                    return html;
                },
                removeClassname(node) {
                    node.className = "";
                },
                addNewClassName(node, newName) {
                    node.className = newName;
                },
                exe(clname, node, e, index) {
                    const ul = node.getElementsByClassName(clname)[0];
                    this.removeClassname(ul.firstElementChild);
                    this.addNewClassName(
                        ul.lastElementChild,
                        "flip-clock-before"
                    );
                    ul.insertAdjacentHTML(
                        "beforeend",
                        this.create_Module("flip-clock-active", e)
                    );
                    ul.firstElementChild.remove();
                    this.time_arr[index] = e;
                },
                f0(node, e, index) {
                    this.exe("flip ahour", node, e, index);
                },
                f1(node, e, index) {
                    this.exe("flip bhour", node, e, index);
                },
                f2(node, e, index) {
                    this.exe("flip play aminute", node, e, index);
                },
                firstRun: false,
                f3(node, e, index) {
                    this.exe("flip play bminute", node, e, index);
                    this.firstRun = true;
                },
                change_time_status(node, value) {
                    const a = node
                        .getElementsByClassName("flip-clock-meridium")[0]
                        .getElementsByTagName("a")[0];
                    a.innerText = value;
                    this.currentHour = value;
                },
                clock() {
                    const css = `
                        <style>
                            .clock {
                                width: auto;
                                zoom: 0.6;
                            }
                            .flip-clock-dot {
                                background: #ccc;
                            }
                            .flip-clock-meridium a {
                                color: #ccc;
                            }
                            #box {
                                display: table;
                            }
                            #content {
                                text-align: center;
                                display: table-cell;
                                vertical-align: middle;
                            }
                        </style>
                        <style>
                            .flip-clock-wrapper * {
                                -webkit-box-sizing: border-box;
                                -moz-box-sizing: border-box;
                                -ms-box-sizing: border-box;
                                -o-box-sizing: border-box;
                                box-sizing: border-box;
                                -webkit-backface-visibility: hidden;
                                -moz-backface-visibility: hidden;
                                -ms-backface-visibility: hidden;
                                -o-backface-visibility: hidden;
                                backface-visibility: hidden;
                            }
                            .flip-clock-wrapper a {
                                cursor: pointer;
                                text-decoration: none;
                                color: #ccc;
                            }
                            .flip-clock-wrapper a:hover {
                                color: #fff;
                            }
                            .flip-clock-wrapper ul {
                                list-style: none;
                            }
                            .flip-clock-wrapper.clearfix:before,
                            .flip-clock-wrapper.clearfix:after {
                                content: " ";
                                display: table;
                            }
                            .flip-clock-wrapper.clearfix:after {
                                clear: both;
                            }
                            .flip-clock-wrapper.clearfix {
                                *zoom: 1;
                            } /* Main */
                            .flip-clock-wrapper {
                                font: normal 11px "Helvetica Neue", Helvetica, sans-serif;
                                -webkit-user-select: none;
                            }
                            .flip-clock-meridium {
                                background: none !important;
                                box-shadow: 0 0 0 !important;
                                font-size: 36px !important;
                            }
                            .flip-clock-meridium a {
                                color: #313333;
                            }
                            .flip-clock-wrapper {
                                text-align: center;
                                position: relative;
                                width: 100%;
                                margin: 1em;
                            }
                            .flip-clock-wrapper:before,
                            .flip-clock-wrapper:after {
                                content: " "; /* 1 */
                                display: table; /* 2 */
                            }
                            .flip-clock-wrapper:after {
                                clear: both;
                            } /* Skeleton */
                            .flip-clock-wrapper ul {
                                position: relative;
                                float: left;
                                margin: 5px;
                                width: 60px;
                                height: 90px;
                                font-size: 80px;
                                font-weight: bold;
                                line-height: 87px;
                                border-radius: 6px;
                                background: #000;
                            }
                            .flip-clock-wrapper ul li {
                                z-index: 1;
                                position: absolute;
                                left: 0;
                                top: 0;
                                width: 100%;
                                height: 100%;
                                line-height: 87px;
                                text-decoration: none !important;
                            }
                            .flip-clock-wrapper ul li:first-child {
                                z-index: 2;
                            }
                            .flip-clock-wrapper ul li a {
                                display: block;
                                height: 100%;
                                -webkit-perspective: 200px;
                                -moz-perspective: 200px;
                                perspective: 200px;
                                margin: 0 !important;
                                overflow: visible !important;
                                cursor: default !important;
                            }
                            .flip-clock-wrapper ul li a div {
                                z-index: 1;
                                position: absolute;
                                left: 0;
                                width: 100%;
                                height: 50%;
                                font-size: 80px;
                                overflow: hidden;
                                outline: 1px solid transparent;
                            }
                            .flip-clock-wrapper ul li a div .shadow {
                                position: absolute;
                                width: 100%;
                                height: 100%;
                                z-index: 2;
                            }
                            .flip-clock-wrapper ul li a div.up {
                                -webkit-transform-origin: 50% 100%;
                                -moz-transform-origin: 50% 100%;
                                -ms-transform-origin: 50% 100%;
                                -o-transform-origin: 50% 100%;
                                transform-origin: 50% 100%;
                                top: -0.1px;
                            }
                            .flip-clock-wrapper ul li a div.up:after {
                                content: "";
                                position: absolute;
                                top: 44px;
                                left: 0;
                                z-index: 5;
                                width: 100%;
                                height: 3px;
                                background-color: #000;
                                background-color: rgba(0, 0, 0, 0.4);
                            }
                            .flip-clock-wrapper ul li a div.down {
                                -webkit-transform-origin: 50% 0;
                                -moz-transform-origin: 50% 0;
                                -ms-transform-origin: 50% 0;
                                -o-transform-origin: 50% 0;
                                transform-origin: 50% 0;
                                bottom: 0;
                                border-bottom-left-radius: 6px;
                                border-bottom-right-radius: 6px;
                            }
                            .flip-clock-wrapper ul li a div div.inn {
                                position: absolute;
                                left: 0;
                                z-index: 1;
                                width: 100%;
                                height: 200%;
                                color: #ccc;
                                text-shadow: 0 1px 2px #000;
                                text-align: center;
                                background-color: #333;
                                border-radius: 6px;
                                font-size: 70px;
                            }
                            .flip-clock-wrapper ul li a div.up div.inn {
                                top: 0;
                            }
                            .flip-clock-wrapper ul li a div.down div.inn {
                                bottom: 0;
                            } /* PLAY */
                            .flip-clock-wrapper ul.play li.flip-clock-before {
                                z-index: 3;
                            }
                            .flip-clock-wrapper .flip {
                                box-shadow: 0 2px 5px rgba(0, 0, 0, 0.7);
                            }
                            .flip-clock-wrapper ul.play li.flip-clock-active {
                                -webkit-animation: asd 0.01s 0.49s linear both;
                                -moz-animation: asd 0.01s 0.49s linear both;
                                animation: asd 0.01s 0.49s linear both;
                                z-index: 5;
                            }
                            .flip-clock-divider {
                                float: left;
                                display: inline-block;
                                position: relative;
                                width: 20px;
                                height: 100px;
                            }
                            .flip-clock-divider:first-child {
                                width: 0;
                            }
                            .flip-clock-dot {
                                display: block;
                                background: #323434;
                                width: 10px;
                                height: 10px;
                                position: absolute;
                                border-radius: 50%;
                                box-shadow: 0 0 5px rgba(0, 0, 0, 0.5);
                                left: 5px;
                            }
                            .flip-clock-divider .flip-clock-label {
                                position: absolute;
                                top: -1.5em;
                                right: -86px;
                                color: black;
                                text-shadow: none;
                            }
                            .flip-clock-divider.minutes .flip-clock-label {
                                right: -88px;
                            }
                            .flip-clock-divider.seconds .flip-clock-label {
                                right: -91px;
                            }
                            .flip-clock-dot.top {
                                top: 30px;
                            }
                            .flip-clock-dot.bottom {
                                bottom: 30px;
                            }
                            @-webkit-keyframes asd {
                                0% {
                                    z-index: 2;
                                }
                                100% {
                                    z-index: 4;
                                }
                            }
                            @-moz-keyframes asd {
                                0% {
                                    z-index: 2;
                                }
                                100% {
                                    z-index: 4;
                                }
                            }
                            @-o-keyframes asd {
                                0% {
                                    z-index: 2;
                                }
                                100% {
                                    z-index: 4;
                                }
                            }
                            @keyframes asd {
                                0% {
                                    z-index: 2;
                                }
                                100% {
                                    z-index: 4;
                                }
                            }
                            .flip-clock-wrapper ul.play li.flip-clock-active .down {
                                z-index: 2;
                                -webkit-animation: turn 0.5s 0.5s linear both;
                                -moz-animation: turn 0.5s 0.5s linear both;
                                animation: turn 0.5s 0.5s linear both;
                            }
                            @-webkit-keyframes turn {
                                0% {
                                    -webkit-transform: rotateX(90deg);
                                }
                                100% {
                                    -webkit-transform: rotateX(0deg);
                                }
                            }
                            @-moz-keyframes turn {
                                0% {
                                    -moz-transform: rotateX(90deg);
                                }
                                100% {
                                    -moz-transform: rotateX(0deg);
                                }
                            }
                            @-o-keyframes turn {
                                0% {
                                    -o-transform: rotateX(90deg);
                                }
                                100% {
                                    -o-transform: rotateX(0deg);
                                }
                            }
                            @keyframes turn {
                                0% {
                                    transform: rotateX(90deg);
                                }
                                100% {
                                    transform: rotateX(0deg);
                                }
                            }
                            .flip-clock-wrapper ul.play li.flip-clock-before .up {
                                z-index: 2;
                                -webkit-animation: turn2 0.5s linear both;
                                -moz-animation: turn2 0.5s linear both;
                                animation: turn2 0.5s linear both;
                            }
                            @-webkit-keyframes turn2 {
                                0% {
                                    -webkit-transform: rotateX(0deg);
                                }
                                100% {
                                    -webkit-transform: rotateX(-90deg);
                                }
                            }
                            @-moz-keyframes turn2 {
                                0% {
                                    -moz-transform: rotateX(0deg);
                                }
                                100% {
                                    -moz-transform: rotateX(-90deg);
                                }
                            }
                            @-o-keyframes turn2 {
                                0% {
                                    -o-transform: rotateX(0deg);
                                }
                                100% {
                                    -o-transform: rotateX(-90deg);
                                }
                            }
                            @keyframes turn2 {
                                0% {
                                    transform: rotateX(0deg);
                                }
                                100% {
                                    transform: rotateX(-90deg);
                                }
                            }
                            .flip-clock-wrapper ul li.flip-clock-active {
                                z-index: 3;
                            } /* SHADOW */
                            .flip-clock-wrapper ul.play li.flip-clock-before .up .shadow {
                                background: -moz-linear-gradient(
                                    top,
                                    rgba(0, 0, 0, 0.1) 0%,
                                    black 100%
                                );
                                background: -webkit-gradient(
                                    linear,
                                    left top,
                                    left bottom,
                                    color-stop(0%, rgba(0, 0, 0, 0.1)),
                                    color-stop(100%, black)
                                );
                                background: linear, top, rgba(0, 0, 0, 0.1) 0%, black 100%;
                                background: -o-linear-gradient(
                                    top,
                                    rgba(0, 0, 0, 0.1) 0%,
                                    black 100%
                                );
                                background: -ms-linear-gradient(
                                    top,
                                    rgba(0, 0, 0, 0.1) 0%,
                                    black 100%
                                );
                                background: linear, to bottom, rgba(0, 0, 0, 0.1) 0%, black 100%;
                                -webkit-animation: show 0.5s linear both;
                                -moz-animation: show 0.5s linear both;
                                animation: show 0.5s linear both;
                            }
                            .flip-clock-wrapper ul.play li.flip-clock-active .up .shadow {
                                background: -moz-linear-gradient(
                                    top,
                                    rgba(0, 0, 0, 0.1) 0%,
                                    black 100%
                                );
                                background: -webkit-gradient(
                                    linear,
                                    left top,
                                    left bottom,
                                    color-stop(0%, rgba(0, 0, 0, 0.1)),
                                    color-stop(100%, black)
                                );
                                background: linear, top, rgba(0, 0, 0, 0.1) 0%, black 100%;
                                background: -o-linear-gradient(
                                    top,
                                    rgba(0, 0, 0, 0.1) 0%,
                                    black 100%
                                );
                                background: -ms-linear-gradient(
                                    top,
                                    rgba(0, 0, 0, 0.1) 0%,
                                    black 100%
                                );
                                background: linear, to bottom, rgba(0, 0, 0, 0.1) 0%, black 100%;
                                -webkit-animation: hide 0.5s 0.3s linear both;
                                -moz-animation: hide 0.5s 0.3s linear both;
                                animation: hide 0.5s 0.3s linear both;
                            } /*DOWN*/
                            .flip-clock-wrapper ul.play li.flip-clock-before .down .shadow {
                                background: -moz-linear-gradient(
                                    top,
                                    black 0%,
                                    rgba(0, 0, 0, 0.1) 100%
                                );
                                background: -webkit-gradient(
                                    linear,
                                    left top,
                                    left bottom,
                                    color-stop(0%, black),
                                    color-stop(100%, rgba(0, 0, 0, 0.1))
                                );
                                background: linear, top, black 0%, rgba(0, 0, 0, 0.1) 100%;
                                background: -o-linear-gradient(
                                    top,
                                    black 0%,
                                    rgba(0, 0, 0, 0.1) 100%
                                );
                                background: -ms-linear-gradient(
                                    top,
                                    black 0%,
                                    rgba(0, 0, 0, 0.1) 100%
                                );
                                background: linear, to bottom, black 0%, rgba(0, 0, 0, 0.1) 100%;
                                -webkit-animation: show 0.5s linear both;
                                -moz-animation: show 0.5s linear both;
                                animation: show 0.5s linear both;
                            }
                            .flip-clock-wrapper ul.play li.flip-clock-active .down .shadow {
                                background: -moz-linear-gradient(
                                    top,
                                    black 0%,
                                    rgba(0, 0, 0, 0.1) 100%
                                );
                                background: -webkit-gradient(
                                    linear,
                                    left top,
                                    left bottom,
                                    color-stop(0%, black),
                                    color-stop(100%, rgba(0, 0, 0, 0.1))
                                );
                                background: linear, top, black 0%, rgba(0, 0, 0, 0.1) 100%;
                                background: -o-linear-gradient(
                                    top,
                                    black 0%,
                                    rgba(0, 0, 0, 0.1) 100%
                                );
                                background: -ms-linear-gradient(
                                    top,
                                    black 0%,
                                    rgba(0, 0, 0, 0.1) 100%
                                );
                                background: linear, to bottom, black 0%, rgba(0, 0, 0, 0.1) 100%;
                                -webkit-animation: hide 0.5s 0.3s linear both;
                                -moz-animation: hide 0.5s 0.3s linear both;
                                animation: hide 0.5s 0.2s linear both;
                            }
                            @-webkit-keyframes show {
                                0% {
                                    opacity: 0;
                                }
                                100% {
                                    opacity: 1;
                                }
                            }
                            @-moz-keyframes show {
                                0% {
                                    opacity: 0;
                                }
                                100% {
                                    opacity: 1;
                                }
                            }
                            @-o-keyframes show {
                                0% {
                                    opacity: 0;
                                }
                                100% {
                                    opacity: 1;
                                }
                            }
                            @keyframes show {
                                0% {
                                    opacity: 0;
                                }
                                100% {
                                    opacity: 1;
                                }
                            }
                            @-webkit-keyframes hide {
                                0% {
                                    opacity: 1;
                                }
                                100% {
                                    opacity: 0;
                                }
                            }
                            @-moz-keyframes hide {
                                0% {
                                    opacity: 1;
                                }
                                100% {
                                    opacity: 0;
                                }
                            }
                            @-o-keyframes hide {
                                0% {
                                    opacity: 1;
                                }
                                100% {
                                    opacity: 0;
                                }
                            }
                            @keyframes hide {
                                0% {
                                    opacity: 1;
                                }
                                100% {
                                    opacity: 0;
                                }
                            }
                        </style>`;
                    const info = this.formated_Time;
                    const pref = "flip-clock-";
                    const html = `
                    <div
                        id="clock_box"
                        style="
                            top: 2%;
                            width: 20%;
                            float: left;
                            left: 50px;
                            z-index: 1000;
                            position: fixed;
                        "
                    >
                        ${css}
                        <div id="content">
                            <div class="clock flip-clock-wrapper" id="flipclock">
                                <span class="flip-clock-divider"
                                    ><span class="flip-clock-label"></span
                                    ><span class="flip-clock-dot top"></span
                                    ><span class="flip-clock-dot bottom"></span
                                ></span>
                                <ul class="flip ahour">${this.create_Module(
                                    pref + "before",
                                    info.before[0]
                                )}${this.create_Module(
                        pref + "active",
                        this.time_arr[0]
                    )}</ul>
                                <ul class="flip bhour">${this.create_Module(
                                    pref + "before",
                                    info.before[1]
                                )}${this.create_Module(
                        pref + "active",
                        this.time_arr[1]
                    )}</ul>
                                <span class="flip-clock-divider"
                                    ><span class="flip-clock-label"></span
                                    ><span class="flip-clock-dot top"></span
                                    ><span class="flip-clock-dot bottom"></span
                                ></span>
                                <ul class="flip play aminute">${this.create_Module(
                                    pref + "before",
                                    info.before[2]
                                )}${this.create_Module(
                        pref + "active",
                        this.time_arr[2]
                    )}</ul>
                                <ul class="flip play bminute">${this.create_Module(
                                    pref + "before",
                                    info.before[3]
                                )}${this.create_Module(
                        pref + "active",
                        this.time_arr[3]
                    )}</ul>
                                <ul class="flip-clock-meridium">
                                    <li><a href="#">${(this.currentHour =
                                        this.getCurrentHour_status(
                                            info.hour
                                        ))}</a></li>
                                </ul>
                            </div>
                        </div>
                    </div>`;
                    document.body.insertAdjacentHTML("beforeend", html);
                    this.event();
                },
                getCurrentHour_status(hour) {
                    return hour > 11 ? "PM" : "AM";
                },
                currentHour: "",
                get Date_format() {
                    const date = new Date();
                    const h = date.getHours();
                    const hs = "".slice.call(`0${h.toString()}`, -2);
                    const ms = "".slice.call(
                        `0${date.getMinutes().toString()}`,
                        -2
                    );
                    return { h: h, string: hs + ms };
                },
                change(node) {
                    const time = this.Date_format;
                    [...time.string].forEach(
                        (s, index) =>
                            s !== this.time_arr[index] &&
                            this["f" + index](node, s, index)
                    );
                    const ts = this.getCurrentHour_status(time.h);
                    ts !== this.currentHour &&
                        this.change_time_status(node, ts);
                },
                get clock_box() {
                    return document.getElementById("clock_box");
                },
                event() {
                    setTimeout(() => {
                        const clock = this.clock_box;
                        let id = setInterval(() => {
                            !this.paused && this.change(clock);
                            if (this.firstRun) {
                                clearInterval(id);
                                setInterval(
                                    () => !this.paused && this.change(clock),
                                    60 * 1000
                                );
                            }
                        }, 1000);
                    }, 0);
                },
                /**
                 * @param {boolean} e
                 */
                set clock_paused(e) {
                    const box = this.clock_box;
                    box.style.display = e ? "none" : "block";
                    !e && this.change(box);
                    this.paused = e;
                },
                paused: false,
                main() {
                    this.clock();
                },
            },
            firstly: true,
            readerMode: false,
            get_article_title(node) {
                return node.getElementsByClassName("ContentItem-title")[0]
                    .innerText;
            },
            Reader(node) {
                /*
                1. adapted from http://www.360doc.com/
                2. css and html is adopted;
                3. rebuild js
                */
                const bgc = GM_getValue("articleBackground");
                const arr = new Array(7);
                let color = "#FFF";
                let bgcimg = "";
                if (bgc) {
                    for (let i = 1; i < 8; i++)
                        arr[i - 1] = bgc === `a_color${i}` ? " cur" : "";
                    if (bgc === "a_color7") {
                        const bgcM = `background-image: url(https://www.cnblogs.com/skins/coffee/images/bg_body.gif);`;
                        bgcimg = `background-image: url(${GM_getValue(
                            "readerbgimg"
                        )});`;
                        bgcimg.length < 50 &&
                            ((bgcimg = bgcM),
                            GM_setValue("readerbgimg_mark", false));
                    } else color = this.colors_list(bgc);
                } else {
                    arr.fill("", 0);
                    arr[5] = " cur";
                }
                let title = "";
                if (this.no_scroll) title = this.get_article_title(node);
                else {
                    title = get_Title();
                }
                const bgpic = GM_getValue("bgpreader");
                const meta = node.getElementsByClassName(
                    "AuthorInfo AnswerItem-authorInfo AnswerItem-authorInfo--related"
                );
                const mBackup =
                    '<div class="AuthorInfo AnswerItem-authorInfo AnswerItem-authorInfo--related" itemprop="author" itemscope="" itemtype="http://schema.org/Person">No_data</div>';
                const content = node.getElementsByClassName(
                    "RichText ztext CopyrightRichText-richText"
                );
                const cBackup = "No_data";
                const html = `
                <div
                    id="artfullscreen"
                    class="artfullscreen__"
                    tabindex="-1"
                    style="display: block; height: -webkit-fill-available; background-image: ${
                        bgpic ? `url(${bgpic})` : "none"
                    };"
                >
                    <style type="text/css">
                        div#artfullscreen {
                            width: 100%;
                            height: 100%;
                            background: #e3e3e3;
                            z-index: 999;
                            position: fixed;
                            left: 0;
                            top: 0;
                            text-align: left;
                            overflow: auto;
                            display: block;
                        }
                        div#artfullscreen__box {
                            padding: 32px 60px;
                            height: auto;
                            overflow: hidden;
                            background: ${color};
                            ${bgcimg}
                            box-shadow: 0 0 6px #999;
                            display: table;
                            margin: 20px auto;
                            min-height: 95%;
                        }
                        div#artfullscreen__box_scr {
                            word-break: break-word;
                            height: auto;
                            font-size: 16px;
                            color: #2f2f2f;
                            line-height: 1.5;
                            position: relative;
                        }
                        h2#titiletext {
                            font-size: 30px;
                            font-family: simhei;
                            color: #000;
                            line-height: 40px;
                            margin: 0 0 30px 0;
                            overflow: hidden;
                            text-align: left;
                            word-break: break-all;
                        }
                        section.end_article:before {
                            background: url(https://static.zhihu.com/heifetz/assets/bg@2x.033e5b2d.png)
                                repeat-x;
                            background-size: 20px 450px;
                            content: "";
                            height: 140px;
                            left: 0;
                            position: absolute;
                            width: 100%;
                        }
                    </style>
                    <div
                        id="artfullscreen__box"
                        class="artfullscreen__box"
                        style="width: 930px"
                    >
                        <div class="artfullscreen__box_scr" id="artfullscreen__box_scr">
                            <table style="width: 656px">
                                <tbody>
                                    <tr>
                                        <td id="artContent" style="max-width: 1000px">
                                            <h2 id="titiletext">
                                                ${title}
                                            </h2>
                                            ${
                                                meta.length > 0
                                                    ? meta[0].outerHTML
                                                    : mBackup
                                            }
                                            <hr>
                                            <div
                                                style="
                                                    width: 1200px;
                                                    margin: 0;
                                                    padding: 0;
                                                    height: 0;
                                                "
                                            ></div>
                                            <span class="RichText ztext CopyrightRichText-richText" itemprop="text" style="display: block; min-height: ${Math.floor(
                                                window.innerHeight * 0.52
                                            )}px;"></span>
                                        </td>
                                    </tr>
                                </tbody>
                            </table>
                            <section
                                class="end_article"
                                data-role="outer"
                                mpa-from-tpl="t"
                                style="margin-top: 25px"
                            >
                                <section
                                    mpa-from-tpl="t"
                                    style="border-width: 0px; border-style: none; border-color: initial"
                                >
                                    <section mpa-from-tpl="t" style="color: rgb(51, 141, 175)">
                                        <section mpa-from-tpl="t" style="text-align: center">
                                            <section
                                                mpa-from-tpl="t"
                                                style="
                                                    margin-right: auto;
                                                    margin-left: auto;
                                                    width: 30px;
                                                    display: inline-block;
                                                "
                                            >
                                                <img
                                                    data-ratio="2.25"
                                                    data-w="44"
                                                    data-width="100%"
                                                    style="
                                                        display: block;
                                                        width: 30px !important;
                                                        height: auto !important;
                                                        visibility: visible !important;
                                                    "
                                                    _width="30px"
                                                    src=""
                                                    crossorigin="anonymous"
                                                    alt="Image"
                                                    data-fail="0"
                                                />
                                            </section>
                                            <section mpa-from-tpl="t" style="font-size: 14px">
                                                —<span
                                                    data-brushtype="text"
                                                    style="padding-right: 5px; padding-left: 5px"
                                                    >END</span
                                                >—
                                            </section>
                                        </section>
                                    </section>
                                </section>
                                <p><br /></p>
                            </section>
                        </div>
                    </div>
                </div>`;
                document.body.insertAdjacentHTML("beforeend", html);
                setTimeout(() => {
                    const f = this.full;
                    f.getElementsByClassName(
                        "RichText ztext CopyrightRichText-richText"
                    )[0].innerHTML =
                        content.length > 0 ? content[0].innerHTML : cBackup;
                    this.Navigator();
                    this.toolBar(arr);
                    this.creatEvent(f);
                }, 50);
            },
            Navigator() {
                //this css adapted from @vizo, https://greasyfork.org/zh-CN/scripts/373008-%E7%99%BE%E5%BA%A6%E6%90%9C%E7%B4%A2%E4%BC%98%E5%8C%96sp
                const [statusl, titlel] = this.prevNode
                    ? ["", "previous answer"]
                    : [" disa", "no more content"];
                const [statusr, titler] = this.nextNode
                    ? ["", "next answer"]
                    : [" disa", "no more content"];
                const html = `
                        <<div id="reader_navigator">
                            <style type="text/css">
                                .readerpage-l,
                                .readerpage-r {
                                    width: 300px;
                                    height: 500px;
                                    overflow: hidden;
                                    cursor: pointer;
                                    position: fixed;
                                    top: 0;
                                    bottom: 0;
                                    margin: auto;
                                    z-index: 1000;
                                }
                                .readerpage-l.disa,
                                .readerpage-r.disa {
                                    cursor: not-allowed;
                                }
                                .readerpage-l:hover,
                                .readerpage-r:hover {
                                    background: rgba(100, 100, 100, 0.03);
                                }
                                .readerpage-l::before,
                                .readerpage-r::before {
                                    content: '';
                                    width: 100px;
                                    height: 100px;
                                    border-left: 2px solid #ADAFB0;
                                    border-bottom: 2px solid #ADAFB0;
                                    position: absolute;
                                    top: 0;
                                    bottom: 0;
                                    margin: auto;
                                }
                                .readerpage-l:hover::before,
                                .readerpage-r:hover::before {
                                    border-color: #fff;
                                }
                                .readerpage-l {
                                    left: 0;
                                }
                                .readerpage-l::before {
                                    left: 45%;
                                    transform: rotate(45deg);
                                }
                                .readerpage-r {
                                    right: 0;
                                }
                                .readerpage-r::before {
                                    right: 45%;
                                    transform: rotate(225deg);
                                }
                                @media (max-width: 1280px) {
                                    .readerpage-l,
                                    .readerpage-r {
                                        width: 150px;
                                    }
                                }
                            </style>
                            <div class="readerpage-l${statusl}" title=${escapeBlank(
                    titlel
                )}></div>
                            <div class="readerpage-r${statusr}" title=${escapeBlank(
                    titler
                )}></div>
                        </div>`;
                document.body.insertAdjacentHTML("beforeend", html);
            },
            check_answer_collected(qid, aid) {
                return collect_Answers.length === 0
                    ? false
                    : collect_Answers.some((e) =>
                          e.qid === qid
                              ? e.data.some((n) => n.aid === aid)
                              : false
                      );
            },
            change_Collected(tool, result) {
                const b =
                    tool.lastElementChild.lastElementChild
                        .previousElementSibling.previousElementSibling;
                const cl = b.className;
                const tmp = cl.endsWith(" on")
                    ? result
                        ? null
                        : cl.slice(0, cl.length - 3)
                    : result
                    ? cl + " on"
                    : null;
                if (tmp) {
                    b.className = tmp;
                    b.title = tmp.endsWith(" on")
                        ? `cancel the collection of ${this.content_type}`
                        : `add the ${this.content_type} to collection`;
                }
                tool = null;
            },
            check_article_collected(tool, aid) {
                dataBaseInstance.TableName = "collection";
                dataBaseInstance.check(aid).then(
                    (result) => this.change_Collected(tool, result),
                    () =>
                        colorful_Console(
                            {
                                title: "Warning:",
                                content: "failed to check article collection",
                            },
                            colorful_Console.colors.warning
                        )
                );
            },
            get_pocket_ico(f) {
                return f
                    ? ""
                    : "";
            },
            get check_pocket() {
                const recent = GM_getValue("recent");
                return (
                    recent &&
                    Array.isArray(recent) &&
                    recent.length > 0 &&
                    recent.some(
                        (r) => r.url.endsWith(`/${this.aid}`) && r.type === "p"
                    )
                );
            },
            toolBar(arr) {
                let collected = false;
                collect_Answers = GM_getValue("collect_a");
                collect_Answers && Array.isArray(collect_Answers)
                    ? this.content_type === "answer" &&
                      (collected = this.check_answer_collected(
                          this.qid,
                          this.aid
                      ))
                    : (collect_Answers = []);
                const is_pocket = this.check_pocket;
                const gifBase64 = `
                `;
                const jpgBase64 = `
                `;
                const html = `
                <div
                    class="artfullscreen_toolbar"
                    id="artfullscreen_toolbar"
                    style="z-index: 9999; left: 1540px"
                >
                    <style type="text/css">
                        .artfullscreen_toolbar {
                            width: 24px;
                            height: 166px;
                            position: fixed;
                            top: 15px;
                        }
                        a#artfullscreen_closer {
                            color: #4a4a4a;
                            width: 30px;
                            height: 24px;
                            line-height: 24px;
                            text-align: center;
                            display: block;
                            font-size: 28px;
                            font-weight: bold;
                            text-decoration: none;
                            float: right;
                        }
                        .artfullscreen_toolbar>div {
                            height: 19px;
                            clear: both;
                            padding: 18px 3px 0 0;
                        }
                        .artfullscreen_toolbar .a_colorlist,.artfullscreen_toolbar .fschange {
                            box-shadow: 0 0 5px #ccc;
                        }
                        .artfullscreen_toolbar .fschange input {
                            margin-right: 2px;
                        }
                        .artfullscreen_toolbar .a_bgcolor {
                            margin-left: 0;
                        }
                        .a_colorlist {
                            z-index: 2;
                            width: 146px;
                            height: 35px;
                            border: 1px solid #cbcbcb;
                            background: #fff;
                            position: absolute;
                            right: 38px;
                            top: 30px;
                            text-align: center;
                            display: none;
                        }
                        .a_colorlist span {
                            display: inline-block;
                            width: 13px;
                            height: 13px;
                            overflow: hidden;
                            border: solid 1px #cbcbcb;
                            margin: 10px 1px 0;
                            cursor: pointer;
                            font-size: 12px;
                        }
                        .a_colorlist img {
                            vertical-align: top;
                            visibility: hidden;
                        }
                        .a_colorlist .cur img {
                            visibility: visible !important;
                        }
                        .a_color1 {
                            background: #E3EDCD !important;
                        }
                        .a_color2 {
                            background: #f5f1e6 !important;
                        }
                        .a_color3 {
                            background: #B6B6B6 !important;
                        }
                        .a_color4 {
                            background: #FFF2E2 !important;
                        }
                        .a_color5 {
                            background: #FAF9DE !important;
                        }
                        .a_color6 {
                            background: #FFF !important;
                        }
                        .a_color7 {
                            background: #EC9857 !important;
                        }
                        button.Button.ContentItem-action.Button--plain.Button--withIcon.Button--withLabel.on {
                            color: #00a1d6;
                        }
                    </style>
                    <a href="#" class="artfullscreen_closer" id="artfullscreen_closer" title="exit reader">×</a>
                    <div class="d1">
                        <div class="a_bgcolor">
                            <img src=${jpgBase64} style="height: 20px; width:20px; margin-left: -1px;" />
                            <div class="a_colorlist" style="display: none">
                                <span class="a_color1${arr[0]}">
                                    <img src=${gifBase64}
                                /></span>
                                <span class="a_color2${arr[1]}">
                                    <img src=${gifBase64}
                                /></span>
                                <span class="a_color3${arr[2]}">
                                    <img src=${gifBase64}
                                /></span>
                                <span class="a_color4${arr[3]}">
                                    <img src=${gifBase64}
                                /></span>
                                <span class="a_color5${arr[4]}">
                                    <img src=${gifBase64}
                                /></span>
                                <span class="a_color6${arr[5]}">
                                    <img src=${gifBase64}
                                /></span>
                                <span class="a_color7${
                                    arr[6]
                                }" title="background image">
                                    <img src=${gifBase64}
                                /></span>
                            </div>
                        </div>
                    </div>
                    <div class="background_image_c">
                        <style>
                        i {
                            height: 20px;
                            width: 24px;
                            margin-left: -3px;
                        }
                        </style>
                        <i
                            class="disable_bgp"
                            title="disable background image"
                            style="
                                content: url();
                            "
                            >disable</i
                        >
                        <i
                            class="change_bgp"
                            title="click to change background image"
                            style="
                                margin-top: 10px;
                                content: url();
                            "
                            >change</i
                        >
                        <button
                            style="margin-left: -5px;margin-top: 10px;"
                            class="Button ContentItem-action Button--plain Button--withIcon Button--withLabel${
                                collected ? " on" : ""
                            }"
                            type="button"
                            title=${escapeBlank(
                                collected
                                    ? `cancel the collection of ${this.content_type}`
                                    : `add the ${this.content_type} to collection`
                            )}
                        >
                            <span style="display: inline-flex; align-items: center"
                                >&#8203;<svg
                                    class="Zi Zi--Star Button-zi"
                                    fill="currentColor"
                                    viewBox="0 0 24 24"
                                    width="2.0em"
                                    height="2.0em"
                                >
                                    <path
                                        d="M5.515 19.64l.918-5.355-3.89-3.792c-.926-.902-.639-1.784.64-1.97L8.56 7.74l2.404-4.871c.572-1.16 1.5-1.16 2.072 0L15.44 7.74l5.377.782c1.28.186 1.566 1.068.64 1.97l-3.89 3.793.918 5.354c.219 1.274-.532 1.82-1.676 1.218L12 18.33l-4.808 2.528c-1.145.602-1.896.056-1.677-1.218z"
                                        fill-rule="evenodd"
                                    ></path></svg
                            ></span>
                        </button>
                        <i
                            class="read_it_later ${is_pocket ? " on" : ""}"
                            title="read it later"
                            style="
                                height: 30px;
                                display: block;
                                margin-top: 10px;
                                content: url(${this.get_pocket_ico(is_pocket)});
                            "
                            >Pocket</i
                        >
                        <i
                            class="load_more"
                            title="load more answers"
                            style="
                                display: none;
                                margin-top: 10px;
                                content: url();
                            "
                            >load</i
                        >
                    </div>
                </div>`;
                document.body.insertAdjacentHTML("beforeend", html);
                this.toolBar_event();
            },
            color_chooser_display(e, display) {
                const target = e.target;
                const localName = target.localName;
                const c =
                    localName === "img"
                        ? target.nextElementSibling
                        : target.lastElementChild;
                c.style.display = display;
            },
            colors_list(name) {
                const colors = {
                    a_color1: "#E3EDCD",
                    a_color2: "#f5f1e6",
                    a_color3: "#B6B6B6",
                    a_color4: "#FFF2E2",
                    a_color5: "#FAF9DE",
                    a_color6: "#FFF",
                };
                const color = colors[name];
                return color ? color : "#FFF";
            },
            timeID: null,
            toolBar_event() {
                setTimeout(() => {
                    let tool = this.Toolbar;
                    let bg = tool.getElementsByClassName("a_bgcolor")[0];
                    bg.onmouseenter = (e) => {
                        if (this.timeID) {
                            clearTimeout(this.timeID);
                            this.timeID = null;
                        }
                        this.color_chooser_display(e, "block");
                    };
                    bg.onmouseleave = (e) =>
                        (this.timeID = setTimeout(() => {
                            this.timeID = null;
                            this.color_chooser_display(e, "none");
                        }, 300));
                    bg = null;
                    let closer = tool.getElementsByClassName(
                        "artfullscreen_closer"
                    )[0];
                    closer.onclick = (e) => {
                        if (
                            this.autoScroll.scrollState ||
                            !this.load_content_finished
                        )
                            return;
                        let node = e.target.parentNode;
                        let ic = 0;
                        let cn = node.className;
                        while (cn !== "artfullscreen_toolbar") {
                            node = node.parentNode;
                            if (!node || ic > 4) {
                                node = null;
                                break;
                            }
                            cn = node.className;
                            ic++;
                        }
                        node && (node.style.display = "none");
                        this.ShowOrExit(false);
                    };
                    let colorlist =
                        tool.getElementsByClassName("a_colorlist")[0];
                    colorlist.onclick = (e) => {
                        const target = e.target;
                        const className = target.className;
                        if (
                            className &&
                            className.startsWith("a_color") &&
                            !className.endsWith("cur")
                        ) {
                            const nodes = target.parentNode.children;
                            for (const node of nodes) {
                                const cn = node.className;
                                if (cn.endsWith("cur")) {
                                    node.className = cn.slice(0, cn.length - 4);
                                    break;
                                }
                            }
                            target.className = className + " cur";
                            const box =
                                document.getElementById("artfullscreen__box");
                            if (className === "a_color7") {
                                const img = GM_getValue("readerbgimg");
                                box.style.backgroundImage =
                                    (img && `url(${img})`) ||
                                    "url(https://www.cnblogs.com/skins/coffee/images/bg_body.gif)";
                            } else {
                                box.style.backgroundImage = "none";
                                box.style.background =
                                    this.colors_list(className);
                            }
                            GM_setValue("articleBackground", className);
                        }
                    };
                    tool.lastElementChild.onclick = (e) => {
                        let name = e.target.className;
                        let t = null;
                        if (!name || typeof name !== "string") {
                            const path = e.path;
                            for (const p of path) {
                                const n = p.className;
                                if (
                                    n &&
                                    typeof n === "string" &&
                                    n.startsWith("Button")
                                ) {
                                    t = p;
                                    name = n.endsWith(" on")
                                        ? "remove_collect"
                                        : "collect";
                                    break;
                                }
                            }
                        } else if (name) {
                            if (name.startsWith("Button")) {
                                t = e.target;
                                name = name.endsWith(" on")
                                    ? "remove_collect"
                                    : "collect";
                            } else if (name.startsWith("read_")) {
                                t = e.target;
                                name = "read_it_later";
                            }
                        }
                        name && this[name](t);
                    };
                    if (this.content_type === "article")
                        this.check_article_collected(tool, this.aid);
                    else tool = null;
                    colorlist = null;
                    closer = null;
                    const i = GM_getValue("bgpindex");
                    this.picIndex = i ? i : 0;
                }, 50);
            },
            content_type: null,
            check_Answers(qid) {
                return collect_Answers.findIndex((e) => e.qid === qid);
            },
            Redirect: {
                remove(type, href) {
                    this.columnsModule.recentModule.remove(type, href);
                },
                collect(node, pid) {
                    dataBaseInstance.TableName = "collection";
                    dataBaseInstance.additem(
                        this.home_Module.current_Column_id,
                        node,
                        pid
                    );
                },
                log(config) {
                    this.columnsModule.recentModule.log("c", config);
                },
            },
            remove_pocket(mode) {
                const href =
                    this.content_type === "answer"
                        ? `https://www.zhihu.com/question/${this.qid}/answer/${this.aid}`
                        : `https://zhuanlan.zhihu.com/p/${this.aid}`;
                if (mode) {
                    const config = {};
                    config.type = "p";
                    config.url = href;
                    config.update = Date.now();
                    config.title = get_Title();
                    this.Redirect.log.call(zhihu.Column, config);
                } else this.Redirect.remove.call(zhihu.Column, "p", href);
            },
            change_pocket_status() {
                const f = this.check_pocket;
                const tool = this.Toolbar;
                const p =
                    tool.lastElementChild.lastElementChild
                        .previousElementSibling;
                const cl = p.className;
                const tmp = cl.endsWith(" on")
                    ? f
                        ? null
                        : cl.slice(0, cl.length - 3)
                    : f
                    ? cl + " on"
                    : null;
                if (tmp) {
                    p.className = tmp;
                    p.style.content = `url(${this.get_pocket_ico(
                        tmp.endsWith(" on")
                    )})`;
                }
            },
            read_it_later(target) {
                const cl = target.className;
                const f = cl.endsWith(" on");
                target.style.content = `url(${this.get_pocket_ico(!f)})`;
                target.className = f ? cl.slice(0, cl.length - 3) : `${cl} on`;
                this.remove_pocket(!f);
            },
            collect(target) {
                const config = {};
                config.type = "c";
                const upt = Date.now();
                if (this.content_type === "answer") {
                    const index = this.check_Answers(this.qid);
                    const pref = "https://www.zhihu.com/question/";
                    if (index > -1) {
                        const c = collect_Answers[index];
                        const data = c.data;
                        if (!data.some((e) => e.aid === this.aid))
                            data.push({ aid: this.aid, update: upt });
                        config.title = c.title;
                        config.url = `${pref}${c.qid}/answer/${this.aid}`;
                    } else {
                        const info = {};
                        config.title = info.title = get_Title();
                        info.qid = this.qid;
                        info.data = [];
                        info.data.push({ aid: this.aid, update: upt });
                        config.url = `${pref}${this.qid}/answer/${this.aid}`;
                        collect_Answers.push(info);
                    }
                    GM_setValue("collect_a", collect_Answers);
                } else if (this.content_type === "article") {
                    this.Redirect.collect.call(
                        zhihu.Column,
                        this.curNode,
                        this.aid
                    );
                    config.url = `https://zhuanlan.zhihu.com/p/${this.aid}`;
                    config.title = get_Title();
                }
                config.update = upt;
                this.Redirect.log.call(zhihu.Column, config);
                this.change_collect_status(target, true);
            },
            change_collect_status(target, mode) {
                const className = target.className;
                target.title = mode
                    ? `cancel the collection of ${this.content_type}`
                    : `add the ${this.content_type} to collection`;
                target.className = mode
                    ? className + " on"
                    : className.slice(0, className.length - 3);
            },
            remove_collect(target) {
                if (
                    !confirm(
                        `are you sure want to remove this ${this.content_type}`
                    )
                )
                    return;
                let href = "";
                if (this.content_type === "answer") {
                    const index = this.check_Answers(this.qid);
                    if (index > -1) {
                        const data = collect_Answers[index].data;
                        if (data.length > 1) {
                            const i = data.findIndex((e) => e.aid === this.aid);
                            i > -1 && data.splice(i, 1);
                        } else collect_Answers.splice(index, 1);
                        GM_setValue("collect_a", collect_Answers);
                    }
                    href = `https://www.zhihu.com/question/${this.qid}/answer/${this.aid}`;
                } else if (this.content_type === "article") {
                    dataBaseInstance.TableName = "collection";
                    dataBaseInstance.dele(false, this.aid);
                    href = `https://zhuanlan.zhihu.com/p/${this.aid}`;
                }
                this.change_collect_status(target, false);
                this.Redirect.remove.call(zhihu.Column, "c", href);
            },
            picIndex: 0,
            disable_bgp() {
                this.full.style.backgroundImage = "none";
                this.picIndex = 0;
                GM_setValue("bgpindex", this.picIndex);
                GM_setValue("bgpreader", "");
            },
            change_bgp_timeID: null,
            change_bgp() {
                this.change_bgp_timeID && clearTimeout(this.change_bgp_timeID);
                this.change_bgp_timeID = setTimeout(() => {
                    this.change_bgp_timeID = null;
                    const pics = [
                        "d40e6fbabe339af2f9cc1b9a0c49b97c212417",
                        "50bfb5a296779e17d2901c6d13eb15cf215229",
                        "02ed4231ffac5cb7553df89a8ef6b3c4122737",
                        "826d9cecc220b11c5502f247ff046387247423",
                        "02f8f662ea840da01a9645fc2d18d3ac170812",
                        "f2a5e663cce16208e9e85c9a4bec502a101513",
                        "a3dc87ca1d8021f5091acf288c4fe0a4154893",
                    ];
                    const pref = "https://img.meituan.net/csc/";
                    const suffix = ".jpg";
                    this.backgroundImage_cache.reader(
                        this.full,
                        pref + pics[this.picIndex] + suffix
                    );
                    this.picIndex + 1 === pics.length
                        ? (this.picIndex = 0)
                        : (this.picIndex += 1);
                    GM_setValue("bgpindex", this.picIndex);
                }, 300);
            },
            load_more() {
                if (this.isRunning) return;
                this.try_status = false;
                this.isRunning = true;
                this.simulation_scroll(this.full, this.curNode, true);
                this.scroll_record = 0;
            },
            /**
             * @param {boolean} mode
             */
            set display_load_more(mode) {
                if (this.load_more_status === mode) return;
                this.Toolbar.lastElementChild.lastElementChild.style.display =
                    mode ? "block" : "none";
                this.load_more_status = mode;
            },
            get_video_info(v, video_id) {
                const api = `https://lens.zhihu.com/api/v4/videos/${video_id}`;
                xmlHTTPRequest(api).then(
                    (json) => {
                        typeof json === "string" && (json = JSON.parse(json));
                        let videoURL = "";
                        const vlb = json.playlist;
                        const keys = Object.keys(vlb);
                        let back = "";
                        for (const k of keys) {
                            if (k === "HD" || k === "FHD") {
                                videoURL = vlb[k].play_url;
                                break;
                            } else back = vlb[k].play_url;
                        }
                        if (!videoURL && !back) return;
                        else if (!videoURL) videoURL = back;
                        const picURL = json.cover_url;
                        this.replace_Video(v, picURL, videoURL);
                    },
                    (err) => console.log(err)
                );
            },
            get_video_ID(v) {
                let set = v.dataset.zaExtraModule;
                if (!set) set = v.dataset.zaModuleInfo;
                if (!set) return;
                typeof set === "string" && (set = JSON.parse(set));
                const content = set.card.content;
                if (content.is_playable === false) {
                    console.log("current video is not playable");
                    return;
                }
                this.get_video_info(v, content.video_id);
            },
            rawVideo_html(picURL, videoURL) {
                const html = `
                <img class="_video_cover" src=${picURL} style="object-fit: cover; position: absolute; left: 0px; top: 0px; width: 100%; height: 100%; z-index: 1;" />
                <video preload="metadata" width="100%" height="-webkit-fill-available" controls="">
                    <source
                        src=${videoURL}
                        type="video/mp4"
                    />
                </video>
                <div
                    class="_player_ico"
                    style="
                        position: absolute;
                        left: 50%;
                        top: 50%;
                        transform: translate(-50%, -50%);
                        z-index: 2;
                    "
                >
                    <div class="_play_ico" style="width: 50px; height: 50px">
                        <span class="_play_ico_content"
                            ><svg viewBox="0 0 72 72" class="_ico_content">
                                <g fill="none" fill-rule="evenodd">
                                    <circle
                                        cx="36"
                                        cy="36"
                                        r="36"
                                        fill="#FFF"
                                        fill-opacity=".95"
                                    ></circle>
                                    <path
                                        fill="#444"
                                        fill-rule="nonzero"
                                        d="M50.8350169,37.0602664 L29.4767217,49.9693832 C28.900608,50.3175908 28.1558807,50.1251285 27.8133266,49.5395068 C27.701749,49.3487566 27.6428571,49.1309436 27.6428571,48.9090213 L27.6428571,23.0907876 C27.6428571,22.4094644 28.1862113,21.8571429 28.8564727,21.8571429 C29.0747919,21.8571429 29.2890685,21.9170066 29.4767217,22.0304257 L50.8350169,34.9395425 C51.4111306,35.28775 51.6004681,36.0447682 51.257914,36.6303899 C51.154433,36.8072984 51.0090531,36.9550776 50.8350169,37.0602664 Z"
                                    ></path>
                                </g></svg
                        ></span>
                    </div>
                </div>`;
                return html;
            },
            replace_Video(v, picURL, videoURL) {
                let player = v.getElementsByClassName("VideoCard-player");
                if (player.length > 0) {
                    const html = this.rawVideo_html(picURL, videoURL, false);
                    player[0].children.length === 0
                        ? player[0].insertAdjacentHTML("afterbegin", html)
                        : (player[0].innerHTML = html);
                } else {
                    player = v.getElementsByClassName(
                        "ZVideoLinkCard-playerContainer"
                    );
                    if (player.length > 0) {
                        const html = this.rawVideo_html(picURL, videoURL, true);
                        player[0].style.paddingBottom = 0;
                        player[0].children.length === 0
                            ? player[0].insertAdjacentHTML("afterbegin", html)
                            : (player[0].innerHTML = html);
                    } else console.log("warning, the id of player is null");
                }
            },
            //click image to show the raw pic
            imgClick: {
                create(node, info) {
                    const html = `
                            <div class="ImageView is-active" style="padding-bottom: 10px">
                                <div class="ImageView-inner" style="overflow: auto">
                                    <img
                                        src=${info.url}
                                        class="ImageView-img"
                                        alt="preview"
                                        style="
                                            width: ${info.width};
                                            transform: ${info.transform};
                                            opacity: 1;
                                        "
                                    />
                                </div>
                            </div>`;
                    node.insertAdjacentHTML("beforeend", html);
                },
                imgPosition(info, target) {
                    const rw = target.dataset.rawwidth * 1;
                    const rh = target.dataset.rawheight * 1;
                    const owh = window.outerHeight * 1;
                    const wh = window.innerHeight * 1;
                    const ww = window.innerWidth * 1;
                    const sww = ww * 0.98;
                    const tw = target.width * 1;
                    const th = target.height * 1;
                    const xw = (ww - tw) / 2;
                    let sc = 0;
                    let yh = 0;
                    //if the width of raw pic is bigger than 98% width of window
                    if (rw > sww) {
                        sc = sww / tw;
                    } else {
                        sc = rw / tw;
                    }
                    if (rh > owh) {
                        //if the height is bigger than width
                        yh = rh > rw ? (rh - th) / 2 : (xw * th) / tw;
                    } else {
                        yh = (wh - th) / 2;
                    }
                    //margin
                    if (yh > wh) {
                        yh = yh / 2;
                        while (yh > wh) yh = yh / 2;
                    } else yh += 5;
                    info.transform = `translate(${xw}px, ${yh}px) scale(${sc.toFixed(
                        4
                    )})`;
                    info.width = `${tw}px`;
                },
                isExist: false,
                restore(n) {
                    //retore the status of navigator
                    const l = n.children[1];
                    const r = n.children[2];
                    l.className !== this.lbackupNav[0] &&
                        (l.className = this.lbackupNav[0]);
                    r.className !== this.rbackupNav[0] &&
                        (r.className = this.rbackupNav[0]);
                    l.title = this.lbackupNav[1];
                    r.title = this.rbackupNav[1];
                    n = null;
                },
                remove(n) {
                    if (!this.isExist) return;
                    this.ImageView.parentNode.parentNode.remove();
                    this.isExist = false;
                    this.currentPicURL = null;
                    this.restore(n);
                    this.imgList = null;
                    this.rbackupNav = null;
                    this.lbackupNav = null;
                    this.ImageView = null;
                    this.toolBar_display(true);
                },
                currentPicURL: null,
                ImageView: null,
                toolBar_display(mode) {
                    const tool = document.getElementById(
                        "artfullscreen_toolbar"
                    );
                    tool && (tool.style.display = mode ? "block" : "none");
                },
                showRawPic(box, target, n) {
                    const url = target.dataset.original;
                    if (!url) return;
                    const info = {};
                    info.url = url;
                    this.imgPosition(info, target);
                    this.create(box, info);
                    this.isExist = true;
                    setTimeout(() => {
                        const viewer =
                            box.getElementsByClassName("ImageView-inner");
                        if (viewer.length > 0)
                            this.ImageView = viewer[0].firstElementChild;
                    }, 10);
                    this.toolBar_display(false);
                    this.currentPicURL = url;
                    this.imgNav(box, n);
                },
                rbackupNav: null,
                lbackupNav: null,
                imgList: null,
                backUP(node, arr) {
                    arr.push(node.className);
                    arr.push(node.title);
                },
                imgNav(box, n) {
                    const imgs = box.getElementsByTagName("img");
                    this.imgList = [];
                    this.rbackupNav = [];
                    this.lbackupNav = [];
                    for (const img of imgs)
                        img.className.endsWith("lazy") &&
                            img.dataset.original &&
                            this.imgList.push(img);
                    const [titlel, namel, titler, namer] =
                        this.imgList.length === 1
                            ? [
                                  "no more picture",
                                  "readerpage-l disa",
                                  "no more picture",
                                  "readerpage-r disa",
                              ]
                            : [
                                  "previous picture",
                                  "readerpage-l",
                                  "next picture",
                                  "readerpage-r",
                              ];
                    const l = n.children[1];
                    const r = n.children[2];
                    this.backUP(l, this.lbackupNav);
                    this.backUP(r, this.rbackupNav);
                    l.title = titlel;
                    this.lbackupNav[0] !== namel && (l.className = namel);
                    r.title = titler;
                    this.rbackupNav[0] !== namer && (r.className = namer);
                },
                change_time_id: null,
                changPic(mode, show_status, f) {
                    this.change_time_id && clearTimeout(this.change_time_id);
                    this.change_time_id = setTimeout(() => {
                        this.change_time_id = null;
                        if (!this.ImageView) {
                            console.log(
                                "warning, the viewer of picture is null"
                            );
                            return;
                        }
                        const i = this.imgList.length - 1;
                        if (i === 0) return;
                        const index = this.imgList.findIndex(
                            (img) => img.dataset.original === this.currentPicURL
                        );
                        if ((index === i && mode) || (index === 0 && !mode)) {
                            show_status.main(
                                show_status.node ? null : f,
                                "no more picture"
                            );
                            return;
                        }
                        const imge = this.imgList[mode ? index + 1 : index - 1];
                        const url = imge.dataset.original;
                        this.currentPicURL = url;
                        const info = {};
                        this.imgPosition(info, imge);
                        this.ImageView.style.width = info.width;
                        this.ImageView.style.transform = info.transform;
                        this.ImageView.src = url;
                    }, 300);
                },
                no_mp4_giflist: null,
                gif_time_id: null,
                GifPlay(target, className) {
                    /*
                    some gifs are actually mp4 videos;
                    must check the gif whether has mp4 video
                    */
                    this.gif_time_id && clearTimeout(this.gif_time_id);
                    this.gif_time_id = setTimeout(() => {
                        this.gif_time_id = null;
                        if (className.endsWith("gif2mp4")) {
                            this.gif_pNode_className(target, !target.paused);
                            target.paused ? target.play() : target.pause();
                            return;
                        }
                        const url = target.src;
                        const reg = /(?<=-)\w+(?=_)/;
                        const match = url.match(reg);
                        if (!match) {
                            console.log("get id of gif pic fail");
                            return;
                        }
                        const id = match[0];
                        if (this.no_mp4_giflist) {
                            if (this.no_mp4_giflist.includes(id)) {
                                this.Gif_Pic(url, target);
                                return;
                            }
                        } else this.no_mp4_giflist = [];
                        this.Gif_MP4(id).then(
                            (src) => {
                                if (!src) {
                                    this.no_mp4_giflist.push(id);
                                    this.Gif_Pic(url, target);
                                    return;
                                }
                                target.insertAdjacentHTML(
                                    "beforebegin",
                                    this.gif_vidoe_raw(src, url)
                                );
                                setTimeout(() => {
                                    this.gif_pNode_className(target, false);
                                    target.previousElementSibling.play();
                                    target.className =
                                        className + " GifPlayer-gif2mp4Image";
                                }, 50);
                            },
                            () =>
                                Notification(
                                    "get the play url of gif_mp4 fail",
                                    "Reader Tips"
                                )
                        );
                    }, 300);
                },
                gif_pNode_className(target, mode) {
                    target.parentNode.className =
                        "GifPlayer" + (mode ? "" : " isPlaying");
                },
                gif_vidoe_raw(src, pic) {
                    const html = `
                        <video
                            class="ztext-gif GifPlayer-gif2mp4"
                            src=${src}"
                            data-thumbnail=${pic}
                            poster=${pic}
                            data-size="normal"
                            preload="metadata"
                            loop=""
                            playsinline=""
                            style="width: auto !important; height: auto !important;"
                        ></video>`;
                    return html;
                },
                Gif_Pic(url, target) {
                    const [a, b] = url.includes(".webp")
                        ? [".webp", ".jpg"]
                        : [".jpg", ".webp"];
                    target.src = url.replace(a, b);
                    this.gif_pNode_className(target, a === ".webp");
                },
                Gif_MP4(id) {
                    return new Promise((resolve, reject) => {
                        const api = `https://api.zhihu.com/gif2mp4/v2-${id}`;
                        xmlHTTPRequest(api).then(
                            (json) => {
                                typeof json === "string" &&
                                    (json = JSON.parse(json));
                                const plist = json.playlist;
                                let src = "";
                                let b = "";
                                for (const e of Object.keys(plist)) {
                                    if (e === "SD") {
                                        src = plist[e].play_url;
                                        break;
                                    } else b = plist[e].play_url;
                                }
                                resolve(src ? src : b);
                            },
                            (err) => {
                                console.log(err);
                                reject(null);
                            }
                        );
                    });
                },
                video_Play(video) {
                    video.nextElementSibling.style.display = "none";
                    video.previousElementSibling.style.display = "none";
                    video.play();
                },
                event(node, n, scroll) {
                    setTimeout(() => {
                        const box = node.lastElementChild;
                        box.onclick = (e) => {
                            if (scroll.scrollState) return;
                            const target = e.target;
                            const className = target.className;
                            if (className && typeof className === "string") {
                                if (className.endsWith("lazy"))
                                    this.showRawPic(box, target, n);
                                else if (className.startsWith("ztext-gif"))
                                    this.GifPlay(target, className);
                                else if (className === "_video_cover")
                                    this.video_Play(target.nextElementSibling);
                                else this.remove(n);
                            } else {
                                const localName = target.localName;
                                if (localName) {
                                    if (
                                        localName === "path" ||
                                        localName === "circle"
                                    ) {
                                        const paths = e.path;
                                        for (const p of paths) {
                                            if (p.className === "_player_ico") {
                                                this.video_Play(
                                                    p.previousElementSibling
                                                );
                                                break;
                                            }
                                        }
                                    } else if (
                                        localName === "img" &&
                                        target.dataset.original
                                    )
                                        this.showRawPic(box, target, n);
                                }
                            }
                        };
                    }, 100);
                },
            },
            scrollListen: false,
            video_list: null,
            loadedList: null,
            getVideo_element(f) {
                //if the video element is visible, load data
                if (this.video_list) {
                    this.video_list = null;
                    this.loadedList = null;
                }
                const videoas = f.getElementsByClassName("RichText-video");
                const videobs = f.getElementsByClassName("ZVideoLinkCard");
                const a = videoas.length;
                const b = videobs.length;
                if (a + b === 0) return;
                this.video_list = [];
                for (let k = 0; k < a; k++)
                    elementVisible.main(f, videoas[k])
                        ? this.get_video_ID(videoas[k])
                        : this.video_list.push(videoas[k]);
                for (let k = 0; k < b; k++)
                    elementVisible.main(f, videobs[k])
                        ? this.get_video_ID(videobs[k])
                        : this.video_list.push(videobs[k]);
                const i = this.video_list.length;
                if (i > 0) {
                    this.loadedList = new Array(i);
                    !this.scrollListen && this.scrollEvent(f);
                } else this.video_list = null;
            },
            scrollEvent(f) {
                let unfinished = false;
                let cache = 0;
                f.onscroll = () => {
                    cache += 1;
                    if (
                        cache < 5 ||
                        unfinished ||
                        !this.video_list ||
                        !this.loadedList
                    )
                        return;
                    cache = 0;
                    unfinished = true;
                    this.video_list.forEach((v, index) => {
                        if (!this.loadedList[index]) {
                            if (elementVisible.main(f, v)) {
                                this.loadedList[index] = true;
                                this.get_video_ID(v);
                            } else this.loadedList[index] = false;
                        }
                    });
                    if (this.loadedList.every((e) => e)) {
                        this.loadedList = null;
                        this.video_list = null;
                    }
                    unfinished = false;
                };
                this.scrollListen = true;
            },
            collected_answer_sync() {
                GM_addValueChangeListener(
                    "collect_a",
                    (name, oldValue, newValue, remote) =>
                        remote && (collect_Answers = newValue)
                );
            },
            creatEvent(f) {
                const n = this.nav;
                n.children[1].onclick = () => this.Previous(f);
                n.children[2].onclick = () => this.Next(f);
                this.loadLazy(f);
                this.answer_card_module.initial(f);
                this.imgClick.event(f, n, this.autoScroll);
                this.getVideo_element(f);
                setTimeout(() => this.time_module.main(), 300);
                this.backgroundImage_cache.article();
                this.collected_answer_sync();
            },
            turnPage: {
                main(mode, node) {
                    const overlap = 100;
                    const wh = window.innerHeight;
                    let height = wh - overlap;
                    height < 0 && (height = 0);
                    let top = node.scrollTop;
                    if (mode) top += height;
                    else top < height ? (top = 0) : (top -= height);
                    node.scrollTo(0, top);
                },
                start(mode, node) {
                    window.requestAnimationFrame(
                        this.main.bind(this, mode, node)
                    );
                },
            },
            autoScroll: {
                stepTime: 40,
                keyCount: 1,
                scrollState: false,
                scrollTime: null,
                scrollPos: null,
                node: null,
                pageScroll(TimeStamp) {
                    if (!this.scrollState) return;
                    const position = this.node.scrollTop;
                    if (this.scrollTime) {
                        this.scrollPos =
                            this.scrollPos !== null
                                ? this.scrollPos +
                                  (TimeStamp - this.scrollTime) / this.stepTime
                                : position;
                        this.node.scrollTo(0, this.scrollPos);
                    }
                    this.scrollTime = TimeStamp;
                    const h = this.node.scrollHeight - window.innerHeight;
                    position < h
                        ? window.requestAnimationFrame(
                              this.pageScroll.bind(this)
                          )
                        : this.stopScroll(true);
                },
                autoReader: null,
                scrollEnd: false,
                autoButton_event() {
                    createPopup(5);
                    const tips = document.getElementById("autoscroll-tips");
                    let buttons = tips.getElementsByTagName("button");
                    const id = setTimeout(() => {
                        tips.remove();
                        !this.auto_pause && this.autoReader();
                    }, 5000);
                    buttons[0].onclick = () => {
                        clearTimeout(id);
                        clearTimeout(this.timeID);
                        tips.remove();
                        !this.auto_pause && this.autoReader();
                    };
                    buttons[1].onclick = () => {
                        clearTimeout(id);
                        clearTimeout(this.timeID);
                        this.timeID = null;
                        tips.remove();
                    };
                    buttons = null;
                },
                timeID: null,
                auto_pause: false,
                remain_time: 0,
                stopScroll(mode) {
                    if (this.scrollState) {
                        this.scrollPos = null;
                        this.scrollTime = null;
                        this.scrollState = false;
                        this.keyCount = 1;
                        if (mode && this.autoReader && !this.scrollEnd) {
                            if (this.auto_pause) return;
                            const stop = Date.now();
                            const gap = Math.floor(
                                (stop - this.stopwatch) / 1000
                            );
                            const wtime =
                                gap < 30
                                    ? 2000
                                    : gap > 60 && gap < 90
                                    ? 3000
                                    : gap > 180
                                    ? 5500
                                    : 4000;
                            gap > 180
                                ? this.autoButton_event()
                                : setTimeout(
                                      () =>
                                          !this.auto_pause && this.autoReader(),
                                      wtime + this.remain_time
                                  );
                            this.timeID = setTimeout(() => {
                                this.timeID = null;
                                if (this.auto_pause) return;
                                this.keyCount = 2;
                                this.start();
                            }, wtime + this.remain_time + 3500);
                        } else {
                            this.timeID && clearTimeout(this.timeID);
                            this.timeID = null;
                            !this.autoReader && (this.node = null);
                            this.remain_time = 0;
                        }
                        this.stopwatch = 0;
                    }
                },
                speedUP() {
                    this.scrollState && this.stepTime < 10
                        ? (this.stepTime = 5)
                        : (this.stepTime -= 5);
                },
                slowDown() {
                    this.scrollState && this.stepTime > 100
                        ? (this.stepTime = 100)
                        : (this.stepTime += 5);
                },
                stopwatch: 0,
                startID: null,
                is_nurse: false,
                start() {
                    if (this.is_nurse) return;
                    this.keyCount += 1;
                    if (this.keyCount % 2 === 0) return;
                    if (this.scrollState) this.stopScroll(false);
                    else {
                        if (this.timeID) {
                            if (this.autoReader) return;
                            else clearTimeout(this.timeID);
                        }
                        this.stopwatch = Date.now();
                        this.scrollState = true;
                        !this.node &&
                            (this.node =
                                document.getElementById("artfullscreen"));
                        this.startID && clearTimeout(this.startID);
                        this.startID = setTimeout(() => {
                            this.startID = null;
                            window.requestAnimationFrame(
                                this.pageScroll.bind(this)
                            );
                        }, 600);
                    }
                },
            },
            scroll: {
                toTop(node) {
                    let hTop = node.scrollTop;
                    if (hTop === 0) return;
                    const rate = 8;
                    let sid = 0;
                    const scrollToTop = () => {
                        hTop = node.scrollTop;
                        if (hTop > 0) {
                            sid = window.requestAnimationFrame(scrollToTop);
                            node.scrollTo(0, hTop - hTop / rate);
                        } else {
                            sid !== 0 && window.cancelAnimationFrame(sid);
                        }
                    };
                    scrollToTop();
                },
                toBottom(node) {
                    let sid = 0;
                    let shTop = 0;
                    const initial = 100;
                    const scrollToBottom = () => {
                        const hTop = node.scrollTop || initial;
                        if (hTop !== shTop) {
                            shTop = hTop;
                            sid = window.requestAnimationFrame(scrollToBottom);
                            node.scrollTo(0, hTop + hTop);
                        } else {
                            sid !== 0 && window.cancelAnimationFrame(sid);
                            sid = 0;
                        }
                    };
                    scrollToBottom();
                },
            },
            autoReader_mode: false,
            autoReader() {
                if (this.autoReader_mode) {
                    this.autoReader_mode = false;
                    this.show_status.istimeout = true;
                    this.show_status.main(
                        null,
                        "auto reader mode has ben cancelled"
                    );
                    this.autoScroll.autoReader = null;
                    this.autoScroll.auto_pause = false;
                    this.autoScroll.node = null;
                    this.auto_pause_mode = false;
                } else {
                    this.show_status.main(this.full, "Auto Mode", 0, 0, false);
                    this.autoReader_mode = true;
                    this.autoScroll.autoReader = this.Next.bind(this);
                }
            },
            auto_pause_mode: false,
            auto_pause() {
                this.autoScroll.auto_pause = this.auto_pause_mode =
                    !this.auto_pause_mode;
                this.show_status.auto_scroll_change(
                    this.auto_pause_mode ? "Auto Mode_Paused" : "Auto Mode"
                );
                this.auto_pause_mode && (this.autoScroll.remain_time = 0);
            },
            keyEvent(keyCode, shift) {
                this.imgClick.isExist
                    ? keyCode === 37
                        ? this.Previous()
                        : keyCode === 39
                        ? this.Next()
                        : null
                    : shift
                    ? keyCode === 65
                        ? this.autoReader()
                        : null
                    : keyCode === 65
                    ? this.autoReader_mode && this.auto_pause()
                    : keyCode === 84
                    ? !this.autoScroll.scrollState &&
                      this.scroll.toTop(this.full)
                    : keyCode === 78
                    ? !this.autoScroll.scrollState &&
                      this.turnPage.start(true, this.full)
                    : keyCode === 85
                    ? !this.autoScroll.scrollState &&
                      this.turnPage.start(false, this.full)
                    : keyCode === 82
                    ? !this.autoScroll.scrollState &&
                      this.scroll.toBottom(this.full)
                    : keyCode === 192
                    ? this.autoScroll.start()
                    : keyCode === 187
                    ? this.autoScroll.speedUP()
                    : keyCode === 189
                    ? this.autoScroll.slowDown()
                    : keyCode === 37
                    ? this.Previous()
                    : keyCode === 39
                    ? this.Next()
                    : keyCode === 27
                    ? this.ShowOrExit(false)
                    : zhihu.multiSearch.main(keyCode);
            },
            changeNav(node) {
                const pre = node.children[1];
                const pName = pre.className;
                const [npName, titlel] = this.prevNode
                    ? ["readerpage-l", "previous answer"]
                    : ["readerpage-l disa", "no more content"];
                if (pName !== npName) {
                    pre.className = npName;
                    pre.title = titlel;
                }
                const next = node.children[2];
                const nextName = next.className;
                const [nnextName, titler] = this.nextNode
                    ? ["readerpage-r", "next answer"]
                    : ["readerpage-r disa", "no more content"];
                if (nextName !== nnextName) {
                    next.className = nnextName;
                    next.title = titler;
                }
            },
            getAnswerID(node) {
                let first = node.firstElementChild;
                if (first.className !== "ContentItem AnswerItem")
                    first = this.getAnswerItem(node);
                if (!first) {
                    console.log("warning, the structure of node has changed");
                    return null;
                }
                const ats = first.attributes;
                if (ats)
                    for (const a of ats) if (a.name === "name") return a.value;
            },
            /**
             * @param {any} node
             */
            set answerID(node) {
                this.aid = this.getAnswerID(node);
            },
            // trigger the scroll event to load more answers;
            simulation_scroll(f, node, mode = false) {
                setTimeout(() => {
                    this.overFlow = false;
                    f.style.overflow = "hidden";
                    node.scrollIntoView();
                    setTimeout(() => {
                        (this.scroll_record < 5 || mode) &&
                            window.scrollTo(0, 0.98 * this.DDSH);
                        setTimeout(() => {
                            this.scroll_record -= 1;
                            this.navPannel = this.curNode = node;
                            const time =
                                this.allAnswser_loaded || this.nextNode
                                    ? 50
                                    : 650;
                            setTimeout(() => {
                                this.overFlow = true;
                                f.style.overflow = "auto";
                                this.isRunning = false;
                                mode
                                    ? this.nextNode &&
                                      Notification(
                                          "load more answers successfully",
                                          "Tips"
                                      )
                                    : f.focus();
                            }, time);
                        }, 350);
                    }, 350);
                }, 60);
            },
            isRunning: false,
            scroll_record: 0,
            chang_time_id: null,
            //settimeout => prevent too many times of operation
            changeContent(node, mode = true) {
                this.chang_time_id && clearTimeout(this.chang_time_id);
                this.chang_time_id = setTimeout(() => {
                    this.chang_time_id = null;
                    if (
                        (!this.autoReader_mode && this.autoScroll.node) ||
                        this.isRunning
                    )
                        return;
                    this.isRunning = true;
                    const cName = "RichText ztext CopyrightRichText-richText";
                    const aName =
                        "AuthorInfo AnswerItem-authorInfo AnswerItem-authorInfo--related";
                    const f = this.full;
                    const content = node.getElementsByClassName(cName);
                    const author = node.getElementsByClassName(aName);
                    const cn = f.getElementsByClassName(cName)[0];
                    if (content.length > 0) {
                        cn.innerHTML = content[0].innerHTML;
                        this.autoReader_mode &&
                            this.setRemainTime(content[0].innerText.length);
                    } else cn.innerHTML = "No data";
                    f.getElementsByClassName(aName)[0].innerHTML =
                        author.length > 0 ? author[0].innerHTML : "No data";
                    if (mode) {
                        this.answerID = node;
                        this.isSimple_page || this.allAnswser_loaded
                            ? (this.navPannel = this.curNode = node)
                            : this.simulation_scroll(f, node);
                    } else this.ShowOrExit(true);
                    if (this.no_scroll) {
                        f.getElementsByTagName("h2")[0].innerText =
                            this.get_article_title(node);
                        this.set_current_Type(node);
                    }
                    this.content_type === "answer"
                        ? this.change_Collected(
                              this.Toolbar,
                              this.check_answer_collected(this.qid, this.aid)
                          )
                        : this.check_article_collected(this.Toolbar, this.aid);
                    this.change_pocket_status();
                    this.loadLazy(f);
                    this.answer_card_module.initial(f);
                    this.getVideo_element(f);
                    setTimeout(() => {
                        f.scrollTo(0, 0);
                        (this.allAnswser_loaded ||
                            !mode ||
                            this.isSimple_page) &&
                            ((this.isRunning = false), f.focus());
                    }, 50);
                }, 300);
            },
            setRemainTime(len) {
                if (
                    this.auto_pause_mode ||
                    this.full.scrollHeight > window.innerHeight
                )
                    return;
                const w =
                    len < 30
                        ? 500
                        : len < 50
                        ? 1000
                        : len < 100
                        ? 1500
                        : (len / 100) * 500 + 1500;
                this.autoScroll.remain_time = w > 6000 ? 6000 : w;
            },
            Next(f) {
                if (!this.load_content_finished) return;
                this.imgClick.isExist
                    ? this.imgClick.changPic(true, this.show_status, f)
                    : this.nextNode
                    ? (this.changeContent(this.nextNode),
                      this.autoReader_mode &&
                          (this.autoScroll.scrollEnd = false))
                    : this.autoReader_mode &&
                      (this.autoScroll.scrollEnd = true);
            },
            Previous(f) {
                if (!this.load_content_finished) return;
                this.imgClick.isExist
                    ? this.imgClick.changPic(false, this.show_status, f)
                    : this.prevNode &&
                      (this.autoReader_mode &&
                          (this.autoScroll.scrollEnd = false),
                      this.changeContent(this.prevNode));
            },
            get nav() {
                return document.getElementById("reader_navigator");
            },
            get full() {
                return document.getElementById("artfullscreen");
            },
            /**
             * @param {boolean} mode
             */
            set Element_display(mode) {
                const display = mode ? "block" : "none";
                const n = this.nav;
                n && (n.style.display = display);
                const f = this.full;
                f && (f.style.display = display);
                const tool = this.Toolbar;
                tool && (tool.style.display = display);
                this.time_module.clock_paused = !mode;
            },
            ShowOrExit(mode) {
                if (!mode && (this.isRunning || this.autoScroll.scrollState))
                    return;
                this.Element_display = mode;
                if (mode) return;
                /*
                exit reader mode, then move to the position of current node
                wait the reader is hidden, scroll to current answer
                */
                this.overFlow = false;
                this.readerMode = mode;
                if (document.title === "出了一点问题") {
                    confirm("the webpage has crashed, is reload?") &&
                        location.reload();
                    return;
                }
                this.curNode.offsetTop !== window.pageYOffset &&
                    (this.isSimple_page ||
                        this.nextNode ||
                        this.allAnswser_loaded) &&
                    setTimeout(() => this.curNode.scrollIntoView(), 300);
                this.no_scroll
                    ? change_Title("IGNORANCE IS STRENGTH")
                    : this.ctrl_click.call(zhihu, false);
            },
            // load lazy pic
            loadLazy(node) {
                const imgs = node.getElementsByTagName("img");
                for (const img of imgs) {
                    const name = img.className;
                    name &&
                        name.endsWith("lazy") &&
                        img.src.startsWith("data:image/svg+xml") &&
                        (img.src = img.dataset.actualsrc);
                }
            },
            // load the answer card information
            answer_card_module: {
                card_raw(info, href) {
                    const html = `
                        <div class="RichText-LinkCardContainer">
                            <a
                                target="_blank"
                                href="${href}"
                                data-draft-node="block"
                                data-draft-type="link-card"
                                data-text="${info.title}"
                                class="LinkCard new css-1wr1m8"
                                data-za-detail-view-id="172"
                                ><span class="LinkCard-contents"
                                    ><span class="LinkCard-title two-line"
                                        >${info.title}</span
                                    ><span class="LinkCard-desc"
                                        >${info.voteup_count} 赞同 · ${info.comment_count} 评论<span class="LinkCard-tag">回答</span></span
                                    ></span
                                ></a
                            >
                        </div>`;
                    return html;
                },
                load_card_info(curls, cnodes) {
                    const api = `https://www.zhihu.com/api/v4/link_card_infos?urls=${curls.join(
                        ","
                    )}`;
                    xmlHTTPRequest(api).then(
                        (json) => {
                            if (typeof result === "string")
                                json = JSON.parse(json);
                            cnodes.forEach((c) => {
                                const info = json[c.url];
                                if (info)
                                    c.node.outerHTML = this.card_raw(
                                        info,
                                        c.url
                                    );
                            });
                        },
                        () => console.log("failed to load answer card info")
                    );
                },
                external_raw(info, href) {
                    const url = href.includes("link.zhihu.com/?target=")
                        ? href.split("link.zhihu.com/?target=")[1]
                        : href;
                    const html = `
                        <div class="RichText-LinkCardContainer">
                            <a
                                target="_blank"
                                href="${url}"
                                data-draft-node="block"
                                data-draft-type="link-card"
                                data-text="${info.text}"
                                class="LinkCard new css-1wr1m8"
                                data-image="${info.image}"
                                data-image-width="1920"
                                data-image-height="1080"
                                data-za-detail-view-id="${info.zaDetailViewId}"
                                ><span class="LinkCard-contents"
                                    ><span class="LinkCard-title two-line"
                                        >${info.text}</span
                                    ><span class="LinkCard-desc"
                                        ><span style="display: inline-flex; align-items: center"
                                            >&ZeroWidthSpace;<svg
                                                class="Zi Zi--InsertLink"
                                                fill="currentColor"
                                                viewBox="0 0 24 24"
                                                width="14"
                                                height="14"
                                            >
                                                <path
                                                    d="M13.414 4.222a4.5 4.5 0 1 1 6.364 6.364l-3.005 3.005a.5.5 0 0 1-.707 0l-.707-.707a.5.5 0 0 1 0-.707l3.005-3.005a2.5 2.5 0 1 0-3.536-3.536l-3.005 3.005a.5.5 0 0 1-.707 0l-.707-.707a.5.5 0 0 1 0-.707l3.005-3.005zm-6.187 6.187a.5.5 0 0 1 .638-.058l.07.058.706.707a.5.5 0 0 1 .058.638l-.058.07-3.005 3.004a2.5 2.5 0 0 0 3.405 3.658l.13-.122 3.006-3.005a.5.5 0 0 1 .638-.058l.069.058.707.707a.5.5 0 0 1 .058.638l-.058.069-3.005 3.005a4.5 4.5 0 0 1-6.524-6.196l.16-.168 3.005-3.005zm8.132-3.182a.25.25 0 0 1 .353 0l1.061 1.06a.25.25 0 0 1 0 .354l-8.132 8.132a.25.25 0 0 1-.353 0l-1.061-1.06a.25.25 0 0 1 0-.354l8.132-8.132z"
                                                ></path></svg></span
                                        >${url}</span
                                    ></span
                                ><span class="LinkCard-image" style="height: 60px"
                                    ><img
                                        src="${info.image}"
                                        alt="" /></span
                            ></a>
                        </div>`;
                    return html;
                },
                check_data(a, node, href) {
                    // some card have contained the information
                    const info = a[0].dataset;
                    if (info.image) {
                        node.outerHTML = this.external_raw(
                            info,
                            decodeURIComponent(href)
                        );
                        return true;
                    }
                    return false;
                },
                get_card_answer(node) {
                    const cards = node.getElementsByClassName(
                        "RichText-LinkCardContainer"
                    );
                    if (cards.length === 0) return;
                    const curls = [];
                    const cnodes = [];
                    for (const card of cards) {
                        if (
                            card.getElementsByClassName(
                                "LinkCard-title loading"
                            ).length === 0
                        )
                            continue;
                        const a = card.getElementsByTagName("a");
                        if (a.length > 0) {
                            const href = a[0].href;
                            if (!href || this.check_data(a, card, href))
                                continue;
                            cnodes.push({ url: href, node: card });
                            curls.push(href);
                        }
                    }
                    curls.length > 0 && this.load_card_info(curls, cnodes);
                },
                initial(node) {
                    this.get_card_answer(node);
                },
            },
            /**
             * @param {boolean} mode
             */
            set overFlow(mode) {
                document.documentElement.style.overflow = mode
                    ? "hidden"
                    : "auto";
            },
            getAnswerItem(node) {
                const item = node.getElementsByClassName(
                    "ContentItem AnswerItem"
                );
                return item.length > 0 ? item[0] : null;
            },
            //check the item whether is blocked
            blockCheck(arg) {
                if (
                    Object.prototype.toString.call(arg) ===
                    "[object HTMLCollection]"
                ) {
                    for (const e of arg) {
                        if (e.style.display === "none") continue;
                        const item = this.getAnswerItem(e);
                        if (item && item.style.display === "none") continue;
                        return e;
                    }
                } else {
                    if (arg.style.display === "none") return null;
                    const item = this.getAnswerItem(arg);
                    if (item && item.style.display === "none") return null;
                    else return arg;
                }
            },
            show_status: {
                node: null,
                show(f, tips, info) {
                    const html = `
                        <div
                            id="load_status"
                            style="
                                top: 0%;
                                z-index: 1000;
                                position: fixed;
                                height: 24px;
                                width: 50%;
                                font-size: 14px;
                                font-weight: 500;
                                margin-left: 25%;
                                text-align: center;
                                color: ${info.color};
                                background: ${info.bgc};
                                opacity: 0.8;
                                box-shadow: 0 0 15px #FFBB59;
                            "
                        >
                        ${info.text + tips}...
                        </div>`;
                    this.remove();
                    const anode = document.createElement("div");
                    f.appendChild(anode);
                    anode.outerHTML = html;
                    setTimeout(() => (this.node = f.lastElementChild), 0);
                },
                remove() {
                    if (this.node) {
                        this.timeID && clearTimeout(this.timeID);
                        this.node.remove();
                        this.node = null;
                        this.timeID = null;
                        this.istimeout = true;
                        this.backText = "";
                        this.backColor = "";
                    }
                },
                backText: "",
                backColor: "",
                changeTips(tips, time, info) {
                    if (this.node) {
                        if (this.istimeout) this.timeout(time);
                        else {
                            this.backText = this.node.innerText;
                            this.backColor = this.node.style.background;
                            this.timeID = setTimeout(() => {
                                this.timeID = null;
                                this.node.innerText = this.backText;
                                this.node.style.background = this.backColor;
                            }, time);
                        }
                        this.node.innerText = tips;
                        this.node.style.background = info.bgc;
                    }
                },
                auto_scroll_change(tips) {
                    this.node.innerText = tips;
                },
                timeout(time) {
                    this.timeID && clearTimeout(this.timeID);
                    this.timeID = setTimeout(
                        () => ((this.timeID = null), this.remove()),
                        time
                    );
                },
                timeID: null,
                istimeout: true,
                main(f, tips, type = 1, time = 2500, istimeout = true) {
                    const types = {
                        0: {
                            text: "",
                            bgc: "#FFBB59",
                            color: "#0A0A0D",
                        },
                        1: {
                            text: "Tips: ",
                            bgc: "#E1B5BA",
                            color: "#0A0A0D",
                        },
                        2: {
                            text: "Warning: ",
                            bgc: "#FF3300",
                            color: "#0A0A0D",
                        },
                    };
                    const info = types[type];
                    f
                        ? this.show(f, tips, info)
                        : this.changeTips(info.text + tips, time, info);
                    this.istimeout &&
                        (this.istimeout = istimeout) &&
                        this.timeout(time);
                },
            },
            get Toolbar() {
                return document.getElementById("artfullscreen_toolbar");
            },
            allAnswser_loaded: false,
            try_status: false,
            load_more_status: false,
            /**
             * @param {{ parentNode: any; }} pnode
             */
            set navPannel(pnode) {
                //---------------------------------------check if the node has pre and next node
                const className = pnode.className;
                if (className === "QuestionAnswer-content") {
                    this.prevNode = null;
                    const list = document.getElementsByClassName("List-item");
                    this.nextNode =
                        list.length > 0 ? this.blockCheck(list) : null;
                } else {
                    let next = pnode.nextElementSibling;
                    this.nextNode = null;
                    if (next) {
                        let nextName = next.className;
                        const arr = ["Pc-word", "List-item"];
                        let a = arr.indexOf(nextName);
                        let b = null;
                        while (a > -1) {
                            if (a === 0) {
                                b = next.nextElementSibling;
                                next.remove();
                                next = b;
                            }
                            if (this.blockCheck(next)) {
                                this.nextNode = next;
                                break;
                            } else {
                                next = next.nextElementSibling;
                                if (!next) break;
                                nextName = next.className;
                                a = arr.indexOf(nextName);
                            }
                        }
                    }
                    let pre = pnode.previousElementSibling;
                    this.prevNode = null;
                    if (pre) {
                        let pName = pre.className;
                        if (pName === "List-header") {
                            const c = document.getElementsByClassName(
                                "QuestionAnswer-content"
                            );
                            this.prevNode =
                                c.length > 0 ? this.blockCheck(c[0]) : null;
                        } else {
                            const arr = ["Pc-word", "List-item"];
                            let a = arr.indexOf(pName);
                            let b = null;
                            while (a > -1) {
                                if (a === 0) {
                                    b = pre.previousElementSibling;
                                    pre.remove();
                                    pre = b;
                                }
                                if (this.blockCheck(pre)) {
                                    this.prevNode = pre;
                                    break;
                                } else {
                                    pre = pre.previousElementSibling;
                                    if (!pre) break;
                                    pName = pre.className;
                                    a = arr.indexOf(pName);
                                }
                            }
                        }
                    }
                }
                if (
                    this.allAnswser_loaded ||
                    (!this.nextNode &&
                        (this.isSimple_page ||
                            (this.allAnswser_loaded = this.is_scrollBottom)))
                ) {
                    this.isShowTips &&
                        this.show_status.main(
                            this.show_status.node ? null : this.full,
                            "all answers have been loaded"
                        );
                    this.isShowTips = false;
                } else if (!(this.nextNode || this.try_status)) {
                    window.scrollTo(0, 0.75 * this.DDSH);
                    setTimeout(() => {
                        window.scrollTo(0, 0.98 * this.DDSH);
                        setTimeout(
                            () => (
                                (this.try_status = true),
                                (this.navPannel = pnode)
                            ),
                            300
                        );
                    }, 300);
                    return;
                }
                this.changeNav(this.nav);
                !this.isSimple_page &&
                    (this.display_load_more = !this.nextNode);
            },
            removeADs() {
                const ads = document.getElementsByClassName("Pc-word");
                let i = ads.length;
                if (i > 0) for (i; i--; ) ads[i].remove();
            },
            backgroundImage_cache: {
                _request(url) {
                    return new Promise((resolve, reject) => {
                        xmlHTTPRequest(url, 2500, "blob").then(
                            (blob) => {
                                const file = new FileReader();
                                file.readAsDataURL(blob);
                                file.onload = (result) =>
                                    result.target.readyState === 2
                                        ? resolve(result.target.result)
                                        : reject("file state");
                                file.onerror = (err) => {
                                    console.log(err);
                                    reject("file err");
                                };
                            },
                            (err) => {
                                console.log(err);
                                reject("xml err");
                            }
                        );
                    });
                },
                reader(f, url) {
                    this._request(url).then(
                        (base64) => {
                            f.style.backgroundImage = `url(${base64})`;
                            GM_setValue("bgpreader", base64);
                            colorful_Console.main(
                                {
                                    title: "changeBGP",
                                    content:
                                        "background image has been cached successfully",
                                },
                                colorful_Console.colors.Tips
                            );
                        },
                        () =>
                            Notification(
                                "failed to get background image",
                                "Warning"
                            )
                    );
                },
                try_status: false,
                article(back) {
                    if (GM_getValue("readerbgimg_mark")) return;
                    const url =
                        back ||
                        "https://www.cnblogs.com/skins/coffee/images/bg_body.gif";
                    this._request(url).then(
                        (base64) => {
                            GM_setValue("readerbgimg", base64);
                            GM_setValue("readerbgimg_mark", true);
                            console.log(
                                "background image has been cached successfully"
                            );
                        },
                        (err) => {
                            if (err.startsWith("file")) return;
                            if (this.try_status) {
                                colorful_Console(
                                    {
                                        title: "warning:",
                                        content:
                                            "failed to cach article background image",
                                    },
                                    colorful_Console.colors.warning
                                );
                                return;
                            }
                            this.try_status = true;
                            const b =
                                "https://img.meituan.net/csc/decb7d168d512de5341614a7e22b26e848725.gif";
                            this.article(b);
                        }
                    );
                },
            },
            nextNode: null,
            prevNode: null,
            curNode: null,
            aid: null,
            qid: null,
            isSimple_page: false,
            isShowTips: false,
            initial_id: null,
            ctrl_click(mode) {
                this.Filter.isReader = mode;
                !(mode || location.href.endsWith("#")) &&
                    (this.Filter.is_jump = true);
            },
            get is_scrollBottom() {
                return (
                    document.getElementsByClassName(
                        "Button QuestionAnswers-answerButton Button--blue Button--spread"
                    ).length > 0
                );
            },
            get DDSH() {
                return document.documentElement.scrollHeight;
            },
            initial_set(p) {
                setTimeout(() => {
                    this.overFlow = true;
                    this.navPannel = p;
                }, 100);
            },
            Change(node, aid) {
                aid === this.aid
                    ? this.ShowOrExit(true)
                    : this.changeContent(node, false);
            },
            load_content_finished: true,
            no_scroll: false,
            set_current_Type(node, aid) {
                const a = node
                    .getElementsByClassName("ContentItem-title")[0]
                    .getElementsByTagName("a")[0];
                if (!aid || (aid && this.aid !== aid)) {
                    const href = a.href;
                    this.content_type = href.includes("zhuanlan")
                        ? "article"
                        : href.includes("daily")
                        ? "daily"
                        : "answer";
                    this.qid =
                        this.content_type === "answer"
                            ? href.match(/(?<=question\/)\d+/)[0]
                            : null;
                }
                change_Title(a.innerText);
            },
            main(pnode, aid, mode = false) {
                this.initial_id && clearTimeout(this.initial_id);
                this.initial_id = setTimeout(() => {
                    this.initial_id = null;
                    this.load_content_finished = false;
                    if (this.no_scroll) {
                        this.set_current_Type(pnode, aid);
                        this.isSimple_page = true;
                    } else {
                        const pathname = location.pathname;
                        if (this.firstly) {
                            this.content_type = "answer";
                            this.qid = pathname.match(/(?<=question\/)\d+/)[0];
                        }
                        this.isSimple_page = pathname.includes("/answer/");
                        this.removeADs();
                        this.ctrl_click.call(zhihu, true);
                    }
                    this.curNode =
                        pnode.className === "ContentItem AnswerItem"
                            ? pnode.parentNode.parentNode
                            : pnode;
                    this.firstly ? this.Reader(pnode) : this.Change(pnode, aid);
                    this.aid = aid;
                    if (this.isSimple_page) this.initial_set(this.curNode);
                    else {
                        if ((this.allAnswser_loaded = this.is_scrollBottom))
                            this.initial_set(this.curNode);
                        else {
                            setTimeout(() => {
                                window.scrollTo(0, 0.75 * this.DDSH);
                                setTimeout(
                                    () => this.initial_set(this.curNode),
                                    300
                                );
                            }, 100);
                        }
                    }
                    this.firstly = false;
                    this.readerMode = true;
                    this.isShowTips = true;
                    mode && (this.autoScroll.is_nurse = true);
                    setTimeout(() => {
                        this.full.focus();
                        if (mode) {
                            this.autoScroll.is_nurse = false;
                            this.autoReader();
                            this.autoScroll.keyCount = 2;
                            this.autoScroll.start();
                            this.load_content_finished = true;
                        } else this.load_content_finished = true;
                    }, 1500);
                    //shift focus to reader, whick can make navigator control key can scroll the page
                }, 300);
            },
        },
        getData() {
            blackName = GM_getValue("blackname");
            if (blackName && Array.isArray(blackName)) {
                if (blackName.length > 0) {
                    const bn = GM_getValue("clear_blackname");
                    if (!bn) {
                        blackName = blackName.map((e) => clear_zero_width(e));
                        GM_setValue("blackname", blackName);
                        GM_setValue("clear_blackname", true);
                    }
                }
            } else blackName = [];
            blackTopicAndQuestion = GM_getValue("blacktopicAndquestion");
            (!blackTopicAndQuestion || !Array.isArray(blackTopicAndQuestion)) &&
                (blackTopicAndQuestion = []);
        },
        clipboardClear: {
            clear(text) {
                const cs = [
                    /。/g,
                    /:/g,
                    /;/g,
                    /?/g,
                    /!/g,
                    /(/g,
                    /)/g,
                    /“/g,
                    /”/g,
                    /、/g,
                    /,/g,
                    /《/g,
                    /》/g,
                ];
                const es = [
                    ". ",
                    ": ",
                    "; ",
                    "? ",
                    "! ",
                    "(",
                    ")",
                    '"',
                    '"',
                    ", ",
                    ", ",
                    "<",
                    ">",
                ];
                cs.forEach((s, i) => (text = text.replace(s, es[i])));
                this.write(text);
            },
            write(text) {
                window.navigator.clipboard.writeText(text);
            },
            replace_ZH: true,
            event() {
                this.replace_ZH =
                    GM_getValue("clipboard") === false ? false : true;
                document.oncopy = (e) => {
                    e.preventDefault();
                    e.stopImmediatePropagation();
                    const copytext = getSelection();
                    copytext &&
                        (this.replace_ZH
                            ? this.clear(copytext)
                            : this.write(copytext));
                };
            },
        },
        turnPage: {
            main(mode) {
                const overlap = 100;
                const wh = window.innerHeight;
                let height = wh - overlap;
                height < 0 && (height = 0);
                let top =
                    document.documentElement.scrollTop ||
                    document.body.scrollTop ||
                    window.pageYOffset;
                if (mode) top += height;
                else top < height ? (top = 0) : (top -= height);
                window.scrollTo(0, top);
            },
            start(mode) {
                //n => scroll down ; u => scroll top
                window.requestAnimationFrame(this.main.bind(this, mode));
            },
        },
        scroll: {
            toTop() {
                let hTop =
                    document.documentElement.scrollTop ||
                    document.body.scrollTop;
                if (hTop === 0) return;
                const rate = 8;
                let sid = 0;
                const scrollToTop = () => {
                    hTop =
                        document.documentElement.scrollTop ||
                        document.body.scrollTop;
                    if (hTop > 0) {
                        sid = window.requestAnimationFrame(scrollToTop);
                        window.scrollTo(0, hTop - hTop / rate);
                    } else {
                        sid !== 0 && window.cancelAnimationFrame(sid);
                    }
                };
                scrollToTop();
            },
            toBottom() {
                //take care this, if the webpage adopts waterfall flow design
                const height =
                    document.documentElement.scrollHeight ||
                    document.body.scrollHeight;
                const sTop =
                    document.documentElement.scrollTop ||
                    document.body.scrollTop;
                if (sTop >= height) return;
                let sid = 0;
                let shTop = 0;
                let rate = 6;
                const initial = 100;
                const scrollToBottom = () => {
                    const hTop =
                        document.documentElement.scrollTop ||
                        document.body.scrollTop ||
                        initial;
                    if (hTop < height && hTop > shTop) {
                        shTop = hTop;
                        sid = window.requestAnimationFrame(scrollToBottom);
                        window.scrollTo(0, hTop + hTop / rate);
                        rate += 0.2;
                    } else {
                        sid !== 0 && window.cancelAnimationFrame(sid);
                        sid = 0;
                        rate = 6;
                    }
                };
                scrollToBottom();
            },
        },
        multiSearch: {
            main(keyCode, keyword) {
                const Names = {
                    65: "AboutMe",
                    68: "Douban",
                    71: "Google",
                    73: "Install",
                    72: "Github",
                    77: "MDN",
                    66: "BiliBili",
                    90: "Zhihu",
                    80: "Python",
                    69: "Ecosia",
                    1002: "Douban_movie",
                    1001: "Douban_book",
                };
                const methods = {
                    Protocols: "https://",
                    string_length(str) {
                        const lg = [...str].reduce(
                            (length, e) =>
                                (length +=
                                    e.charCodeAt(0).toString(16).length === 4
                                        ? 2
                                        : 1),
                            0
                        );
                        return Math.floor(lg / 2);
                    },
                    Search(url, parameter = "") {
                        //baidu restrict the length of search keyword is 38;
                        const select = keyword || getSelection();
                        if (!select) return;
                        else {
                            const reg = /[\u4e00-\u9fa5]/;
                            const f = reg.test(select);
                            if (
                                f
                                    ? this.string_length(select) > 38
                                    : select.length > 76
                            ) {
                                Notification(
                                    "the length of keyword is too long",
                                    "Tips"
                                );
                                return;
                            }
                            if (f && blackKey.includes(select)) {
                                Notification(
                                    "your keyword contains rubbish word; don't search rubbish",
                                    "Warning",
                                    3500
                                );
                                colorful_Console.main(
                                    { title: "rubbish:", content: select },
                                    colorful_Console.colors.warning
                                );
                                return;
                            }
                        }
                        url += encodeURIComponent(select);
                        GM_openInTab(this.Protocols + url + parameter, {
                            insert: true,
                            active: true,
                        });
                    },
                    Douban_movie() {
                        this.Search(
                            "search.douban.com/movie/subject_search?search_text=",
                            "&cat=1002"
                        );
                    },
                    Douban_book() {
                        this.Search(
                            "search.douban.com/book/subject_search?search_text=",
                            "&cat=1001"
                        );
                    },
                    Ecosia() {
                        this.Search("www.ecosia.org/search?q=");
                    },
                    Google() {
                        this.Search("www.bing.com/search?q=");
                    },
                    Douban() {
                        this.Search("www.douban.com/search?q=");
                    },
                    Zhihu() {
                        this.Search("www.zhihu.com/search?q=", "&type=content");
                    },
                    MDN() {
                        this.Search("developer.mozilla.org/zh-CN/search?q=");
                    },
                    Github() {
                        this.Search("github.com/search?q=");
                    },
                    BiliBili() {
                        this.Search("search.bilibili.com/all?keyword=");
                    },
                    Install() {
                        this.Search("pypi.org/search/?q=");
                    },
                    Python() {
                        this.Search(
                            "docs.python.org/zh-cn/3/search.html?q=",
                            "&check_keywords=yes&area=default"
                        );
                    },
                    AboutMe() {
                        zhihu.shade.Support.main();
                    },
                };
                const name = Names[keyCode];
                name && methods[name]();
            },
            checkCode(c) {
                const code = c.charCodeAt(0);
                return code > 97 && code < 123
                    ? code - 32
                    : code > 65 && code < 91
                    ? code
                    : 0;
            },
            get Searchbar() {
                const header =
                    document.getElementsByClassName("Sticky AppHeader");
                if (header.length === 0) return "";
                const input = header[0].getElementsByTagName("input");
                return input.length === 0 ? "" : input[0].defaultValue.trim();
            },
            m(keyword) {
                const reg = /(?<=-)([a-z]|d[bm])\s/gi;
                const m = (
                    keyword.slice(-2, -1) === "-" ? `${keyword} ` : keyword
                ).match(reg);
                return m ? (m.length === 0 ? m[0] : [...new Set(m)]) : null;
            },
            last_time_s: "",
            site() {
                let keyword = prompt(
                    "support z, d, g, h, m, b, p, e,db, dm: like: z python; (search in zhihu); default: g",
                    getSelection() || this.Searchbar || this.last_time_s
                );
                if (!keyword || !(keyword = keyword.trim())) return true;
                this.last_time_s = keyword;
                if (keyword[0] === "$") {
                    keyword = keyword.slice(1);
                    const ms = this.m(keyword);
                    if (ms) {
                        const wreg = /(?<=-([a-z]|d[bm])\s)(?!-).+(?=[\s-\b])/i;
                        const tmp = (
                            keyword.slice(-2, -1) === "-"
                                ? keyword
                                : `${keyword} `
                        ).match(wreg);
                        if (tmp) {
                            ms.forEach((e, index) => {
                                setTimeout(() => {
                                    let c = this.checkCode(e);
                                    c &&
                                        (c =
                                            c === 68
                                                ? e[1]
                                                    ? e[1].toLowerCase() === "b"
                                                        ? 1001
                                                        : 1002
                                                    : c
                                                : c) &&
                                        this.main(c, tmp[0]);
                                }, index * 350);
                            });
                        } else Notification("failed to get keyword", "Warning");
                        return true;
                    }
                }
                const tmp = keyword.slice(0, 3).toLowerCase();
                const i = ["db ", "dm "].indexOf(tmp);
                if (i > -1) {
                    this.main(i === 0 ? 1001 : 1002, keyword.slice(3).trim());
                    return;
                }
                const reg = /[a-z]\s/i;
                const code = this.checkCode(keyword[0]);
                code && keyword.match(reg)
                    ? this.main(code, keyword.slice(2).trim())
                    : this.main(71, keyword);
                return true;
            },
        },
        noteHighlight: {
            editable: false,
            disableSiderbar(pevent) {
                const column = document.getElementById("column_lists");
                if (column) column.style.pointerEvents = pevent;
            },
            EditDoc(status) {
                const [edit, tips, pevent] = this.editable
                    ? ["inherit", "exit", "inherit"]
                    : ["true", "enter", "none"];
                document.body.contentEditable = edit;
                Notification(tips + " page editable mode", "Editable");
                this.disableSiderbar(pevent);
                this.editable = !this.editable;
                this.editable
                    ? status.create("Editable Mode")
                    : status.remove();
            },
            get Selection() {
                return window.getSelection();
            },
            setMark(text, type) {
                return `<mark class="AssistantMark ${type}">${text}</mark>`;
            },
            get createElement() {
                return document.createElement("markspan");
            },
            appendNewNode(node, type) {
                const text = node.nodeValue;
                const span = this.createElement;
                node.parentNode.replaceChild(span, node);
                span.outerHTML = this.setMark(text, type);
            },
            getTextNode(node, type) {
                node.nodeType === 3 && this.appendNewNode(node, type);
            },
            Marker(keyCode) {
                const cname = {
                    82: "red",
                    89: "yellow",
                    80: "purple",
                    71: "green",
                };
                const type = cname[keyCode];
                if (!type) return;
                const select = this.Selection;
                if (!select.anchorNode || select.isCollapsed) return;
                let i = select.rangeCount;
                const r = select.getRangeAt(--i);
                let start = r.startContainer;
                const end = r.endContainer;
                const offs = r.startOffset;
                const offe = r.endOffset;
                let nodeValue = r.startContainer.nodeValue;
                if (start !== end) {
                    //start part
                    let next = start.nextSibling;
                    let p = start.parentNode;
                    if (!p.className.startsWith("AssistantMark")) {
                        const text = nodeValue.slice(offs);
                        const span = this.createElement;
                        p.replaceChild(span, start);
                        span.outerHTML =
                            nodeValue.slice(0, offs) + this.setMark(text, type);
                    }
                    //mid part
                    while (true) {
                        if (next) {
                            start = next;
                        } else {
                            next = p.nextSibling;
                            while (!next) {
                                p = p.parentNode;
                                next = p.nextSibling;
                            }
                            start = next;
                        }
                        //get the deepest level node
                        while (start.childNodes.length > 0)
                            start = start.childNodes[0];
                        if (start === end) break;
                        p = start.parentNode;
                        next = p.nextSibling;
                        !p.className.startsWith("AssistantMark") &&
                            this.getTextNode(start, type);
                    }
                    //end part
                    nodeValue = start.nodeValue;
                    start = start.parentNode;
                    if (start.className.startsWith("AssistantMark")) return;
                    const text = nodeValue.slice(0, offe);
                    const epan = this.createElement;
                    start.replaceChild(epan, end);
                    epan.outerHTML =
                        this.setMark(text, type) + nodeValue.slice(offe);
                } else {
                    //all value in one node;
                    const text = nodeValue.slice(offs, offe);
                    const span = this.createElement;
                    start.parentNode.replaceChild(span, start);
                    span.outerHTML =
                        nodeValue.slice(0, offs) +
                        this.setMark(text, type) +
                        nodeValue.slice(offe);
                }
            },
            Restore(node) {
                const p = node.parentNode;
                if (p.className.startsWith("AssistantMark")) {
                    p.parentNode.innerHTML = p.parentNode.innerText;
                    return true;
                }
                return false;
            },
            removeMark() {
                const select = this.Selection;
                if (!select.anchorNode || select.isCollapsed) return;
                let i = select.rangeCount;
                const r = select.getRangeAt(--i);
                let start = r.startContainer;
                const end = r.endContainer;
                if (start !== end) {
                    let t = start.nodeType;
                    if (t !== 3 && r.collapsed) {
                        const nodes =
                            start.getElementsByClassName("AssistantMark");
                        let i = nodes.length;
                        if (i > 0) {
                            for (i; i--; ) {
                                const p = nodes[i].parentNode;
                                p.innerhHTML = p.innerText;
                            }
                        }
                        return;
                    }
                    while (start.childNodes.length > 0)
                        start = start.childNodes[0];
                    let p = start.parentNode.parentNode;
                    let next = start.nextSibling;
                    let result = this.Restore(start);
                    //if this is mark node, will be removed, so we need get the parentnode to backup, if it is not mark node, restore the parentnode
                    !result && (p = start.parentNode);
                    while (true) {
                        if (next) {
                            start = next;
                        } else {
                            next = p.nextSibling;
                            while (!next) {
                                p = p.parentNode;
                                next = p.nextSibling;
                            }
                            start = next;
                        }
                        while (start.childNodes.length > 0)
                            start = start.childNodes[0];
                        if (start === end) break;
                        p = start.parentNode.parentNode;
                        next = start.nextSibling;
                        result = this.Restore(start);
                        !result && (p = start.parentNode);
                    }
                }
                this.Restore(start);
            },
        },
        autoScroll: {
            stepTime: 40,
            keyCount: 1,
            scrollState: false,
            scrollTime: null,
            scrollPos: null,
            bottom: 100,
            zhuanlanAuto_mode: false,
            pageScroll(TimeStamp) {
                const position =
                    document.documentElement.scrollTop ||
                    document.body.scrollTop ||
                    window.pageYOffset;
                if (this.scrollTime) {
                    this.scrollPos =
                        this.scrollPos !== null
                            ? this.scrollPos +
                              (TimeStamp - this.scrollTime) / this.stepTime
                            : position;
                    window.scrollTo(0, this.scrollPos);
                }
                this.scrollTime = TimeStamp;
                if (this.scrollState) {
                    let h =
                        document.documentElement.scrollHeight ||
                        document.body.scrollHeight;
                    h = h - window.innerHeight - this.bottom;
                    position < h
                        ? window.requestAnimationFrame(
                              this.pageScroll.bind(this)
                          )
                        : this.stopScroll(true);
                }
            },
            disableEvent(mode) {
                const h = document.getElementsByClassName(
                    "RichText ztext Post-RichText"
                );
                if (h.length === 0) return;
                h[0].style.pointerEvents = mode ? "none" : "inherit";
            },
            show_status: {
                node: null,
                show(tips) {
                    const html = `
                        <div
                            id="load_status"
                            style="
                                top: 0%;
                                z-index: 1000;
                                position: fixed;
                                height: 24px;
                                width: 50%;
                                font-size: 14px;
                                font-weight: 500;
                                margin-left: 25%;
                                text-align: center;
                                background: #FFBB59;
                                opacity: 0.8;
                                box-shadow: 0 0 15px #FFBB59;
                            "
                        >
                        Tips: ${tips}...
                        </div>`;
                    const anode = document.createElement("div");
                    document.documentElement.appendChild(anode);
                    anode.outerHTML = html;
                    setTimeout(
                        () =>
                            (this.node =
                                document.documentElement.lastElementChild),
                        0
                    );
                },
                remove() {
                    if (this.node) {
                        this.node.remove();
                        this.node = null;
                    }
                },
                create(tips) {
                    this.show(tips);
                },
            },
            zhuanlanAuto() {
                if (!this.zhuanlanAuto_mode && zhihu.Column.targetIndex === 0) {
                    Notification("current article is not in left menu");
                    return;
                }
                this.zhuanlanAuto_mode = !this.zhuanlanAuto_mode;
                const text = `${
                    this.zhuanlanAuto_mode ? "enter" : "exit"
                } autoscroll mode`;
                Notification(text, "Tips");
                this.zhuanlanAuto_mode
                    ? this.show_status.create("Auto Mode")
                    : this.show_status.remove();
            },
            get article_lists() {
                const c = document.getElementById("column_lists");
                if (!c) return null;
                const a = c.getElementsByClassName("article_lists");
                return a.length === 0 ? null : a[0].children;
            },
            nextPage() {
                const ch = this.article_lists;
                if (!ch) return;
                const i = zhihu.Column.targetIndex;
                if (ch.length === 0 || i === ch.length) {
                    Notification(
                        "no more content, have reach the last page of current menu",
                        "tips"
                    );
                    return;
                }
                ch[i].children[2].click();
                setTimeout(() => {
                    this.keyCount = 2;
                    this.start();
                }, 2500);
            },
            //0-9 => click target article;
            key_Click(keyCode) {
                const ch = this.article_lists;
                let index = keyCode - 47;
                if (index > ch.length) return;
                ch[--index].children[2].click();
            },
            key_next_Pre(keyCode) {
                const c = document.getElementById("column_lists");
                if (!c) return;
                const n = c.getElementsByClassName("nav button");
                let i = keyCode - 188;
                n.length > 0 &&
                    (i === 0
                        ? n[0].children[i].click()
                        : n[0].children[--i].click());
            },
            popup() {
                createPopup();
                const tips = document.getElementById("autoscroll-tips");
                let buttons = tips.getElementsByTagName("button");
                const id = setTimeout(() => {
                    tips.remove();
                    zhihu.scroll.toTop();
                    this.nextPage();
                }, 3500);
                buttons[0].onclick = () => {
                    clearTimeout(id);
                    tips.remove();
                    zhihu.scroll.toTop();
                    this.nextPage();
                };
                buttons[1].onclick = () => {
                    clearTimeout(id);
                    tips.remove();
                };
                buttons = null;
            },
            stopScroll(mode = false) {
                if (this.scrollState) {
                    this.scrollPos = null;
                    this.scrollTime = null;
                    this.scrollState = false;
                    this.keyCount = 1;
                }
                if (mode) {
                    this.disableEvent(false);
                    this.zhuanlanAuto_mode &&
                        setTimeout(() => this.popup(), 1500);
                }
            },
            speedUP() {
                this.stepTime < 10 ? (this.stepTime = 5) : (this.stepTime -= 5);
            },
            slowDown() {
                this.stepTime > 100
                    ? (this.stepTime = 100)
                    : (this.stepTime += 5);
            },
            start() {
                this.keyCount += 1;
                if (this.keyCount % 2 === 0) return;
                this.scrollState
                    ? (this.stopScroll(), this.disableEvent(false))
                    : ((this.scrollState = true),
                      this.disableEvent(true),
                      window.requestAnimationFrame(this.pageScroll.bind(this)));
            },
            noColorful() {
                const color = GM_getValue("nocolofultext");
                if (color) {
                    GM_setValue("nocolofultext", false);
                    const text = "the feature of colorful text has been enable";
                    zhihu.colorAssistant.main();
                    Notification(text, "Tips");
                } else {
                    if (!confirm("disable colorful text?")) return;
                    GM_setValue("nocolofultext", true);
                    const text = "colorful text has been disable";
                    Notification(text, "Tips");
                    confirm("reload current webpage?") &&
                        (sessionStorage.clear(), location.reload());
                }
            },
            Others(keyCode, shift, auto) {
                shift
                    ? keyCode === 67
                        ? this.noteHighlight.removeMark()
                        : keyCode === 219
                        ? auto
                            ? Notification(
                                  "please exit auto mode firstly",
                                  "Tips"
                              )
                            : this.Column.pagePrint(this.autoScroll.show_status)
                        : keyCode === 70
                        ? !this.autoScroll.scrollState && this.Column.follow()
                        : keyCode === 76
                        ? this.Column.columnsModule.recentModule.log("p")
                        : keyCode === 83
                        ? this.Column.subscribe()
                        : keyCode === 68
                        ? this._top_Picture.main()
                        : this.noteHighlight.Marker(keyCode)
                    : keyCode === 113
                    ? !this.autoScroll.scrollState &&
                      this.noteHighlight.EditDoc(this.autoScroll.show_status)
                    : keyCode === 78
                    ? !this.autoScroll.scrollState && this.turnPage.start(true)
                    : keyCode === 84
                    ? !this.autoScroll.scrollState && this.scroll.toTop()
                    : keyCode === 82
                    ? !this.autoScroll.scrollState && this.scroll.toBottom()
                    : keyCode === 85
                    ? !this.autoScroll.scrollState && this.turnPage.start(false)
                    : this.multiSearch.main(keyCode);
            },
            check_common_key(shift, keyCode) {
                return this.common_KeyEevnt(shift, keyCode, 5);
            },
            key_conflict(keyCode, shift) {
                return (
                    68 === keyCode || (shift && [85, 71, 67].includes(keyCode))
                );
            },
            keyBoardEvent() {
                document.addEventListener(
                    "keydown",
                    (e) => {
                        const keyCode = e.keyCode;
                        if (e.ctrlKey || e.altKey) return;
                        const className = e.target.className;
                        if (
                            (className &&
                                typeof className === "string" &&
                                className.includes("DraftEditor")) ||
                            e.target.localName === "input"
                        )
                            return;
                        const shift = e.shiftKey;
                        if (this.key_conflict(keyCode, shift)) {
                            e.preventDefault();
                            e.stopPropagation();
                        }
                        if (this.check_common_key.call(zhihu, shift, keyCode))
                            return;
                        shift
                            ? keyCode === 65
                                ? zhihu.Column.modePrint
                                    ? Notification(
                                          "please exit print mode firstly",
                                          "Tips"
                                      )
                                    : this.zhuanlanAuto()
                                : keyCode === 66
                                ? !this.scrollState &&
                                  MangeData.exportData.main(false)
                                : keyCode === 84
                                ? this.noColorful()
                                : keyCode === 73
                                ? !this.scrollState &&
                                  MangeData.importData.main(true)
                                : this.Others.call(
                                      zhihu,
                                      keyCode,
                                      shift,
                                      this.zhuanlanAuto_mode
                                  )
                            : keyCode === 192
                            ? this.start()
                            : keyCode === 187
                            ? this.speedUP()
                            : keyCode === 189
                            ? this.slowDown()
                            : keyCode > 47 && keyCode < 58
                            ? !this.scrollState && this.key_Click(keyCode)
                            : keyCode === 188 || keyCode === 190
                            ? this.key_next_Pre(keyCode)
                            : this.Others.call(zhihu, keyCode);
                    },
                    true
                );
            },
        },
        shade: {
            Support: {
                interval: 0,
                support: null,
                tips: null,
                opacity: null,
                opacityChange(opacity) {
                    const target =
                        document.getElementById("screen_shade_cover");
                    target &&
                        (this.opacity === null
                            ? (this.opacity = target.style.opacity)
                            : target.style.opacity !== opacity) &&
                        (target.style.opacity = opacity);
                },
                share_weibo() {
                    const url = `https%3A%2F%2Fservice.weibo.com%2Fshare%2Fshare.php%3Furl%3D${
                        Assist_info_URL.greasyfork
                    }%26title%3D%E4%B9%9F%E8%AE%B8%E8%BF%99%E6%98%AF%E9%92%88%E5%AF%B9%E7%9F%A5%E4%B9%8E%E6%9C%80%E6%A3%92%E7%9A%84GM%E6%B2%B9%E7%8C%B4%E8%84%9A%E6%9C%AC...%26summery%3Dundefined%26pic%3D${
                        Assist_info_URL.Overview
                    }%23_loginLayer_${Date.now()}`;
                    GM_openInTab(decodeURIComponent(url), {
                        insert: true,
                        active: true,
                    });
                },
                creatPopup() {
                    this.opacityChange(0);
                    const mt = -5;
                    const html = `
                        <div
                            id="support_me"
                            style="
                                background: darkgray;
                                text-align: justify;
                                width: 700px;
                                font-size: 16px;
                                height: 468px;
                                position: fixed;
                                top:50%;
                                left:50%;
                                transform: translate(-50%,-50%);
                                z-index: 100000;
                            "
                        >
                            <div style="padding: 2.5%; font-weight: bold; font-size: 18px">
                                Support Me!
                                <span
                                    class="shorts_cut"
                                    style="font-size: 12px; font-weight: normal; float: right"
                                >
                                    <i
                                        title="Share with your friends. Share to Weibo"
                                        style="
                                            margin-right: 10px;
                                            content: url();
                                        "
                                    ></i>
                                    <a
                                        href=${Assist_info_URL.shortcuts}
                                        target="_blank"
                                        title="shortcuts diagram"
                                        style="color: #2b638b"
                                        >Shortcuts</a
                                    >
                                    <span> || Version: ${GM_info.script.version}</span>
                                </span>
                            </div>
                            <div style="font-style: italic; font-size: 16px; padding-left: 2%">
                                Make Thing Better &amp;&amp; Simpler!
                                <img
                                    src=""
                                    style="
                                        float: left;
                                        height: 42px;
                                        width: 42px;
                                        margin: -10px 4px 0 0px;
                                    "
                                />
                            </div>
                            <div
                                class="support_img"
                                style="padding-top: 4%; width: 100%; padding-left: 7.5%"
                            >
                                <div class="qrCode">
                                    <img
                                        src=""
                                    />
                                </div>
                            </div>
                            <div class="timeout" style="font-size: 12px; padding: 3%">
                                15s, this Tips will be automatically closed or can you just click
                            </div>
                            <a
                                href=${Assist_info_URL.usermanual}
                                target="blank"
                                style="margin: ${mt}px 10px 0px 0px; float: right; font-size: 14px"
                                title="user manual"
                            >
                                Github: Kyouichirou
                            </a>
                        </div>`;
                    document.documentElement.insertAdjacentHTML(
                        "beforeend",
                        html
                    );
                    this.support = document.getElementById("support_me");
                    this.tips =
                        this.support.getElementsByClassName("timeout")[0];
                    let time = 15;
                    this.interval = setInterval(() => {
                        time--;
                        this.tips.innerText = `${time}s, this Tips will be automatically closed or you can just click`;
                        time === 0 && this.remove();
                    }, 1000);
                    this.support.onclick = (e) =>
                        e.target.localName === "i"
                            ? this.share_weibo()
                            : e.target.localName !== "a" &&
                              setTimeout(() => this.remove(), 120);
                },
                remove() {
                    clearInterval(this.interval);
                    this.opacityChange(this.opacity);
                    this.opacity = null;
                    this.interval = null;
                    this.support.remove();
                    this.support = null;
                    this.tips = null;
                },
                main() {
                    this.support ? this.remove() : this.creatPopup();
                },
            },
            cover(color, opacity = 0.5) {
                const html = `
                    <div
                        id="screen_shade_cover"
                        style="
                            transition: opacity 0.1s ease 0s;
                            z-index: 10000000;
                            margin: 0;
                            border-radius: 0;
                            padding: 0;
                            background: ${color};
                            pointer-events: none;
                            position: fixed;
                            top: -10%;
                            right: -10%;
                            width: 120%;
                            height: 120%;
                            opacity: ${opacity};
                            mix-blend-mode: multiply;
                            display: block;
                        "
                    ></div>`;
                const cnode = document.documentElement;
                cnode && cnode.insertAdjacentHTML("afterbegin", html);
            },
            menu(e) {
                const target = document.getElementById("screen_shade_cover");
                target &&
                    target.style.background !== this[e] &&
                    (target.style.background = this[e]) &&
                    arguments.length === 2 &&
                    GM_setValue("color", e);
                GM_deleteValue("tmp_cover");
            },
            get opacity() {
                const date = new Date();
                const m = date.getMonth();
                const h = date.getHours();
                const [start, a] = m > 9 ? [15, 0.08] : [16, 0.12];
                let opacity =
                    h > 20
                        ? h > 22
                            ? 0.6
                            : 0.5
                        : h < 8
                        ? 0.65
                        : h > start
                        ? h === 18
                            ? 0.35
                            : h === 19
                            ? 0.45
                            : h === 20
                            ? 0.5
                            : 0.3
                        : 0.15;
                return (opacity += opacity < 0.2 ? 0 : a);
            },
            opacityMonitor() {
                const opacity = GM_getValue("opacity");
                const target = document.getElementById("screen_shade_cover");
                target &&
                    opacity &&
                    target.style.opacity !== opacity &&
                    (target.style.opacity = opacity);
            },
            supportID: null,
            SupportMenu() {
                this.supportID = GM_registerMenuCommand(
                    "Support || Donation",
                    this.Support.main.bind(this.Support),
                    "d4"
                );
            },
            disableShade: {
                id: null,
                cmenu() {
                    this.id = GM_registerMenuCommand(
                        "Switch",
                        this.func.bind(this),
                        "s5"
                    );
                },
                rmenu() {
                    GM_unregisterMenuCommand(this.id);
                },
                func() {
                    const target =
                        document.getElementById("screen_shade_cover");
                    target &&
                        (target.style.display =
                            target.style.display === "block"
                                ? "none"
                                : "block");
                },
            },
            menuID: null,
            Switchfunc() {
                const target = document.getElementById("screen_shade_cover");
                let result = false;
                if (target) {
                    if (arguments.length > 0 && !arguments[0]) return;
                    target.remove();
                    result = true;
                    this.disableShade.rmenu();
                    let i = this.menuID.length;
                    GM_removeValueChangeListener(this.opacitylistenID);
                    GM_removeValueChangeListener(this.colorlistenID);
                    for (i; i--; ) GM_unregisterMenuCommand(this.menuID[i]);
                    this.menuID = null;
                } else {
                    //rebuild menu
                    if (arguments.length > 0 && arguments[0]) return;
                    if (this.menuID) return;
                    GM_unregisterMenuCommand(this.switchID);
                    GM_unregisterMenuCommand(this.supportID);
                    this.createShade();
                    this.SwitchMenu();
                    this.SupportMenu();
                }
                arguments.length === 0 && GM_setValue("turnoff", result);
            },
            switchID: null,
            SwitchMenu() {
                this.switchID = GM_registerMenuCommand(
                    "Turn(On/Off)",
                    this.Switchfunc.bind(this),
                    "t6"
                );
            },
            turnoffID: null,
            start() {
                !GM_getValue("turnoff") && this.createShade();
                this.SwitchMenu();
                this.SupportMenu();
                this.turnoffID = GM_addValueChangeListener(
                    "turnoff",
                    (name, oldValue, newValue, remote) => {
                        if (!remote || oldValue === newValue) return;
                        this.Switchfunc(newValue, true);
                    }
                );
            },
            colorlistenID: null,
            opacitylistenID: null,
            get tmp_cover() {
                const tc = GM_getValue("tmp_cover");
                if (!tc) return null;
                if (tc.status !== "a") return null;
                const u = tc.update;
                if (!u) return null;
                const r = tc.rtime;
                if (!r) return null;
                if (Date.now() - u > r) return null;
                return tc;
            },
            sing_protect: false,
            tmp_c(color) {
                const c = document.getElementById("screen_shade_cover");
                c.style.background = color;
            },
            createShade() {
                const colors = {
                    yellow: "rgb(247, 232, 176)",
                    green: "rgb(202 ,232, 207)",
                    grey: "rgb(182, 182, 182)",
                    olive: "rgb(207, 230, 161)",
                };
                const tc = this.tmp_cover;
                let color = GM_getValue("color");
                color && (color = colors[color]);
                tc && tc.color && (color = tc.color);
                if (!color) {
                    const h = new Date().getHours();
                    color = h > 8 && h < 17 ? colors.yellow : colors.grey;
                }
                const opacity =
                    (tc && tc.opacity && tc.opacity) || this.opacity;
                this.cover(color, opacity);
                const UpperCase = (e) =>
                    e.slice(0, 1).toUpperCase() + e.slice(1);
                this.menuID = [];
                for (const c of Object.entries(colors)) {
                    const id = GM_registerMenuCommand(
                        UpperCase(c[0]),
                        this.menu.bind(colors, c[0], true),
                        c[0]
                    );
                    this.menuID.push(id);
                }
                //note, who is the "this" in the GM_registerMenuCommand? take care of "this", must bind (function => this)
                this.colorlistenID = GM_addValueChangeListener(
                    "color",
                    (name, oldValue, newValue, remote) => {
                        if (
                            !remote ||
                            oldValue === newValue ||
                            this.sing_protect
                        )
                            return;
                        newValue.startsWith("#")
                            ? this.tmp_c(newValue)
                            : this.menu.call(colors, newValue);
                    }
                );
                GM_setValue("opacity", opacity);
                this.opacitylistenID = GM_addValueChangeListener(
                    "opacity",
                    (name, oldValue, newValue, remote) =>
                        remote &&
                        !this.sing_protect &&
                        oldValue !== newValue &&
                        this.opacityMonitor()
                );
                this.disableShade.cmenu();
                !tc && GM_deleteValue("tmp_cover");
            },
        },
        settings_Popup: {
            /*
            1. adopted from https://greasyfork.org/zh-CN/scripts/27752-searchenginejump
            2. part of html and css is adpoted;
            */
            node: null,
            create() {
                const html = `
                <div
                    id="settingLayerMask"
                    style="
                        display: flex;
                        justify-content: center;
                        align-items: stretch;
                        opacity: 1;
                    "
                >
                    <style>
                        #settingLayerMask {
                            display: none;
                            justify-content: center;
                            align-items: center;
                            position: fixed;
                            top: 0;
                            right: 0;
                            bottom: 0;
                            left: 0;
                            background-color: rgba(0, 0, 0, 0.5);
                            z-index: 200000000;
                            overflow: auto;
                            font-family: arial, sans-serif;
                            min-height: 100%;
                            font-size: 16px;
                            transition: 0.5s;
                            opacity: 0;
                            user-select: none;
                            -moz-user-select: none;
                            padding-bottom: 80px;
                            box-sizing: border-box;
                        }
                        #settingLayer {
                            height: 360px;
                            display: flex;
                            flex-wrap: wrap;
                            padding: 20px;
                            margin: 0px 25px 50px 5px;
                            background-color: #fff;
                            border-radius: 4px;
                            position: absolute;
                            min-width: 700px;
                            transition: 0.5s;
                        }
                        span.drag {
                            display: block;
                            position: relative;
                        }
                        span.sej-engine {
                            background-color: #ccc;
                            border-radius: 2px;
                            width: 100%;
                            box-sizing: border-box;
                            cursor: pointer;
                            line-height: 2;
                            display: inline-block;
                            margin: 0 0px 0 0;
                            border: none;
                            padding: 0 6px;
                            font-weight: 500;
                            color: #333 !important;
                            transition: background-color 0.15s ease-in-out;
                        }
                        #btnEle {
                            position: absolute;
                            width: 100%;
                            bottom: 4px;
                            right: 0;
                            background: #fff;
                            border-radius: 4px;
                        }
                        #btnEle > div {
                            width: 100%;
                            margin-bottom: -100%;
                            display: flex;
                            justify-content: space-around;
                            background: #eff4f8;
                            border-radius: 4px;
                        }
                        #btnEle .feedback {
                            border-color: #aaa;
                        }
                        #btnEle span {
                            display: inline-block;
                            background: #eff4f8;
                            border: 1px solid #3abdc1;
                            margin: 12px auto 10px;
                            color: #3abdc1;
                            padding: 5px 10px;
                            border-radius: 4px;
                            cursor: pointer;
                            outline: none;
                            transition: 0.3s;
                        }
                        #btnEle a {
                            color: #999;
                            text-decoration: none;
                        }
                        #xin-close {
                            background: white;
                            color: #3abdc1;
                            line-height: 20px;
                            text-align: center;
                            height: 20px;
                            width: 20px;
                            text-align: center;
                            font-size: 20px;
                            padding: 10px;
                            border: 3px solid #3abdc1;
                            border-radius: 50%;
                            transition: 0.5s;
                            top: -20px;
                            right: -20px;
                            position: absolute;
                        }
                        #xin-close::before {
                            content: "\\2716";
                        }
                    </style>
                    <div
                        id="settingLayer"
                        style="top: 45%; left: 50%; transform: translate(-50%, -50%)"
                    >
                        <div class="setting_content" style="margin-left: 5%"></div>
                        <div id="btnEle">
                            <div class="btnEleLayer">
                                <span class="feedback"
                                    ><a
                                        target="_blank"
                                        href=${Assist_info_URL.greasyfork}
                                        >Greasyfork</a
                                    ></span
                                ><span class="feedback"
                                    ><a
                                        target="_blank"
                                        title="user manual"
                                        href=${Assist_info_URL.usermanual}
                                        >GitHub</a
                                    ></span
                                ><span class="feedback"
                                    ><a
                                        target="_blank"
                                        href=${Assist_info_URL.shortcuts}
                                        >Shortcuts</a
                                    ></span
                                ><span class="feedback"
                                ><a
                                    target="_blank"
                                    href=${Assist_info_URL.feedback}
                                    >Feedback</a
                                ></span
                                >
                                <span id="xin-save" title="save &amp; close">Save&Close</span>
                            </div>
                        </div>
                        <span id="xin-close" title="close"></span>
                    </div>
                </div>`;
                document.body.insertAdjacentHTML("beforeend", html);
                this.event();
            },
            event() {
                setTimeout(() => {
                    this.node = document.getElementById("settingLayerMask");
                    this.node.onclick = (e) => {
                        const id = e.target.id;
                        id && id.includes("close") && this.remove();
                    };
                }, 50);
            },
            remove() {
                if (this.node) {
                    this.node.remove();
                    this.node = null;
                }
            },
            main() {
                this.node ? this.remove() : this.create();
            },
        },
        white_noise: {
            get Index() {
                const index = GM_getValue("white_noise");
                return typeof index === "number" ? index : 0;
            },
            get_musice(index) {
                const audio_sources = [
                    "rain_sound_1",
                    "rain_sound_2",
                    "white_noise_1",
                    "river_stream_1",
                    "campfire_1",
                    "winter_traffic_1",
                    "ocean_waves_1",
                    "blizzard_1",
                    "forest_wind_1",
                    "crickets_1",
                ];
                const pref = "https://noizzze.com/audio/";
                const suffix = ".mp3";
                const f = index > audio_sources.length - 1;
                f && GM_setValue("white_noise", 0);
                return pref + audio_sources[f ? 0 : index] + suffix;
            },
            audio_ctrl: {
                audio: null,
                volume: 0,
                play_pause() {
                    this.audio && this.audio.paused
                        ? this.audio.play()
                        : this.audio.pause();
                    return true;
                },
                voice_up_down(e) {
                    if (!this.audio) return true;
                    let vx = this.volume;
                    if (e) {
                        if (vx >= 1) return true;
                        vx += 0.1;
                    } else {
                        if (vx <= 0) return true;
                        vx -= 0.1;
                    }
                    this.audio.volume = vx;
                    this.volume = vx;
                    return true;
                },
            },
            set_audio(audio, mode = false) {
                audio.loop = true;
                this.audio_ctrl.volume = audio.volume;
                audio.oncanplay = () => (mode ? (mode = false) : audio.play());
                audio.onerror = (e) => {
                    console.log(e);
                    Notification(
                        "failed to load mp3 of white noise",
                        "Warning"
                    );
                };
            },
            just_audio() {
                const audio = document.createElement("audio");
                audio.src = this.get_musice(this.Index);
                this.set_audio(audio);
                Notification("create audio successfully", "Tips");
                this.audio_ctrl.audio = audio;
            },
            destroy_audio() {
                if (this.audio_ctrl.audio) {
                    this.audio_ctrl.audio.pause();
                    this.audio_ctrl.audio = null;
                    Notification("the audio has been destroyed", "Tips");
                } else this.just_audio();
                return true;
            },
            create(node) {
                const html = `
                    <div class="white_noise" style="position: absolute; margin-left: -35%; opacity:0.6;">
                    <button
                        style="
                            background-image: url(https://img.meituan.net/csc/e360823b460fab14d02bbb9b56fb55825756.png);
                            display: inline-flex;
                            font-size: 12px;
                            text-align: center;
                            cursor: pointer;
                            border: 0.5px solid lightgray;
                            border-radius: 3px;
                            height: 25px;
                            width: 52px;
                            margin-right: 10px;
                            background-repeat: no-repeat;
                            background-position: center;
                            background-size: contain;
                        "
                        title="change style of white noise"
                    >
                    </button>
                    <audio
                        class="rain_sound_1 allaudio"
                        controls="controls"
                        style="position: absolute; height: 28px; width: 216px"
                    >
                        <source
                            src=${this.get_musice(this.Index)}
                            type="audio/mpeg"
                        />
                    </audio>
                </div>`;
                node.insertAdjacentHTML("afterbegin", html);
                this.event(node);
            },
            change_music(mode) {
                if (!this.audio_ctrl.audio) return true;
                const index = this.Index + 1;
                this.audio_ctrl.audio.src = this.get_musice(index);
                GM_setValue("white_noise", index);
                mode &&
                    Notification("change style of white noise successfully");
                return true;
            },
            event(node) {
                setTimeout(() => {
                    let f = node.firstElementChild.firstElementChild;
                    f.onclick = () => this.change_music();
                    const audio = f.nextElementSibling;
                    this.set_audio(audio, true);
                    this.audio_ctrl.audio = audio;
                    f = null;
                }, 50);
            },
        },
        create_settings(node) {
            const html = `
            <i
                class="my_settings"
                title="settings"
                style="
                    content: url();
                    height: 28px;
                    width: 28px;
                    margin-left: 120%;
                    position: absolute;
                    border: 1px solid #06f;
                "
            ></i>`;
            let f = node.previousElementSibling.getElementsByClassName(
                "ColumnPageHeader-Button"
            )[0];
            let n = f.firstElementChild;
            n.innerText = "Note";
            n.title = "right mouse click to open note";
            n.oncontextmenu = (e) => {
                e.preventDefault();
                this.settings_Popup.main();
            };
            n = null;
            f.insertAdjacentHTML("afterend", html);
            setTimeout(() => {
                f.nextElementSibling.onclick = (e) =>
                    this.settings_Popup.main();
                this.white_noise.create(f.parentNode);
                f = null;
            }, 50);
        },
        column_homePage(mode) {
            // 这个css被从页面移除掉了
            const css = `
            <link
                rel="stylesheet"
                type="text/css"
                href="https://static.zhihu.com/heifetz/main.shared_a7b367b7bf06fb0fc5585fec26d0675bd42be57e_CSS.216a26f4.9e02bee3c14871521874.css"
                crossorigin="anonymous"
            />`;
            document.head.insertAdjacentHTML("beforeend", css);
            const ch = document.getElementsByClassName("ColumnHome")[0];
            let chs = ch.children;
            let i = chs.length;
            if (i === 0) {
                colorful_Console.main(
                    {
                        title: "warning",
                        content: "failed to get target element",
                    },
                    colorful_Console.colors.warning
                );
                return;
            }
            for (i; i--; ) {
                if (chs[i].className === "ColumnHomeTop") break;
                else chs[i].remove();
            }
            let k = chs[i].children.length;
            for (k; k--; ) chs[i].children[k].remove();
            ch.style.display = "block";
            if (mode) {
                const r = GM_getValue("recent");
                if (!this.Column.home_Module.loaded_list)
                    this.Column.home_Module.loaded_list = [];
                if (r && Array.isArray(r) && r.length > 0) {
                    r.forEach((e) => {
                        const url = e.url;
                        const id = url.slice(url.lastIndexOf("/") + 1);
                        if (
                            !zhihu.Column.home_Module.loaded_list.includes(id)
                        ) {
                            this.Column.home_Module.loaded_list.push(id);
                            column_Home.single_Content_request(
                                url.includes("answer") ? 0 : 2,
                                id,
                                chs[i],
                                e
                            );
                        }
                    });
                }
            }
            const targetElements = this.Filter.getTagetElements(1);
            this.qaReader.no_scroll = true;
            chs[i].onclick = (e) => {
                const target = e.target;
                const item = this.Filter.check_click_all(
                    target.className,
                    target.localName,
                    target,
                    targetElements
                );
                if (!item) return;
                this.qaReader.main(item, this.Filter.foldAnswer.getid(item));
            };
            this.create_settings(chs[i]);
            chs = null;
            this.QASkeyBoardEvent(9);
            unsafeWindow.addEventListener("visibilitychange", () =>
                this.visibleChange()
            );
            this.rightMouse_OpenQ(9);
        },
        antiRedirect() {
            // only those links can be capture, which has the attribute of classname with ' external' ?
            // some problems in video tag
            const links = Object.getOwnPropertyDescriptors(
                HTMLAnchorElement.prototype
            ).href;
            Object.defineProperty(HTMLAnchorElement.prototype, "href", {
                ...links,
                get() {
                    const href = decodeURIComponent(links.get.call(this));
                    const tmp = href.split("link.zhihu.com/?target=");
                    if (tmp.length > 1) {
                        this.href = tmp[1];
                        return tmp[1];
                    }
                    return href;
                },
            });
        },
        //if has logined or the login window is not loaded(or be blocked) when the page is loaded;
        hasLogin: false,
        is_delayed: false,
        antiLogin() {
            /*
            note:
            the timing of the js injection is uncertain, and for some reason the injection maybe late,
            so that the occurrence of the event cannot be accurately captured
            don't use dom load event =>
            most of zhihu webpages require login
            */
            let mo = new MutationObserver((events) => {
                if (this.hasLogin) {
                    mo.disconnect();
                    mo = null;
                    document.documentElement.style.overflow = "auto";
                    return;
                }
                events.forEach((e) =>
                    e.addedNodes.forEach((node) => {
                        const type = node.nodeType;
                        if (
                            (type === 1 || type === 9 || type === 11) &&
                            node.getElementsByClassName("signFlowModal")
                                .length > 0
                        ) {
                            node.style.display = "none";
                            setTimeout(() => {
                                this.is_delayed = false;
                                mo.disconnect();
                                mo = null;
                                node.remove();
                                document.documentElement.style.overflow =
                                    "auto";
                            }, 0);
                        }
                    })
                );
            });
            document.body
                ? mo.observe(document.body, { childList: true })
                : (document.onreadystatechange = () =>
                      mo && mo.observe(document.body, { childList: true }));
            setTimeout(() => (this.is_delayed = true), 10000);
        },
        //the original js(int.js) of zhihu, which will cause stuck autoscroll
        anti_setInterval() {
            unsafeWindow.setInterval = new Proxy(unsafeWindow.setInterval, {
                apply: (target, thisArg, args) => {
                    const f = args[0];
                    let fn = "";
                    f && (fn = f.name);
                    fn &&
                        fn === "i" &&
                        args[1] === 2000 &&
                        (args[1] = 10000000);
                    return target.apply(thisArg, args);
                },
            });
        },
        Filter: {
            /*
            1. userName
            2. question
            3. answer
            4. article
            5. content keyword
            */
            //click the ico of button
            svgCheck(node, targetElements) {
                let pnode = node.parentNode;
                if (pnode.className === targetElements.buttonClass)
                    return pnode;
                else {
                    pnode = pnode.parentNode;
                    let className = pnode.className;
                    let ic = 0;
                    while (className !== targetElements.buttonClass) {
                        pnode = pnode.parentNode;
                        if (!node || ic > 3) return null;
                        className = pnode.className;
                        ic++;
                    }
                    return pnode;
                }
            },
            getiTem(target, targetElements) {
                let item = target.parentNode;
                if (item.className === targetElements.itemClass) {
                    return item;
                } else {
                    item = item.parentNode;
                    let ic = 0;
                    let className = item.className;
                    while (className !== targetElements.itemClass) {
                        item = item.parentNode;
                        if (!item || ic > 3) return null;
                        className = item.className;
                        ic++;
                    }
                    return item;
                }
            },
            //get the url id of the answer || article
            getTargetID(item) {
                const a = item.getElementsByTagName("a");
                if (a.length === 0) return null;
                const pathname = a[0].pathname;
                return pathname.slice(pathname.lastIndexOf("/") + 1);
            },
            /*
            0, normal
            1, searchpage => check username
            2, click => check all content
            3, article page => check expand
            if the item has been checked, return
            */
            get Topic_question_ID() {
                const href = location.href;
                const reg = /(?<=(question|topic)\/)\d+/;
                const match = href.match(reg);
                if (!match) return null;
                return match[0];
            },
            Topic_questionButton(targetElements) {
                const header_line = (event, mode) => {
                    let question = null;
                    if (event) {
                        const path = event.path;
                        for (const e of path) {
                            if (e.className === targetElements.headerID) {
                                question = e;
                                break;
                            }
                        }
                    } else {
                        question = document.getElementsByClassName(
                            targetElements.headerID
                        )[0];
                    }
                    if (!question) return;
                    const h = question.getElementsByClassName(
                        targetElements.headerTitle
                    );
                    if (h.length === 0) return;
                    h[0].style.textDecoration = mode ? "line-through" : "none";
                };
                let q = document.getElementsByClassName(
                    targetElements.inserButtonID
                );
                if (q.length === 0) return;
                const id = this.Topic_question_ID;
                if (!id) return;
                let mode = false;
                const type = targetElements.index === 2 ? "topic" : "question";
                let [name, title] = (mode = blackTopicAndQuestion.some(
                    (e) => e.id === id
                ))
                    ? ["Remove", `remove the ${type} from block list`]
                    : ["Block", `add the ${type} to block list`];
                const html = `
                <button
                    title=${escapeBlank(title)}
                    style="
                        font-size: 14px;
                        height: 22px;
                        width: 60px;
                        margin-left: 10px;
                        box-shadow: 1px 1px 4px #888888;
                    "
                >
                    ${name}
                </button>`;
                let button = document.createElement("button");
                q[0].insertAdjacentElement("beforeend", button);
                button.outerHTML = html;
                q[0].lastChild.onclick = function (event) {
                    if (mode) {
                        const index = blackTopicAndQuestion.findIndex(
                            (e) => e.id === id
                        );
                        if (index > -1) {
                            blackTopicAndQuestion.splice(index, 1);
                            GM_setValue(
                                targetElements.blockValueName,
                                blackTopicAndQuestion
                            );
                        }
                    } else {
                        if (blackTopicAndQuestion.some((e) => e.id === id))
                            return;
                        const info = {};
                        info.id = id;
                        info.type = type;
                        info.update = Date.now();
                        blackTopicAndQuestion.push(info);
                        GM_setValue(
                            targetElements.blockValueName,
                            blackTopicAndQuestion
                        );
                    }
                    mode = !mode;
                    [name, title] = mode
                        ? ["Remove", `remove the ${type} from block list`]
                        : ["Block", `add the ${type} to block list`];
                    this.title = title;
                    this.innerText = name;
                    header_line(event, mode);
                };
                button = null;
                q = null;
                mode && header_line(null, true);
                GM_addValueChangeListener(
                    targetElements.blockValueName,
                    (name, oldValue, newValue, remote) =>
                        remote && (blackTopicAndQuestion = newValue)
                );
            },
            get_content_element(item, targetElements) {
                const content = item.getElementsByClassName(
                    targetElements.contentID
                );
                return content.length === 0 ? null : content[0];
            },
            content_check(item, targetElements, node) {
                const content =
                    node || this.get_content_element(item, targetElements);
                if (!content) return false;
                const text = content.innerText;
                return text.length > 2500
                    ? false
                    : blackKey.some((e) => {
                          if (text.includes(e)) {
                              colorful_Console.main(
                                  {
                                      title: "content block:",
                                      content: "rubbish word " + e,
                                  },
                                  colorful_Console.colors.warning
                              );
                              this.hidden_item(item, targetElements);
                              return true;
                          }
                          return false;
                      });
            },
            hidden_item(item, targetElements) {
                (item.className === "ContentItem AnswerItem"
                    ? targetElements.index === 3
                        ? item.parentNode.parentNode
                        : item.parentNode
                    : item
                ).style.display = "none";
            },
            user_check(item, targetElements) {
                const user = item.getElementsByClassName(targetElements.userID);
                if (user.length === 0) {
                    colorful_Console.main(
                        { title: "anonymous user", content: "no user info" },
                        colorful_Console.colors.info
                    );
                    return false;
                }
                const i = user.length - 1;
                const name = user[i > 1 ? 1 : i].innerText;
                if (!name) return false;
                const nwname = clear_zero_width(name);
                const result = blackName.includes(nwname);
                if (result) {
                    colorful_Console.main(
                        { title: "Blocked User", content: nwname },
                        colorful_Console.colors.warning
                    );
                    this.hidden_item(item, targetElements);
                }
                return result;
            },
            search_check(item, targetElements) {
                const content = this.get_content_element(item, targetElements);
                if (!content) return false;
                let user = content.firstElementChild.innerText;
                if (!user) return false;
                user = clear_zero_width(user);
                if (blackName.includes(user)) {
                    colorful_Console.main(
                        { title: "Blocked User", content: user },
                        colorful_Console.colors.warning
                    );
                    this.hidden_item(item, targetElements);
                    return true;
                }
                return this.content_check(item, targetElements, content);
            },
            get_main_element(item) {
                return item.className === "ContentItem AnswerItem"
                    ? item
                    : item.getElementsByClassName("ContentItem AnswerItem")[0];
            },
            check_Hot(item) {
                const hot_list = ["TimeBox-MainContent", "MinorHotSpot"];
                return hot_list.some(
                    (e) => item.getElementsByClassName(e).length > 0
                );
            },
            check_salt(item) {
                return (
                    item.getElementsByClassName(
                        "KfeCollection-OrdinaryLabel-content"
                    ) > 0
                );
            },
            check(item, targetElements) {
                const i = targetElements.index;
                if (i === 3 && this.check_Hot(item)) {
                    if (this.is_simple_search) {
                        const items = item.getElementsByClassName(
                            targetElements.itemClass
                        );
                        for (const it of items) this.check(it, targetElements);
                    }
                    return;
                }
                const tmp = this.get_main_element(item);
                (tmp
                    ? !(i === 3
                          ? this.search_check(tmp, targetElements)
                          : this.user_check(tmp, targetElements) ||
                            this.content_check(tmp, targetElements))
                    : true) &&
                    this.dbInitial &&
                    (i < 2
                        ? this.foldAnswer.check(tmp)
                        : this.foldAnswer.Three.main(
                              item,
                              i === 3 && this.is_simple_search
                          ));
            },
            /*
            1. URL change, for example, forward or backward, ...disable MutationObserver
            2. URL match specific zone;
            */
            checkURL(targetElements) {
                const href = location.href;
                return targetElements.index < 2
                    ? true
                    : (targetElements.index === 2 &&
                          !href
                              .slice(href.lastIndexOf("/topic/") + 7)
                              .includes("/")) ||
                          targetElements.zone.some((e) => href.includes(e));
            },
            clickCheck(item, targetElements) {
                /*
                without userid when in the search page, if the answer is not expanded
                the content only has the abstract section, if in search page and topic page
                check the content, and record the cheched status;
                */
                const id = this.foldAnswer.getid(item);
                (id ? !this.checked_list.includes(id) : true) &&
                    setTimeout(
                        () =>
                            !this.content_check(item, targetElements) &&
                            this.checked_list.push(id),
                        350
                    );
            },
            //check the content when the content expanded
            colorIndicator: {
                lasttarget: null,
                index: 0,
                change: false,
                stat: false,
                color(target) {
                    if (!this.stat) return;
                    this.restore();
                    const colors = ["green", "red", "blue", "purple"];
                    target.style.color = colors[this.index];
                    target.style.fontSize = "16px";
                    target.style.letterSpacing = "0.3px";
                    if (target.style.fontWeight !== 600) {
                        target.style.fontWeight = 600;
                        this.change = true;
                    } else this.change = false;
                    this.index > 2 ? (this.index = 0) : (this.index += 1);
                    this.lasttarget = target;
                },
                restore() {
                    if (this.lasttarget) {
                        this.lasttarget.style.color = "";
                        this.lasttarget.style.fontSize = "";
                        this.lasttarget.style.letterSpacing = "";
                        if (this.change)
                            this.lasttarget.style.fontWeight = "normal";
                        this.lasttarget = null;
                    }
                },
            },
            isReader: false,
            auto_load_reader: false,
            is_scroll_state: false,
            check_click_all(className, localName, target, targetElements) {
                if (className === targetElements.buttonClass) {
                    return this.getiTem(target, targetElements);
                    //click the ico of expand button
                } else if (localName === "svg") {
                    const button = this.svgCheck(target, targetElements);
                    if (button) return this.getiTem(button, targetElements);
                    //click the answer, the content will be automatically expanded
                }
                return null;
            },
            search_clear(target) {
                if (
                    target.localName === "a" &&
                    target.search.includes("&hybrid_search")
                ) {
                    const href = target.href.split("&search_source");
                    if (href.length > 1) {
                        const a = href[0] + "&type=content";
                        GM_openInTab(a, {
                            insert: true,
                            active: true,
                        });
                        target.href = a;
                    }
                    return true;
                }
                return false;
            },
            clickMonitor(node, targetElements) {
                if (this.isMonitor) return;
                const tags = ["blockquote", "p", "br", "li"];
                this.colorIndicator.stat = GM_getValue("highlight");
                if (!node) return;
                node.onclick = (e) => {
                    //when open reader mode, if create click event of document, no node
                    const target = e.target;
                    if (this.search_clear(target)) {
                        e.preventDefault();
                        e.stopImmediatePropagation();
                        e.stopPropagation();
                        return;
                    }
                    if (this.isReader) return;
                    let limit = true;
                    if (targetElements.index < 2)
                        limit = e.path.some(
                            (a) => a.className === targetElements.header
                        );
                    const localName = target.localName;
                    if (limit && tags.includes(localName)) {
                        if (target.style.color || this.foldAnswer.editableMode)
                            return;
                        this.colorIndicator.color(target);
                        return;
                    }
                    this.colorIndicator.restore();
                    const className = target.className;
                    //take care of svg element, the classname
                    if (
                        className &&
                        typeof className === "string" &&
                        className.startsWith("fold") &&
                        this.dbInitial
                    ) {
                        if (targetElements.index < 2) {
                            const ends = [
                                "block",
                                "temp",
                                "element",
                                "select",
                                "edit",
                                "reader",
                            ];
                            !this.is_scroll_state &&
                                ends.some((e) => className.endsWith(e)) &&
                                !this.auto_load_reader &&
                                this.foldAnswer.buttonclick(target);
                        } else {
                            const ends = [
                                "question",
                                "topic",
                                "element",
                                "temp",
                                "answer",
                                "article",
                            ];
                            !this.is_scroll_state &&
                                ends.some((e) => className.endsWith(e)) &&
                                !this.auto_load_reader &&
                                this.foldAnswer.Three.btnClick(target);
                        }
                        return;
                    }
                    //click the expand button, the rich node has contained all content in q & a webpage;
                    if (targetElements.index < 2) return;
                    //-----------------------------------------------
                    let item = this.check_click_all(
                        className,
                        localName,
                        target,
                        targetElements
                    );
                    if (!item) {
                        if (className !== targetElements.expand) return;
                        for (const node of e.path) {
                            const className = node.className;
                            if (
                                className === targetElements.itemClass ||
                                className === targetElements.answerID
                            ) {
                                item = node;
                                break;
                            }
                        }
                    }
                    if (item) this.clickCheck(item, targetElements);
                    else {
                        //some internal url with redirect paramter, the antiredirect function does not treat
                        const path = e.path;
                        let ic = 0;
                        for (const c of path) {
                            if (c.localName === "a") {
                                const cl = c.className;
                                if (
                                    cl &&
                                    typeof cl === "string" &&
                                    !cl.endsWith("external")
                                ) {
                                    const s = c.search;
                                    if (s.startsWith("?target=")) {
                                        e.preventDefault();
                                        e.stopImmediatePropagation();
                                        const h = s.slice(s.indexof("=") + 1);
                                        c.href = h;
                                        window.open(h, "_blank");
                                        break;
                                    }
                                }
                            }
                            ic++;
                            if (ic > 4) break;
                        }
                    }
                };
            },
            topicAndquestion(targetElements, info, index) {
                const items =
                    document.getElementsByClassName("ContentItem-meta");
                let n = items.length;
                for (n; n--; ) {
                    const item = items[n];
                    const a = item.getElementsByClassName("UserLink-link");
                    let i = a.length;
                    if (i > 0) {
                        const username = a[--i].innerText;
                        if (username === info.username) {
                            const t = this.getiTem(item, targetElements);
                            t &&
                                (index === 0
                                    ? this.setDisplay(t.parentNode, info)
                                    : this.setDisplay(t, info));
                        }
                    }
                }
            },
            setDisplay(t, info) {
                info.mode === "block"
                    ? t.style.display !== "none" && (t.style.display = "none")
                    : t.style.display === "none" && (t.style.display = "block");
            },
            userChange(index) {
                const info = GM_getValue("blacknamechange");
                if (!info) return;
                const targetElements = this.getTagetElements(
                    index === 0 ? 1 : index
                );
                index === 0 &&
                    (targetElements.itemClass = "ContentItem AnswerItem");
                if (!this.checkURL(targetElements)) return;
                if (index === 3) {
                    const items = document.getElementsByClassName(
                        targetElements.itemClass
                    );
                    let n = items.length;
                    for (n; n--; ) {
                        const item = items[n];
                        const a = item.getElementsByClassName(
                            targetElements.userID
                        );
                        let i = a.length;
                        if (i > 0) {
                            const name = a[--i].innerText;
                            name === info.username &&
                                this.setDisplay(item, info);
                        } else {
                            const content = item.getElementsByClassName(
                                targetElements.contentID
                            );
                            if (content.length === 0) continue;
                            const text = content[0].innerText;
                            const name = text.startsWith("匿名用户:")
                                ? ""
                                : text.slice(0, text.indexOf(":"));
                            name &&
                                name === info.username &&
                                this.setDisplay(item, info);
                        }
                    }
                } else this.topicAndquestion(targetElements, info, index);
            },
            //standby, commmunication between others and column
            connectColumn() {
                const r = GM_getValue("removearticleA");
                if (r && Array.isArray(r) && r.length > 0) {
                    for (const e of r) dataBaseInstance.dele(false, e);
                    GM_setValue("removearticleA", "");
                }
                const b = GM_getValue("blockarticleA");
                if (b && Array.isArray(b) && b.length > 0) {
                    for (const e of b) dataBaseInstance.fold(e);
                    GM_setValue("blockarticleA", "");
                }
                GM_addValueChangeListener(
                    "blockarticleA",
                    (name, oldValue, newValue, remote) => {
                        if (remote) {
                            dataBaseInstance.fold(newValue[0]);
                            GM_setValue("blockarticleA", "");
                        }
                    }
                );
                GM_addValueChangeListener(
                    "removearticleA",
                    (name, oldValue, newValue, remote) => {
                        if (remote) {
                            dataBaseInstance.dele(false, newValue[0]);
                            GM_setValue("removearticleA", "");
                        }
                    }
                );
                this.is_connect = true;
            },
            simple_search(item) {
                const arr = [
                    "RelevantQuery",
                    "KfeCollection-PcCollegeCard-wrapper",
                    "ContentItem ZvideoItem",
                    "SearchClubCard",
                    "ContentItem-extra",
                    "SearchItem-userTitleWrapper",
                ];
                for (const e of arr) {
                    if (item.getElementsByClassName(e).length > 0) {
                        item.className = `${item.className} hidden`;
                        item.style.display = "none";
                        break;
                    }
                }
            },
            monitor(targetElements, node) {
                //only in answer || question webpage, the muta will note be destroyed, without need to rebuilded;
                const mo = new MutationObserver((e) => {
                    e.forEach((item) =>
                        item.addedNodes.length > 0 &&
                        item.addedNodes[0].className ===
                            targetElements.itemClass
                            ? this.check(item.addedNodes[0], targetElements)
                            : targetElements.index === 3 &&
                              this.is_simple_search &&
                              this.simple_search(item.addedNodes[0])
                    );
                    targetElements.index === 1 &&
                        e.length > 2 &&
                        this.reader_sync();
                });
                mo.observe(node.parentNode, { childList: true });
            },
            getTagetElements(index) {
                const pos = {
                    0: "answerPage",
                    1: "questionPage",
                    2: "topicPage",
                    3: "searchPage",
                };
                const targetElements = this[pos[index]](index);
                return targetElements;
            },
            dbInitial: false,
            reader_sync: null,
            checked_list: null,
            is_simple_search: false,
            main(index, reader_sync) {
                this.foldAnswer.initial().then((r) => {
                    index > 1 && (this.checked_list = []);
                    this.dbInitial = r;
                    this.reader_sync = reader_sync;
                    const targetElements = this.getTagetElements(index);
                    index !== 0 && this.firstRun(targetElements);
                    index !== 3 && this.Topic_questionButton(targetElements);
                    const w =
                        window.onurlchange === null
                            ? window
                            : unsafeWindow.onurlchange === null
                            ? unsafeWindow
                            : null;
                    w &&
                        w.addEventListener("urlchange", () =>
                            this.backwardORforward(targetElements)
                        );
                    this.connectColumn();
                    !this.isMonitor &&
                        this.clickMonitor(
                            document.getElementById(targetElements.mainID),
                            targetElements
                        );
                });
            },
            is_jump: false,
            bf_time_id: null,
            backwardORforward(targetElements) {
                //monitor forward or backward, this operation maybe not fire dom change event
                if (this.is_jump) {
                    this.is_jump = false;
                    return;
                }
                this.bf_time_id && clearTimeout(this.bf_time_id);
                this.bf_time_id = setTimeout(() => {
                    this.bf_time_id = null;
                    this.firstRun(targetElements);
                }, 350);
            },
            is_update: false,
            search_items() {
                return document.getElementsByClassName("List")[0]
                    .firstElementChild.children;
            },
            common_items(cl) {
                return document.getElementsByClassName(cl);
            },
            get_items(cl, ic, n, mode, targetElements, w) {
                setTimeout(() => {
                    const items = this[w](cl);
                    const i = items.length;
                    if (i > n || ic > 20) {
                        if (i === 0) {
                            colorful_Console.main(
                                {
                                    title: "info:",
                                    content: "failed to get items",
                                },
                                colorful_Console.colors.info
                            );
                            return;
                        }
                        for (const item of items)
                            this.check(item, targetElements, 0);
                        mode && this.monitor(targetElements, items[0]);
                    } else this.get_items(cl, ++ic, n, mode, targetElements, w);
                }, 50 + 5 * ic);
            },
            firstRun(targetElements) {
                if (!this.checkURL(targetElements)) return;
                let n = 4;
                let cl = "";
                let mode = true;
                const i = targetElements.index;
                if (i === 3 && this.is_simple_search) {
                    this.get_items(
                        cl,
                        0,
                        n,
                        mode,
                        targetElements,
                        "search_items"
                    );
                    return;
                }
                if (i > 1) cl = targetElements.itemClass;
                else {
                    const p = location.pathname;
                    const a = p.includes("/answers/");
                    cl = !(n = p.includes("/answer/") ? 0 : 4)
                        ? targetElements.header
                        : targetElements.itemClass;
                    mode = this.is_update ? !a : true;
                    this.is_update = a;
                    targetElements.index = n === 0 ? 0 : 1;
                    if (this.is_update && !mode) return;
                }
                this.get_items(cl, 0, n, mode, targetElements, "common_items");
            },
            isMonitor: false,
            answerPage() {
                const targetElements = this.questionPage(1);
                const items = document.getElementsByClassName(
                    targetElements.header
                );
                for (const item of items)
                    this.check(item.parentNode, targetElements, 0);
                this.clickMonitor(document, targetElements);
                this.isMonitor = true;
                return targetElements;
            },
            questionPage(index) {
                const targetElements = {
                    buttonClass:
                        "Button ContentItem-rightButton ContentItem-expandButton Button--plain",
                    itemClass: "List-item",
                    mainID: "QuestionAnswers-answers",
                    contentID: "RichText ztext CopyrightRichText-richText",
                    userID: "UserLink-link",
                    backupClass: "Question-main",
                    header: "ContentItem AnswerItem",
                    headerID: "QuestionHeader",
                    headerTitle: "QuestionHeader-title",
                    expand: "RichText ztext CopyrightRichText-richText",
                    answerID: "ContentItem AnswerItem",
                    inserID: "LabelContainer-wrapper",
                    blockValueName: "blacktopicAndquestion",
                    inserButtonID: "QuestionHeaderActions",
                    index: index,
                };
                return targetElements;
            },
            searchPage(index) {
                const nocontent = document.getElementsByClassName(
                    "SearchNoContent-title"
                );
                if (nocontent.length > 0) return null;
                const targetElements = {
                    buttonClass: "Button ContentItem-more Button--plain",
                    itemClass: "Card SearchResult-Card",
                    mainID: "SearchMain",
                    contentID: "RichText ztext CopyrightRichText-richText",
                    expand: "RichContent-inner",
                    userID: "UserLink-link",
                    zone: ["type=content"],
                    index: index,
                };
                return targetElements;
            },
            topicPage(index) {
                const targetElements = {
                    buttonClass: "Button ContentItem-more Button--plain",
                    itemClass: "List-item TopicFeedItem",
                    mainID: "TopicMain",
                    userID: "UserLink-link",
                    headerID: "ContentItem-head",
                    headerTitle: "ContentItem-title",
                    contentID: "RichText ztext CopyrightRichText-richText",
                    expand: "RichContent-inner",
                    inserButtonID: "TopicActions",
                    zone: ["/top-answers", "/hot", "newest"],
                    blockValueName: "blacktopicAndquestion",
                    index: index,
                };
                return targetElements;
            },
            foldAnswer: {
                Three: {
                    initialR: false,
                    btnClick(button) {
                        const text = button.innerText;
                        if (text.startsWith("show")) {
                            this.showFold(button);
                        } else {
                            let bid = "";
                            if (text !== "Fold") {
                                bid = this.getbid(button);
                                if (!bid) return;
                            }
                            this[text](button, bid);
                        }
                    },
                    getbid(button) {
                        const attrs = button.attributes;
                        if (!attrs) return null;
                        for (const e of attrs)
                            if (e.name === "bid") return e.value;
                        return null;
                    },
                    Answer_Article(button, bid, type, n, t, from = "") {
                        const p = button.parentNode.parentNode;
                        const user = p.getElementsByClassName("UserLink-link");
                        let i = user.length - 1;
                        const info = {};
                        info.userID = "";
                        info.userName = "";
                        if (i > 0) {
                            i = i > 1 ? 1 : i;
                            info.userName = user[i].innerText;
                            const pn = user[i].pathname;
                            info.userID = pn.slice(pn.lastIndexOf("/") + 1);
                        }
                        info.from = from;
                        info.name = bid;
                        info.update = Date.now();
                        info.type = type;
                        dataBaseInstance.fold(info);
                        this.changeBtn(button, n, t);
                    },
                    changeBtn(button, n, t) {
                        button.innerText = n;
                        button.title = t;
                        this.Fold(button, "show blocked");
                    },
                    Question_Topic(button, bid, type, n, t) {
                        if (blackTopicAndQuestion.some((e) => e.id === bid))
                            return;
                        const info = {};
                        info.id = bid;
                        info.type = type;
                        info.update = Date.now();
                        blackTopicAndQuestion.push(info);
                        GM_setValue(
                            "blacktopicAndquestion",
                            blackTopicAndQuestion
                        );
                        this.changeBtn(button, n, t);
                    },
                    Answer(button, bid) {
                        const from = this.getbid(button.nextElementSibling);
                        this.Answer_Article(
                            button,
                            bid,
                            "answer",
                            "Remove",
                            "unblock the answer",
                            from ? from : ""
                        );
                    },
                    Question(button, bid) {
                        this.Question_Topic(
                            button,
                            bid,
                            "question",
                            "Remove",
                            "unblock the question"
                        );
                    },
                    Article(button, bid) {
                        this.Answer_Article(
                            button,
                            bid,
                            "article",
                            "Remove",
                            "unblock the article"
                        );
                        let b = GM_getValue("blockarticleB");
                        if (b && Array.isArray(b)) {
                            for (const e of b) if (e.pid === bid) return;
                        } else b = [];
                        const r = GM_getValue("removearticleB");
                        if (r && Array.isArray(r)) {
                            const i = r.indexOf(bid);
                            if (i > -1) {
                                r.splice(i, 1);
                                GM_setValue("removearticleB", r);
                            }
                        }
                        const info = {};
                        info.userID = "";
                        info.from = "";
                        info.pid = bid;
                        info.value = 0;
                        info.update = Date.now();
                        b.push(info);
                        GM_setValue("blockarticleB", b);
                    },
                    Topic(button, bid) {
                        this.Question_Topic(
                            button,
                            bid,
                            "topic",
                            "Remove",
                            "unblock the topic"
                        );
                    },
                    Remove(button, bid) {
                        const className = button.className;
                        const method = {
                            changeBtn(n, t) {
                                button.innerText = n;
                                button.title = t;
                                button.parentNode.previousElementSibling.innerText =
                                    "show folded";
                            },
                            answer() {
                                dataBaseInstance.dele(false, bid);
                                this.changeBtn("Answer", "block the answer");
                            },
                            remove() {
                                let i = -1;
                                for (const e of blackTopicAndQuestion) {
                                    i++;
                                    if (e.id === bid) break;
                                }
                                if (i < 0) return;
                                blackTopicAndQuestion.splice(i, 1);
                                GM_setValue(
                                    "blacktopicAndquestion",
                                    blackTopicAndQuestion
                                );
                            },
                            topic() {
                                this.remove();
                                this.changeBtn("Topic", "block the topic");
                            },
                            article() {
                                dataBaseInstance.dele(false, bid);
                                this.changeBtn("Article", "block the article");
                                let r = GM_getValue("removearticleB");
                                if (r && Array.isArray(r) && r.length > 0) {
                                    if (r.includes(bid)) return;
                                } else r = [];
                                const b = GM_getValue("blockarticleB");
                                if (b && Array.isArray(b)) {
                                    const i = b.findIndex((e) => e.pid === bid);
                                    if (i > -1) {
                                        b.splice(i, 1);
                                        GM_setValue("blockarticleB", b);
                                    }
                                }
                                r.push(bid);
                                GM_setValue("removearticleB", r);
                            },
                            question() {
                                this.remove();
                                this.changeBtn(
                                    "Question",
                                    "block the question"
                                );
                            },
                        };
                        const n = className.slice(className.indexOf("_") + 1);
                        method[n]();
                    },
                    Fold(button, n = "") {
                        const p = button.parentNode;
                        p.previousElementSibling.style.display = "block";
                        p.nextElementSibling.style.display = "none";
                        p.style.display = "none";
                        n && (p.previousElementSibling.innerText = n);
                    },
                    showFold(button) {
                        button.style.display = "none";
                        const n = button.nextElementSibling;
                        if (!n) return;
                        n.style.display = "grid";
                        n.nextElementSibling.style.display = "block";
                    },
                    simple_search_hide(item) {
                        item.className = `${item.className} hidden`;
                        item.style.display = "none";
                    },
                    check_simple_search(p) {
                        const arr = [
                            "/zvideo/",
                            "/lives/",
                            "/club/",
                            "/market/",
                            "/people/",
                        ];
                        return arr.some((e) => p.includes(e));
                    },
                    market_ad(item) {
                        const arr = [
                            "RelevantQuery",
                            "KfeCollection-PcCollegeCard-wrapper",
                            "ContentItem ZvideoItem",
                            "SearchClubCard",
                            "ContentItem-extra",
                            "SearchItem-userTitleWrapper",
                        ];
                        for (const e of arr) {
                            if (item.getElementsByClassName(e).length > 0) {
                                this.simple_search_hide(item);
                                break;
                            }
                        }
                    },
                    getInfo(item, mode) {
                        const title = item.getElementsByTagName("h2");
                        if (title.length === 0) {
                            mode && this.market_ad(item);
                            return null;
                        }
                        if (mode && title[0].innerText === "相关搜索") {
                            this.simple_search_hide(item);
                            return null;
                        }
                        const a = title[0].getElementsByTagName("a");
                        if (a.length === 0) {
                            mode && this.market_ad(item);
                            return null;
                        }
                        const text = a[0].innerText;
                        if (
                            text &&
                            blackKey.some((e) => {
                                if (text.includes(e)) {
                                    item.style.display = "none";
                                    colorful_Console.main(
                                        {
                                            title: "Blocked title: ",
                                            content: "rubbish word " + e,
                                        },
                                        colorful_Console.colors.warning
                                    );
                                    return true;
                                }
                                return false;
                            })
                        )
                            return null;
                        const p = a[0].pathname;
                        if (mode && this.check_simple_search(p)) {
                            this.simple_search_hide(item);
                            return null;
                        }
                        const info = {};
                        info.cblock = false;
                        info.ablcok = false;
                        info.qblock = false;
                        info.tblock = false;
                        const flags = ["/question", "/p/", "/topic"];
                        const index = flags.findIndex((e) => p.includes(e));
                        if (index === 0) {
                            info.type = "answer";
                            const tmp = p.split("/");
                            if (tmp.length !== 5) return null;
                            info.qid = tmp[2];
                            info.aid = tmp[4];
                        } else if (index === 1) {
                            info.type = "column";
                            info.cid = p.slice(p.lastIndexOf("/") + 1);
                        } else if (index === 2) {
                            info.type = "topic";
                            info.tid = p.slice(p.lastIndexOf("/") + 1);
                        } else return null;
                        return info;
                    },
                    btnRaw(c, t, n, b) {
                        return `<button class="fold_${c}" title=${escapeBlank(
                            t
                        )} bid=${b}>${n}</button>`;
                    },
                    foldRaw(arr, info, adisplay, bdisplay, name) {
                        const d = JSON.stringify(info);
                        return `
                            <div
                                class="fold_element"
                                title="the ${info.type} has been folded"
                                data-info=${d}
                                style="display:${adisplay};"
                            >
                                show ${name}
                            </div>
                                <div class="hidden_fold" data-info=${d} style="display:${bdisplay};">
                                ${arr.join("")}
                                <button class="fold_temp" title="temporarily fold the item">Fold</button>
                            </div>`;
                    },
                    checkAnswerOrArticle(id) {
                        return new Promise((resolve, reject) => {
                            dataBaseInstance.check(id).then(
                                (result) => resolve(result),
                                (err) => {
                                    console.log(err);
                                    reject();
                                }
                            );
                        });
                    },
                    checkTAndQ(id) {
                        return blackTopicAndQuestion.some((e) => e.id === id);
                    },
                    answer(item, info) {
                        const exe = (r) => {
                            const f = this.checkTAndQ(info.qid);
                            const html = [];
                            info.qblock = f;
                            info.ablcok = r;
                            (r || f) && this.hide_content(item);
                            let [t, n] = r
                                ? ["unblock", "Remove"]
                                : ["block", "Answer"];
                            html.push(
                                this.btnRaw(
                                    "answer",
                                    t + " the answer",
                                    n,
                                    info.aid
                                )
                            );
                            [t, n] = f
                                ? ["unblock", "Remove"]
                                : ["block", "Question"];
                            html.push(
                                this.btnRaw(
                                    "question",
                                    t + " the question",
                                    n,
                                    info.qid
                                )
                            );
                            const [a, b, name] =
                                r || f
                                    ? ["grid", "none", "blocked"]
                                    : ["none", "grid", "folded"];
                            item.insertAdjacentHTML(
                                "afterbegin",
                                this.foldRaw(html, info, a, b, name)
                            );
                        };
                        this.initialR
                            ? this.checkAnswerOrArticle(info.aid).then(
                                  (r) => exe(r),
                                  () => console.log("check blocked answer fail")
                              )
                            : exe(false);
                    },
                    hide_content(item) {
                        item.firstChild.style.display = "none";
                    },
                    column(item, info) {
                        const exe = (r) => {
                            info.cblock = r;
                            r && this.hide_content(item);
                            const [t, n, a, b, name] = r
                                ? [
                                      "unblock",
                                      "Remove",
                                      "grid",
                                      "none",
                                      "blocked",
                                  ]
                                : [
                                      "block",
                                      "Article",
                                      "none",
                                      "grid",
                                      "folded",
                                  ];
                            item.insertAdjacentHTML(
                                "afterbegin",
                                this.foldRaw(
                                    [
                                        this.btnRaw(
                                            "article",
                                            t + " the article",
                                            n,
                                            info.cid
                                        ),
                                    ],
                                    info,
                                    a,
                                    b,
                                    name
                                )
                            );
                        };
                        this.initialR
                            ? this.checkAnswerOrArticle(info.cid).then(
                                  (r) => exe(r),
                                  () =>
                                      console.log("check blocked article fail")
                              )
                            : exe(false);
                    },
                    topic(item, info) {
                        const f = this.checkTAndQ(info.tid);
                        info.tblock = f;
                        f && this.hide_content(item);
                        const [t, n, a, b, name] = f
                            ? ["unblock", "Remove", "block", "none", "blocked"]
                            : ["block", "Topic", "none", "grid", "folded"];
                        item.insertAdjacentHTML(
                            "afterbegin",
                            this.foldRaw(
                                [
                                    this.btnRaw(
                                        "topic",
                                        t + " the topic",
                                        n,
                                        info.tid
                                    ),
                                ],
                                info,
                                a,
                                b,
                                name
                            )
                        );
                    },
                    main(item, mode) {
                        const info = this.getInfo(item, mode);
                        if (!info) return;
                        this[info.type](item, info);
                    },
                },
                //---------------------------------------------------------------------------------------
                buttonclick(button) {
                    const text = button.innerText;
                    text.startsWith("show")
                        ? this.showFold(button, text)
                        : this[text](button);
                },
                getcontent(button, pnode) {
                    const p = pnode || this.getpNode(button);
                    if (!p) return null;
                    const expand = p.getElementsByClassName(
                        "Button ContentItem-rightButton ContentItem-expandButton Button--plain"
                    );
                    if (expand.length > 0) expand[0].click();
                    const ele = p.getElementsByClassName(
                        "RichText ztext CopyrightRichText-richText"
                    );
                    return ele.length === 0 ? null : ele[0];
                },
                editableMode: false,
                Edit(button) {
                    const ele = this.getcontent(button);
                    if (!ele) return;
                    this.editableMode = true;
                    ele.contentEditable = true;
                    button.innerText = "Exit";
                    button.title = "exit editable mode";
                },
                Reader(button) {
                    if (this.editableMode) {
                        Notification("please exit editable mode", "Tips");
                        return;
                    }
                    const pnode = this.getpNode(button);
                    if (!pnode) return;
                    const aid = this.getid(pnode);
                    if (!aid) return;
                    zhihu.qaReader.main(pnode, aid);
                },
                Exit(button) {
                    const ele = this.getcontent(button);
                    if (!ele) return;
                    this.editableMode = false;
                    ele.contentEditable = false;
                    button.innerText = "Edit";
                    button.title = "edit the answer";
                },
                Select(button) {
                    if (this.editableMode) {
                        Notification("please exit editable mode", "Tips");
                        return;
                    }
                    const ele = this.getcontent(button);
                    if (!ele) return;
                    const selection = window.getSelection();
                    selection.removeAllRanges();
                    const range = new Range();
                    range.selectNodeContents(ele);
                    selection.addRange(range);
                },
                getid(item) {
                    const tmp =
                        item.className === "ContentItem AnswerItem"
                            ? item
                            : item.getElementsByClassName(
                                  "ContentItem AnswerItem"
                              )[0];
                    if (!tmp) return null;
                    const attrs = tmp.attributes;
                    if (!attrs) return null;
                    for (const a of attrs)
                        if (a.name === "name") return a.value;
                    return null;
                },
                getpNode(button) {
                    let pnode = button.parentNode;
                    let ic = 0;
                    while (pnode.className !== "ContentItem AnswerItem") {
                        pnode = pnode.parentNode;
                        ic++;
                        if (ic > 4 || !pnode) return null;
                    }
                    return pnode;
                },
                get from() {
                    const p = location.pathname;
                    const reg = /(?<=question\/)\d+/;
                    const m = p.match(reg);
                    return m ? m[0] : "";
                },
                Block(button, type = "answer") {
                    const p = this.getpNode(button);
                    if (!p) return;
                    const id = this.getid(p);
                    if (!id) return;
                    const info = {};
                    const user = p.getElementsByClassName("UserLink-link");
                    let i = user.length - 1;
                    if (i > 0) {
                        i = i > 1 ? 1 : i;
                        info.userName = user[i].innerText;
                        const pn = user[i].pathname;
                        info.userID = pn.slice(pn.lastIndexOf("/") + 1);
                    }
                    info.name = id;
                    info.update = Date.now();
                    info.type = type;
                    info.from = this.from;
                    dataBaseInstance.fold(info);
                    button.innerText = "Remove";
                    button.title = "remove the answer from block list";
                    this.insertShowFolded(p, "blocked");
                },
                Remove(button) {
                    //show temp button
                    const p = this.getpNode(button);
                    if (!p) return;
                    const id = this.getid(p);
                    if (!id) return;
                    dataBaseInstance.dele(false, id);
                    button.innerText = "Block";
                    button.title = "fold the answer forever";
                },
                Fold(button) {
                    const p = this.getpNode(button);
                    if (!p) return;
                    this.insertShowFolded(p, "folded");
                },
                showFold(button, text) {
                    const next = button.nextElementSibling;
                    next.style.display = "block";
                    button.style.display = "none";
                    if (text.endsWith("blocked")) this.insertButton(next, true);
                },
                check(item) {
                    if (!item) return;
                    if (this.initialR === 0) {
                        this.insertButton(item);
                        return;
                    }
                    const id = this.getid(item);
                    if (!id) return;
                    dataBaseInstance.check(id).then(
                        (r) => {
                            r
                                ? this.insertShowFolded(item, "blocked")
                                : this.insertButton(item);
                        },
                        (err) => console.log(err)
                    );
                },
                initialR: 0,
                initial() {
                    return new Promise((resolve) => {
                        const tables = ["foldedAnswer"];
                        dataBaseInstance.initial(tables, true, "name").then(
                            (result) => {
                                resolve(true),
                                    (this.Three.initialR = this.initialR =
                                        result);
                            },
                            (err) => {
                                console.log(err);
                                resolve(false);
                            }
                        );
                    });
                },
                insertShowFolded(item, name) {
                    //if it has already contained this element , to hide or show
                    const f =
                        item.parentNode.getElementsByClassName("fold_element");
                    if (f.length === 0) {
                        const html = `
                        <div
                            class="fold_element"
                            title="the answer has been folded"
                        >
                            show ${name}
                        </div>`;
                        item.insertAdjacentHTML("beforebegin", html);
                        item.style.display = "none";
                    } else {
                        item.style.display = "none";
                        f[0].style.display = "block";
                        f[0].innerText = `show ${name}`;
                    }
                },
                insertButton(item, mode = false) {
                    const h = item.getElementsByClassName("hidden_fold");
                    if (h.length === 0) {
                        const obutton = `
                            <button class="fold_temp" title="temporarily fold the answer">Fold</button>
                            <button class="fold_edit" title="edit the answer">Edit</button>
                            <button class="fold_select" title="select the answer">Select</button>
                            <button class="fold_reader" title="open the answer in reader" style="color: #F3752C; margin-right: 12px;">Reader</button>`;
                        const html = `
                        <div class="hidden_fold">
                            <button class="fold_block" title="fold the answer forever">Block</button>
                            ${obutton}
                        </div>`;
                        const r = `
                        <div class="hidden_fold">
                            <button class="fold_block" title="remove the answer from block list">Remove</button>
                            ${obutton}
                        </div>`;
                        item.firstElementChild.lastElementChild.insertAdjacentHTML(
                            "beforebegin",
                            mode ? r : html
                        );
                    }
                },
            },
        },
        searchPage: {
            is_simple_search: false,
            get search_simple() {
                let simple = "";
                const common = `
                    .RelevantQuery,
                    .KfeCollection-PcCollegeCard-wrapper,
                    .ContentItem.ZvideoItem,
                    .SearchClubCard{display: none !important;}`;
                for (let i = 2; i < 10; i++)
                    simple += `.SearchTabs-actions li.Tabs-item.Tabs-item--noMeta:nth-of-type(${i}),`;
                return simple + common;
            },
            get raw() {
                //html & css, adaped from bilibili
                const [c, t] = this.is_simple_search
                    ? [" on", escapeBlank("restore normal mode")]
                    : [
                          "",
                          escapeHTML(
                              "remove other items, just keep question, topic, article"
                          ),
                      ];
                const html = `
                <div class="simple_header" title=${t}>
                    <style>
                        .next-button {
                            font-size: 12px;
                            color: #999;
                            line-height: 22px;
                            cursor: pointer;
                        }
                        .next-button .txt {
                            margin-right: 8px;
                            vertical-align: middle;
                        }
                        .next-button .switch-button.on {
                            border: 1px solid #00a1d6;
                            background: #00a1d6;
                        }
                        .next-button .switch-button {
                            margin: 0;
                            display: inline-block;
                            position: relative;
                            width: 30px;
                            height: 20px;
                            border: 1px solid #ccc;
                            outline: none;
                            border-radius: 10px;
                            box-sizing: border-box;
                            background: #ccc;
                            cursor: pointer;
                            transition: border-color 0.2s, background-color 0.2s;
                            vertical-align: middle;
                        }
                        .next-button .switch-button.on:after {
                            left: 11px;
                        }
                        .next-button .switch-button:after {
                            content: "";
                            position: absolute;
                            top: 1px;
                            left: 1px;
                            border-radius: 100%;
                            width: 16px;
                            height: 16px;
                            background-color: #fff;
                            transition: all 0.2s;
                        }
                    </style>
                    <span class="next-button"
                        ><span class="txt">Simple Mode</span><span class="switch-button${c}"></span
                    ></span>
                </div>`;
                return html;
            },
            getPos(node) {
                const search =
                    node.getElementsByClassName("SearchTabs-actions");
                if (search.length === 0) {
                    colorful_Console.main(
                        {
                            title: "Warning:",
                            content:
                                "search page does not get the target element",
                        },
                        colorful_Console.colors.warning
                    );
                    return null;
                }
                return search;
            },
            buttons: null,
            main() {
                const search = this.getPos(document);
                if (!search) return;
                const i = search.length;
                const html = this.raw;
                this.buttons = [];
                if (i > 1) for (const c of search) this.click_event(c, html);
                else {
                    this.click_event(search[0], html);
                    this.monitor();
                }
            },
            monitor() {
                let mo = new MutationObserver((e) => {
                    if (e.length === 1 && e[0].addedNodes.length === 1) {
                        const newNode = e[0].addedNodes[0];
                        if (newNode.className.startsWith("PageHeader")) {
                            const search = this.getPos(newNode);
                            if (!search) return;
                            mo.disconnect();
                            mo = null;
                            this.click_event(search[0], this.raw);
                        }
                    }
                });
                mo.observe(
                    document.getElementsByClassName("Sticky AppHeader")[0]
                        .lastElementChild,
                    { childList: true }
                );
            },
            get _list() {
                const list = document.getElementsByClassName("List");
                return list.length > 0
                    ? list[0].firstElementChild.children
                    : null;
            },
            pNode_hide() {
                const arr = [
                    "RelevantQuery",
                    "KfeCollection-PcCollegeCard-wrapper",
                    "ContentItem ZvideoItem",
                    "SearchClubCard",
                    "ContentItem-extra",
                    "SearchItem-userTitleWrapper",
                ];
                const list = this._list;
                if (!list) return;
                for (const l of list) {
                    for (const e of arr) {
                        if (l.getElementsByClassName(e).length > 0) {
                            l.className = `${l.className} hidden`;
                            l.style.display = "none";
                            break;
                        }
                    }
                }
            },
            pNode_show() {
                const list = this._list;
                if (!list) return;
                for (const l of list) {
                    const c = l.className;
                    if (c && c.endsWith(" hidden")) {
                        l.className = c.slice(0, -7);
                        l.style.display = "block";
                    }
                }
            },
            click_event(c, html) {
                c.insertAdjacentHTML("beforeend", html);
                setTimeout(() => {
                    let timeid = null;
                    const node =
                        c.lastElementChild.lastElementChild.lastElementChild;
                    this.buttons.push(node);
                    node.onclick = (e) => {
                        timeid && clearTimeout(timeid);
                        timeid = setTimeout(() => {
                            timeid = null;
                            let f = false;
                            const className = e.target.className;
                            let newName = "";
                            if (className.endsWith(" on")) {
                                if (this.id) {
                                    const style = document.getElementById(
                                        this.id
                                    );
                                    this.pNode_show();
                                    style && style.remove();
                                    this.id = null;
                                }
                                newName = className.slice(
                                    0,
                                    className.length - 3
                                );
                            } else {
                                this.id = GM_addStyle(this.search_simple).id;
                                f = true;
                                newName = className + " on";
                                this.pNode_hide();
                            }
                            this.buttons.forEach(
                                (e) => (e.className = newName)
                            );
                            this.is_simple_search = f;
                            GM_setValue("simplesearch", f);
                        }, 300);
                    };
                }, 0);
            },
            id: null,
            add(common, inpustyle, search, topicAndquestion, ad, bgi) {
                GM_addStyle(
                    common + inpustyle + search + topicAndquestion + ad + bgi
                );
                (this.is_simple_search = GM_getValue("simplesearch")) &&
                    (this.id = GM_addStyle(this.search_simple).id);
            },
        },
        body_img_update: null,
        body_image() {
            const bgi = this.commander.bgi.get;
            let s = "";
            if (bgi && (s = bgi.status) && s !== "tmp") {
                const url =
                    s === "auto"
                        ? (this.body_img_update = bgi) && bgi.url
                        : s === "fixed" && GM_getValue("fixed_image");
                return url ? `body{background-image: url(${url});}` : "";
            }
            return "";
        },
        addStyle(index) {
            const common = `
                .css-1hwwfws,
                .css-1ynzxqw,
                .ModalExp-content{display: none !important;}
                span.RichText.ztext.CopyrightRichText-richText{text-align: justify !important;}
                body{background-attachment: fixed !important;text-shadow: #a9a9a9 0.025em 0.015em 0.02em;}`;
            const showfold = `
                .fold_element{
                    max-height: 24px;
                    margin-left: 45%;
                    margin-top: 3px;
                    width: 100px;
                    text-align: center;
                    border: 1px solid #ccc !important;
                }`;
            const contentstyle = `
                ${showfold}
                .hidden_fold:hover {
                     opacity: 1;
                     transition: opacity 2s;
                }
                .hidden_fold{opacity: 0.15;}
                .hidden_fold button {
                    float: right;
                    border: 1px solid #ccc!important;
                    box-shadow: 1px 1px 4px #888888;
                    height: 21px;
                    font-size: 14px;
                    width: 54px;
                    border-radius: 5px;
                }
                div.Question-mainColumn{
                    margin: auto !important;
                    width: 100% !important;
                    max-width: 1000px !important;
                    min-width: 1000px !important;
                }
                .RichContent.RichContent--unescapable{width: 100% !important;}
                figure{max-width: 70% !important;}
                .RichContent-inner{
                    line-height: 30px !important;
                    margin: 35px 0px !important;
                    padding: 25px 30px !important;
                    border: 6px dashed rgba(133,144,166,0.2) !important;
                    border-radius: 6px !important;
                }
                .Comments{padding: 12px !important; margin: 60px !important;}`;
            const inpustyle = `
                input::-webkit-input-placeholder {
                    font-size: 0px !important;
                    text-align: right;
                }`;
            const search = `
                .SearchResult-Card{opacity: 0.95 !important;}
                .SearchMain{width: 930px !important;}
                .SearchSideBar,
                .Card.TopSearch{display: none !important;}
                .KfeCollection-PcCollegeCard-wrapper,
                .List-item{border: 1px solid transparent;}
                .List-item{position: inherit !important;}
                .KfeCollection-PcCollegeCard-wrapper:hover,
                .List-item:hover {
                    border: 1px solid #B9D5FF;
                    box-shadow: 1px 1px 2px 0 rgba(0, 0, 0, 0.10);
                }`;
            const topicAndquestion = `
                ${showfold}
                .hidden_fold {
                    opacity: 0.15;
                    float: right;
                }
                .hidden_fold:hover {
                     opacity: 1;
                     transition: opacity 2s;
                }
                .hidden_fold button {
                    border: 1px solid #ccc!important;
                    box-shadow: 1px 1px 0px #888888;
                    height: 18px;
                    font-size: 12px;
                    width: 54px;
                    border-radius: 5px;
                    margin-top: 2px;
                }`;
            const topic = `
                .ContentLayout{width: 100% !important;}
                .ContentLayout-mainColumn{
                    margin-left: 23%;
                    width: 930px !important;
                }
                .ContentLayout-sideColumn{
                    margin-right: 15%;
                }
                .List-item.TopicFeedItem{border: 1px solid transparent;}
                .List-item.TopicFeedItem:hover {
                    border: 1px solid #B9D5FF;
                    box-shadow: 1px 1px 2px 0 rgba(0, 0, 0, 0.10);
                }`;
            const ad = `
                .css-1ynzxqw,
                a[href*="u.jd.com"],
                .Pc-word,
                .RichText-ADLinkCardContainer,
                .MCNLinkCard,
                .RichText-MCNLinkCardContainer,
                div.Question-sideColumn,.Kanshan-container,
                span.LinkCard-content.LinkCard-ecommerceLoadingCard,
                .RichText-MCNLinkCardContainer{display: none !important;}`;
            index === 3
                ? this.searchPage.add(
                      common,
                      inpustyle,
                      search,
                      topicAndquestion,
                      ad,
                      this.body_image()
                  )
                : GM_addStyle(
                      common +
                          (index < 2
                              ? contentstyle +
                                inpustyle +
                                this.body_image() +
                                ad
                              : index === 2
                              ? inpustyle +
                                topicAndquestion +
                                topic +
                                this.body_image() +
                                ad
                              : inpustyle)
                  );
        },
        clearStorage() {
            const rubbish = {};
            rubbish.timeStamp = Date.now();
            rubbish.words = [];
            //localstorage must storage this info to ensure the history show
            for (let i = 0; i < 5; i++)
                rubbish.words.push({ displayQuery: "", query: "" });
            localStorage.setItem("search::top-search", JSON.stringify(rubbish));
            localStorage.setItem("search:preset_words", "");
            localStorage.setItem("zap:SharedSession", "");
        },
        /*
        disable blank search hot word;
        disable show hot seach result;
        clear placeholder
        导致Promise出现错误
        */
        inputBox: {
            box: null,
            monitor(index, visibleChange) {
                this.box = document.getElementsByTagName("input")[0];
                this.box.placeholder = "";
                index > 3 &&
                    unsafeWindow.addEventListener(
                        "popstate",
                        (e) => {
                            e.preventDefault();
                            e.stopPropagation();
                        },
                        true
                    );
                unsafeWindow.addEventListener(
                    "visibilitychange",
                    (e) => {
                        e.preventDefault();
                        this.box.placeholder = "";
                        e.stopPropagation();
                        index < 4 && visibleChange(document.hidden);
                    },
                    true
                );
                let button = document.getElementsByClassName(
                    "Button SearchBar-searchButton Button--primary"
                );
                if (button.length > 0) {
                    button[0].onclick = (e) => {
                        if (this.box.value.length === 0) {
                            e.preventDefault();
                            e.stopPropagation();
                        }
                    };
                }
                button = null;
                this.box.addEventListener(
                    "keydown",
                    (fuckzhihu) => {
                        if (fuckzhihu.keyCode !== 13) return;
                        if (
                            this.box.value.length === 0 ||
                            this.box.value.trim().length === 0
                        ) {
                            fuckzhihu.preventDefault();
                            fuckzhihu.stopImmediatePropagation();
                            fuckzhihu.stopPropagation();
                        } else {
                            if (
                                blackKey.some((e) => this.box.value.includes(e))
                            ) {
                                Notification(
                                    "keyword contains rubbish word",
                                    "Warning"
                                );
                                return;
                            }
                            const url = `https://www.zhihu.com/search?q=${this.box.value}&type=content`;
                            window.open(url, "_blank");
                        }
                    },
                    true
                );
                this.box.onfocus = () => {
                    this.box.value.length === 0 && (this.box.placeholder = "");
                    localStorage.setItem("zap:SharedSession", "");
                };
                this.box.onblur = () => (this.box.placeholder = "");
                this.firstRun();
            },
            firstRun() {
                let mo = new MutationObserver((e) => {
                    if (e.length !== 1 || e[0].addedNodes.length !== 1) return;
                    const target = e[0].addedNodes[0];
                    const p = target.getElementsByClassName("Popover-content");
                    if (p.length === 0) return;
                    const tmp =
                        p[0].getElementsByClassName("AutoComplete-group");
                    if (tmp.length === 0) return;
                    this.AutoComplete = tmp[0];
                    if (p[0].innerText.startsWith("搜索发现"))
                        this.AutoComplete.style.display = "none";
                    mo.disconnect();
                    mo = null;
                    this.secondRun(p[0].parentNode);
                });
                mo.observe(document.body, { childList: true });
            },
            AutoComplete: null,
            secondRun(target) {
                const mo = new MutationObserver((e) => {
                    if (e.length === 1) {
                        if (e[0].addedNodes.length !== 1) {
                            this.AutoComplete = null;
                            return;
                        }
                        const t = e[0].addedNodes[0];
                        this.AutoComplete =
                            t.getElementsByClassName("AutoComplete-group")[0];
                        if (t.innerText.startsWith("搜索发现"))
                            this.AutoComplete.style.display = "none";
                    } else {
                        const style =
                            this.box.value.length > 0 ? "inline" : "none";
                        this.AutoComplete.style.display !== style &&
                            (this.AutoComplete.style.display = style);
                    }
                });
                mo.observe(target, { childList: true, subtree: true });
            },
        },
        ErrorAutoClose() {
            const w = document.getElementsByClassName("PostIndex-warning");
            if (w.length === 0) return;
            const h = document.createElement("h2");
            w[0].insertAdjacentElement("afterbegin", h);
            setTimeout(() => window.close(), 6100);
            let time = 5;
            const dot = ".";
            h.innerText = `5s, current web will be automatically closed.....`;
            let id = setInterval(() => {
                time--;
                h.innerText = `${time}s, current web will be automatically closed${dot.repeat(
                    time
                )}`;
                if (time === 0) clearInterval(id);
            }, 1000);
        },
        zhuanlanStyle(mode) {
            //font, the pic of header, main content, sidebar, main content letter spacing, comment zone, ..
            //@media print, print preview, make the background-color can view when save webpage as pdf file
            const article = `
                mark.AssistantMark.red{background-color: rgba(255, 128, 128, 0.65) !important;box-shadow: rgb(255, 128, 128) 0px 1.2px;border-radius: 0.2em !important;}
                mark.AssistantMark.yellow{background-color: rgba(255, 250, 90, 1) !important;box-shadow: rgb(255, 255, 170) 0px 1.2px;border-radius: 0.2em !important;}
                mark.AssistantMark.green{background-color: rgba(170, 235, 140, 0.8) !important;box-shadow: rgb(170, 255, 170) 0px 2.2px;border-radius: 0.2em !important;}
                mark.AssistantMark.purple{background-color: rgba(255, 170, 255, 0.8) !important;box-shadow: rgb(255, 170, 255) 0px 1.2px;border-radius: 0.2em !important;}
                @media print {
                    mark.AssistantMark { box-shadow: unset !important; -webkit-print-color-adjust: exact !important; }
                    .CornerButtons,
                    div#load_status,
                    .toc-bar.toc-bar--collapsed,
                    div#assist-button-container {display : none;}
                    #column_lists {display : none !important;}
                }
                body{text-shadow: #a9a9a9 0.025em 0.015em 0.02em;}
                .TitleImage{width: 500px !important}
                .Post-Main .Post-RichText{text-align: justify !important;}
                .Post-SideActions{left: calc(50vw - 560px) !important;}
                .RichText.ztext.Post-RichText{letter-spacing: 0.1px;}
                .Sticky.RichContent-actions.is-fixed.is-bottom{position: inherit !important}
                .Comments-container,
                .Post-RichTextContainer{width: 900px !important;}
                .highlight{
                    box-shadow: rgb(170, 170, 170) 0px 0px 4px 0px;
                    box-sizing: border-box !important;
                    overflow-wrap: break-word !important;
                }
                ${
                    mode === 0 && GM_getValue("topnopicture")
                        ? ".TitleImage,"
                        : ""
                }
                a[href*="u.jd.com"],
                .RichText-MCNLinkCardContainer,
                .Post-RichTextContainer .Catalog.isCatalogV2,
                span.LinkCard-content.LinkCard-ecommerceLoadingCard,
                .css-1ynzxqw,
                .RichText-MCNLinkCardContainer{display: none !important}`;
            const list = `.Card:nth-of-type(3),.Card:last-child,.css-8txec3{width: 900px !important;}`;
            const home = `
                .RichContent.is-collapsed{cursor: default !important;}
                .List-item {
                    margin-top: 8px;
                    position: relative;
                    padding: 16px 20px;
                    width: 1000px;
                    margin-left: 23%;
                    box-shadow: 0 1px 3px rgba(18,18,18,.1);
                    border: 1px solid #B9D5FF;
                }
                .ColumnHomeTop{
                    position: relative !important;
                    height: -webkit-fill-available !important;
                }
                .css-1hwwfws{display: none !important;}`;
            if (mode < 2) {
                if (mode === 0) {
                    if (document.title.startsWith("该内容暂无法显示")) {
                        window.onload = () => this.ErrorAutoClose();
                        return;
                    }
                    const r = GM_getValue("reader");
                    if (r) {
                        GM_addStyle(
                            article + this.Column.clearPage(0).join("")
                        );
                        this.Column.readerMode = true;
                    } else GM_addStyle(article);
                    this.Column.isZhuanlan = true;
                } else {
                    document.title = "IGNORANCE IS STRENGTH";
                    Object.defineProperty(document, "title", {
                        writable: true,
                        enumerable: true,
                        configurable: true,
                    });
                    this.Column.is_column_home = true;
                    GM_addStyle(home);
                }
                window.onload = () => {
                    if (mode === 0) {
                        this.colorAssistant.main();
                        this.Column.main(0);
                        this.autoScroll.keyBoardEvent();
                        unsafeWindow.addEventListener("visibilitychange", () =>
                            this.visibleChange(document.hidden)
                        );
                        control_pannel_init();
                    } else this.column_homePage(this.Column.main(1));
                    setTimeout(() => this.show_Total.main(true), 30000);
                    this.key_ctrl_sync(true);
                };
            } else {
                GM_addStyle(list);
                this.Column.main(2);
            }
        },
        Column: {
            isZhuanlan: false,
            is_column_home: false,
            authorID: "",
            authorName: "",
            get ColumnDetail() {
                const header = document.getElementsByClassName(
                    "ColumnLink ColumnPageHeader-TitleColumn"
                );
                if (header.length === 0) {
                    this.columnName = "";
                    this.columnID = "";
                    return false;
                }
                const href = header[0].href;
                this.columnID = href.slice(href.lastIndexOf("/") + 1);
                this.columnName = header[0].innerText;
                const post = document.getElementsByClassName("Post-Author");
                if (post.length > 0) {
                    const user =
                        post[0].getElementsByClassName("UserLink-link");
                    const i = user.length - 1;
                    const p = user[i].pathname;
                    this.authorName = user[i].innerText;
                    this.authorID = p.slice(p.lastIndexOf("/") + 1);
                }
                return true;
            },
            updateAuthor(author) {
                const html = `
                <div
                    class="AuthorInfo"
                    itemprop="author"
                    itemscope=""
                    itemtype="http://schema.org/Person"
                >
                    <meta itemprop="name" content=${author.name} /><meta
                        itemprop="image"
                        content=${author.avatar_url}
                    /><meta
                        itemprop="url"
                        content=https://www.zhihu.com/people/${author.url_token}
                    /><meta itemprop="zhihu:followerCount" content="3287" /><span
                        class="UserLink AuthorInfo-avatarWrapper"
                        ><div class="Popover">
                            <div
                                id="Popover8-toggle"
                                aria-haspopup="true"
                                aria-expanded="false"
                                aria-owns="Popover8-content"
                            >
                                <a
                                    class="UserLink-link"
                                    data-za-detail-view-element_name="User"
                                    target="_blank"
                                    href="//www.zhihu.com/people/${author.url_token}"
                                    ><img
                                        class="Avatar Avatar--round AuthorInfo-avatar"
                                        width="38"
                                        height="38"
                                        src=${author.avatar_url}
                                        srcset=${author.avatar_url}
                                        alt=${author.name}
                                /></a>
                            </div></div
                    ></span>
                    <div class="AuthorInfo-content">
                        <div class="AuthorInfo-head">
                            <span class="UserLink AuthorInfo-name"
                                ><div class="Popover">
                                    <div
                                        id="Popover9-toggle"
                                        aria-haspopup="true"
                                        aria-expanded="false"
                                        aria-owns="Popover9-content"
                                    >
                                        <a
                                            class="UserLink-link"
                                            data-za-detail-view-element_name="User"
                                            target="_blank"
                                            href="//www.zhihu.com/people/${author.url_token}"
                                            >${author.name}</a
                                        >
                                    </div>
                                </div></span
                            >
                        </div>
                        <div class="AuthorInfo-detail">
                            <div class="AuthorInfo-badge">
                                <div class="ztext AuthorInfo-badgeText">
                                    ${author.headline}
                                </div>
                            </div>
                        </div>
                    </div>
                </div>`;
                const authorNode =
                    document.getElementsByClassName("AuthorInfo");
                if (authorNode.length > 0) authorNode[0].outerHTML = html;
                this.authorID = author.url_token;
                this.authorName = author.name;
            },
            //column homepage
            subscribeOrfollow() {
                if (!this.ColumnDetail) return;
                let fn = "follow";
                let sn = "subscribe";
                const f = GM_getValue(fn);
                if (f && Array.isArray(f))
                    f.some((e) => this.columnID === e.columnID) &&
                        (fn = "remove");
                const s = GM_getValue(sn);
                if (s && Array.isArray(s))
                    s.some((e) => this.columnID === e.columnID) &&
                        (sn = "remove");
                let [a, b] =
                    fn === "remove" ? ["remove", "from"] : ["add", "to"];
                const ft = `${a}&nbsp;the&nbsp;column&nbsp;${b}&nbsp;follow&nbsp;list`;
                [a, b] = sn === "remove" ? ["remove", "from"] : ["add", "to"];
                const st = `${a}&nbsp;the&nbsp;column&nbsp;${b}&nbsp;subscribe&nbsp;list`;
                const html = `
                <div class="assistant-button" style="margin-left: 15px">
                    <style type="text/css">
                        .assistant-button button {
                            box-shadow: 1px 1px 2px #848484;
                            height: 24px;
                            border: 1px solid #ccc !important;
                            border-radius: 8px;
                        }
                    </style>
                    <button class="follow" style="width: 80px; margin-right: 5px;color: #2196F3;" title=${ft}>
                        ${fn}
                    </button>
                    <button class="subscribe" style="width: 90px" title=${st}>${sn}</button>
                </div>`;
                //const bhtml = `<button class="block" style="width: 70px" title="block the column">block</button>`;
                const user = document.getElementsByClassName(
                    "AuthorInfo AuthorInfo--plain"
                );
                if (user.length === 0) return;
                user[0].parentNode.insertAdjacentHTML("beforeend", html);
                let buttons =
                    document.getElementsByClassName("assistant-button")[0]
                        .children;
                const exe = (button, mode) => {
                    const name = button.innerText;
                    if (name === "remove") {
                        const vname = mode === 1 ? "follow" : "subscribe";
                        const arr = GM_getValue(vname);
                        if (arr && Array.isArray(arr)) {
                            const index = arr.findIndex(
                                (e) => e.columnID === this.columnID
                            );
                            if (index > -1) {
                                arr.splice(index, 1);
                                GM_setValue(vname, arr);
                                if (mode === 1 && this.columnsModule.node)
                                    this.columnsModule.database = arr;
                            }
                            Notification(`un${vname} successfully`, "Tips");
                        }
                        button.innerText = vname;
                        button.title = `add the column to ${vname} list`;
                    } else {
                        if (mode === 1) {
                            const i = this.follow(true);
                            if (i !== 1) {
                                button.innerText = "remove";
                                button.title =
                                    "remove the column from follow list";
                            }
                        } else {
                            this.subscribe();
                            button.innerText = "remove";
                            button.title =
                                "remove the column from subscribe list";
                        }
                    }
                };
                buttons[1].onclick = function () {
                    exe(this, 1);
                };
                buttons[2].onclick = function () {
                    exe(this, 2);
                };
                /*
                buttons[3].onclick =function () {
                    exe(this, 3);
                }
                */
                buttons = null;
            },
            //shift + f
            follow(mode) {
                if (!this.columnID) return;
                let f = GM_getValue("follow");
                if (f && Array.isArray(f)) {
                    let index = 0;
                    for (const e of f) {
                        if (this.columnID === e.columnID) {
                            const c = confirm(
                                `you have already followed this column on ${this.timeStampconvertor(
                                    e.update
                                )}, is unfollow this column?`
                            );
                            if (!c) return 0;
                            f.splice(index, 1);
                            GM_setValue("follow", f);
                            Notification(
                                "unfollow this column successfully",
                                "Tips"
                            );
                            return 1;
                        }
                        index++;
                    }
                } else f = [];
                const p = prompt(
                    "please input some tags about this column, like: javascript python; multiple tags use blank space to isolate",
                    "javascript python"
                );
                let tags = [];
                if (p && p.trim()) {
                    const tmp = p.split(" ");
                    for (let e of tmp) {
                        e = e.trim();
                        e && tags.push(e);
                    }
                }
                if (tags.length === 0 && !mode) {
                    const top = document.getElementsByClassName(
                        "TopicList Post-Topics"
                    );
                    if (top.length > 0) {
                        const topic = top[0].children;
                        for (const e of topic) tags.push(e.innerText);
                    }
                }
                const info = {};
                info.columnID = this.columnID;
                info.update = Date.now();
                info.columnName = this.columnName;
                info.tags = tags;
                f.push(info);
                this.columnsModule.node && (this.columnsModule.database = f);
                GM_setValue("follow", f);
                Notification(
                    "you have followed this column successfully",
                    "Tips",
                    3500
                );
                return 2;
            },
            //shift + s
            subscribe() {
                if (!this.columnID) return;
                let s = GM_getValue("subscribe");
                if (s && Array.isArray(s)) {
                    let i = 0;
                    for (const e of s) {
                        if (e.columnID === this.columnID) {
                            s.splice(i, 1);
                            break;
                        }
                    }
                } else s = [];
                const i = s.length;
                const info = {};
                info.columnID = this.columnID;
                info.update = Date.now();
                info.columnName = this.columnName;
                if (i === 0) {
                    s.push(info);
                } else {
                    i === 10 && s.pop();
                    s.unshift(info);
                }
                GM_setValue("subscribe", s);
                Notification(
                    "you have subscribed this column successfully",
                    "Tips",
                    3500
                );
            },
            Tabs: {
                get GUID() {
                    // blob:https://xxx.com/+ uuid
                    const link = URL.createObjectURL(new Blob());
                    const blob = link.toString();
                    URL.revokeObjectURL(link);
                    return blob.substr(blob.lastIndexOf("/") + 1);
                },
                save(columnID) {
                    //if currentb window does't close, when reflesh page or open new url in current window(how to detect the change ?)
                    //if open new url in same tab, how to change the uuid?
                    GM_getTab((tab) => {
                        const uuid = this.GUID;
                        tab.id = uuid;
                        tab.columnID = columnID;
                        tab.title = document.title;
                        sessionStorage.setItem("uuid", uuid);
                        GM_saveTab(tab);
                    });
                },
                check(columnID) {
                    return new Promise((resolve) => {
                        GM_getTabs((tabs) => {
                            if (tabs) {
                                //when open a new tab with "_blank" method, this tab will carry the session data of origin tab
                                const uuid = sessionStorage.getItem("uuid");
                                if (!uuid) {
                                    resolve(false);
                                } else {
                                    const tablist = Object.values(tabs);
                                    const title = document.title;
                                    const f = tablist.some(
                                        (e) =>
                                            e.columnID === columnID &&
                                            uuid === e.id &&
                                            e.titlle !== title
                                    );
                                    resolve(f);
                                }
                            } else resolve(false);
                        });
                    });
                },
            },
            tocMenu: {
                change: false,
                appendNode(toc) {
                    if (toc.className.endsWith("collapsed")) return;
                    const header =
                        document.getElementsByClassName("Post-Header");
                    if (header.length === 0) {
                        console.log("the header has been remove");
                        return;
                    }
                    header[0].appendChild(toc);
                    toc.style.position = "sticky";
                    toc.style.width = "900px";
                    this.change = true;
                },
                restoreNode(toc) {
                    if (!this.change) return;
                    document.body.append(toc);
                    toc.removeAttribute("style");
                    this.change = false;
                },
                main(mode) {
                    const toc = document.getElementById("toc-bar");
                    toc &&
                        (mode ? this.restoreNode(toc) : this.appendNode(toc));
                },
            },
            titleChange: false,
            clearPage(mode = 0) {
                const ids = [
                    "Post-Sub Post-NormalSub",
                    "Post-Author",
                    "span.Voters button",
                    "ColumnPageHeader-Wrapper",
                    "Post-SideActions",
                    "Sticky RichContent-actions is-bottom",
                ];
                if (mode === 0) {
                    const reg = /\s/g;
                    const css = ids.map(
                        (e) =>
                            `${
                                e.startsWith("span")
                                    ? e
                                    : `.${e.replace(reg, ".")}`
                            }{display: none;}`
                    );
                    return css;
                } else {
                    const style = mode === 1 ? "block" : "none";
                    ids.forEach((e) => {
                        const tmp = e.startsWith("span");
                        tmp &&
                            (e = e.slice(e.indexOf(".") + 1, e.indexOf(" ")));
                        const t = document.getElementsByClassName(e);
                        t.length > 0 &&
                            (tmp
                                ? (t[0].firstChild.style.display = style)
                                : (t[0].style.display = style));
                    });
                }
            },
            // load all lazy images in 'print' mode
            load_all_lazy_img() {
                const content = document.getElementsByClassName(
                    "RichText ztext Post-RichText"
                );
                if (content.length === 0) return;
                const imgs = content[0].querySelectorAll("img.lazy");
                for (const img of imgs) {
                    const src = img.src;
                    if (
                        src &&
                        src.startsWith(
                            "data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg'"
                        )
                    ) {
                        const org =
                            img.dataset["original"] || img.dataset["actualsrc"];
                        if (org) {
                            const i = org.lastIndexOf("_");
                            img.src =
                                i < 0
                                    ? org
                                    : org.slice(0, i + 1) +
                                      "720w" +
                                      org.slice(org.lastIndexOf("."));
                        }
                    }
                }
            },
            titleAlign() {
                if (this.modePrint && !this.titleChange) return;
                const title = document.getElementsByClassName("Post-Title");
                if (title.length === 0) return;
                if (this.modePrint) {
                    title[0].removeAttribute("style");
                } else {
                    if (title[0].innerText.length > 28) return;
                    title[0].style.textAlign = "center";
                }
                this.titleChange = !this.titleChange;
            },
            modePrint: false,
            time_print_id: null,
            pagePrint(status) {
                // wait the the load of lazy img
                Notification(
                    `${this.modePrint ? "exit" : "enter"} print mode`,
                    "Print",
                    3500
                );
                !this.readerMode && this.clearPage(this.modePrint ? 1 : 2);
                this.tocMenu.main(this.modePrint);
                this.titleAlign();
                if (!this.modePrint) {
                    this.load_all_lazy_img();
                    this.time_print_id && clearTimeout(this.time_print_id);
                    this.time_print_id = setTimeout(() => {
                        this.time_print_id = null;
                        window.print();
                    }, 350);
                }
                this.modePrint = !this.modePrint;
                this.modePrint ? status.create("Print Mode") : status.remove();
            },
            Framework() {
                const html = `
                <div
                    id="column_lists"
                    style="
                        top: 54px;
                        width: 380px;
                        font-size: 14px;
                        box-sizing: border-box;
                        padding: 0 10px 10px 0;
                        box-shadow: 0 1px 3px #ddd;
                        border-radius: 4px;
                        transition: width 0.2s ease;
                        color: #333;
                        background: #fefefe;
                        -moz-user-select: none;
                        -webkit-user-select: none;
                        user-select: none;
                        display: flex;
                        z-index: 1;
                        position: absolute;
                        left: 2%;
                    "
                >
                    <style type="text/css">
                        button.button {
                            margin: 15px 0px 5px 0px;
                            width: 60px;
                            height: 24px;
                            border-radius: 3px;
                            box-shadow: 1px 2px 5px #888888;
                        }
                        div#column_lists .list.num {
                            color: #fff;
                            width: 18px;
                            height: 18px;
                            text-align: center;
                            line-height: 18px;
                            background: #fff;
                            border-radius: 2px;
                            display: inline-block;
                            background: #00a1d6;
                        }
                        div#column_lists ul a:hover{
                            color: blue;
                        }
                        div#column_lists ul {
                            overflow: hidden;
                            text-overflow: ellipsis;
                            white-space: nowrap;
                            display: block;
                            line-height: 1.9;
                        }
                        div#column_lists .header{
                            font-weight: bold;
                            font-size: 16px;
                        }
                    </style>
                    <span
                        class="right_column"
                        style="margin-left: 5%; margin-top: 5px; width: 100%"
                    >
                        <span class="header current column">
                            <a
                                class="column name"
                                href= https://www.zhihu.com/column/${
                                    this.columnID
                                }
                                target="_blank"
                                title=${
                                    this.isZhuanlan
                                        ? ""
                                        : escapeBlank("random column")
                                }
                                >${this.columnName}</a
                            >
                            <span class="tips" style="
                                float: right;
                                font-size: 14px;
                                font-weight: normal;
                            "></span>
                            <hr style="width: 340px" />
                        </span>
                        <ul
                            class="article_lists"
                        >
                        </ul>
                        <div class="nav button">
                            <button class="button last" title="previous page">Pre</button>
                            <button class="button next" title="next page">Next</button>
                            <button class="button hide" title="hide the menu">Hide</button>
                            <button class="button more" title="show more content">More</button>
                            <select class="select-pages" size="1" name="pageslist" style="margin-left: 20px; margin-top: 16px; height: 24px; position: absolute; width: 60px;box-shadow: 1px 2px 5px #888888;">
                                <option value="0" selected>pages</option>
                            </select>
                        </div>
                    </span>
                </div>`;
                document.body.insertAdjacentHTML("beforeend", html);
            },
            timeStampconvertor(timestamp) {
                if (!timestamp) return "undefined";
                if (typeof timestamp === "number") {
                    const s = timestamp.toString();
                    if (s.length === 10) timestamp *= 1000;
                    else if (s.length !== 13) return "undefined";
                } else {
                    if (timestamp.length === 10) {
                        timestamp = parseInt(timestamp);
                        timestamp *= 1000;
                    } else if (timestamp.length === 13)
                        timestamp = parseInt(timestamp);
                    else return "";
                }
                const date = new Date(timestamp);
                const y = date.getFullYear() + "-";
                const gm = date.getMonth();
                const m = (gm + 1 < 10 ? "0" + (gm + 1) : gm + 1) + "-";
                let d = date.getDate();
                d = d < 10 ? "0" + d : d;
                return y + m + d;
            },
            backupInfo: null,
            next: null,
            previous: null,
            index: 1,
            rqReady: false,
            requestData(url) {
                if (this.rqReady) {
                    Notification("please request data slowly...", "Tips");
                    return;
                }
                this.rqReady = true;
                xmlHTTPRequest(url).then(
                    (json) => {
                        typeof json === "string" && (json = JSON.parse(json));
                        const data = json.data;
                        let id = this.isReverse ? data.length : 1;
                        const html = [];
                        this.backupInfo = [];
                        const tips =
                            "click&nbsp;me,&nbsp;show&nbsp;the&nbsp;content&nbsp;in&nbsp;current&nbsp;webpage";
                        const HREF = location.href;
                        for (const e of data) {
                            const type = e.type;
                            if (type !== "article" && type !== "answer")
                                continue;
                            const info = {};
                            info.id = id;
                            let time = e.updated;
                            let title = e.title;
                            let className = '"list_date"';
                            let question = "";
                            info.url = e.url;
                            const tmp = {};
                            let fontWeight = "";
                            if (!time) {
                                time = e.updated_time;
                                className = "list_date_question";
                                title = e.question.title;
                                question =
                                    "this&nbsp;is&nbsp;a&nbsp;question&nbsp;page,&nbsp;do&nbsp;not&nbsp;show&nbsp;in&nbsp;current&nbsp;page";
                                info.url = `https://www.zhihu.com/question/${e.question.id}/answer/${e.id}`;
                            } else {
                                HREF === info.url &&
                                    ((fontWeight =
                                        ' style="font-weight:bold;"'),
                                    (this.targetIndex = id));
                                tmp.author = e.author;
                            }
                            info.excerpt = escapeHTML(
                                `${title} <摘要>: ` + e.excerpt
                            );
                            info.updated = this.timeStampconvertor(time);
                            info.ctitle = escapeHTML(title);
                            title = titleSlice(title);
                            title = escapeHTML(title);
                            info.title = title;
                            html.push(
                                this.liTagRaw(
                                    info,
                                    question || tips,
                                    className,
                                    fontWeight
                                )
                            );
                            if (!question) {
                                tmp.content = e.content;
                                tmp.title = e.title;
                            }
                            this.backupInfo.push(tmp);
                            this.isReverse ? id-- : id++;
                        }
                        if (this.isReverse) {
                            this.backupInfo.reverse();
                            html.reverse();
                        }
                        const pag = json.paging;
                        const totals = pag.totals;
                        this.appendNode(html, totals);
                        this.previous = this.isReverse
                            ? pag.is_end
                                ? ""
                                : pag.next
                            : pag.is_start
                            ? ""
                            : pag.previous;
                        this.next = this.isReverse
                            ? pag.is_start
                                ? ""
                                : pag.previous
                            : pag.is_end
                            ? ""
                            : pag.next;
                        this.rqReady = false;
                    },
                    (err) => {
                        console.log(err);
                        this.rqReady = false;
                    }
                );
            },
            firstAdd: true,
            selectRaw(index, name) {
                return `<option value=${index}>${name}</option>`;
            },
            total_pages: 0,
            isReverse: false,
            //pages list
            appendSelect(node, pages, mode = false) {
                const select = node.getElementsByClassName("select-pages");
                if (select.length === 0) return;
                this.total_pages = Math.ceil(pages / 10);
                const end = this.total_pages > 30 ? 31 : this.total_pages + 1;
                const html = [];
                for (let index = 1; index < end; index++)
                    html.push(this.selectRaw(index, index));
                if (this.total_pages > 1)
                    html.push(this.selectRaw(html.length + 1, "reverse"));
                select[0].insertAdjacentHTML("beforeend", html.join(""));
                const optionNum = select[0].length + 1;
                const [ds, dh] =
                    optionNum < 9
                        ? [optionNum, 15 * optionNum + "px"]
                        : [8, "120px"];
                select[0].onmousedown = function () {
                    this.size = ds;
                    this.style.height = dh;
                };
                select[0].onblur = function () {
                    this.style.height = "24px";
                    this.size = 1;
                };
                //execute
                const exe = (opt) => {
                    if (this.total_pages === 1) return;
                    const i = opt.value * 1;
                    const tmp = i === this.total_pages + 1;
                    if (tmp) {
                        if (this.isReverse) return;
                        this.isReverse = tmp;
                        this.index = 1;
                        opt.value = 1;
                    } else {
                        if (i === 0) {
                            if (this.isReverse) {
                                this.isReverse = false;
                                this.index = 1;
                                opt.value = 1;
                            } else {
                                return;
                            }
                        } else this.index = i;
                    }
                    if (mode) {
                        Reflect.apply(this.homePage.add, this, [
                            this.homePage.initial,
                            "f",
                            true,
                        ]);
                    } else {
                        const m = this.isReverse
                            ? this.total_pages - this.index
                            : this.index - 1;
                        const URL = `http://www.zhihu.com/api/v4/columns/${
                            this.columnID
                        }/items?limit=10&offset=${10 * m}`;
                        this.requestData(URL);
                    }
                };
                select[0].onchange = function () {
                    this.size = 1;
                    this.style.height = "24px";
                    exe(this);
                };
            },
            changeSelect(mode) {
                this.index += mode ? 1 : -1;
                const column = document.getElementById("column_lists");
                const select = column.getElementsByTagName("select")[0];
                select.value = this.index;
            },
            appendNode(html, totals, mode = false) {
                const column = document.getElementById("column_lists");
                if (!column) {
                    console.log("the module of column has been deleted");
                    return;
                }
                const lists = column.getElementsByClassName("article_lists")[0];
                this.firstAdd
                    ? (lists.insertAdjacentHTML("afterbegin", html.join("")),
                      this.appendSelect(column, totals, mode),
                      this.clickEvent(column, mode))
                    : (lists.innerHTML = html.join(""));
                this.firstAdd = false;
            },
            liTagRaw(info, title, className = "list_date", fontWeight = "") {
                const html = `
                <li${fontWeight}>
                    <span class="list num">${info.id}</span>
                    <a
                        href=${info.url}
                        target="_blank"
                        ctitle=${info.ctitle}
                        title=${info.excerpt}>${info.title}</a
                    >
                    <span class=${className} style="float: right" title=${title}>${info.updated}</span>
                </li>`;
                return html;
            },
            nextPage: false,
            tipsTimeout: null,
            showTips(tips) {
                const column = document.getElementById("column_lists");
                if (!column) return;
                const tipNode = column.getElementsByClassName("tips")[0];
                tipNode.innerText = tips;
                this.tipsTimeout && clearTimeout(this.tipsTimeout);
                this.tipsTimeout = setTimeout(() => {
                    tipNode.innerText = "";
                    this.tipsTimeout = null;
                }, 2000);
            },
            /*
            need add new function => simple mode and content mode, if it is simple mode, all page direct show the menu of column?
            */
            followCursor: null,
            homePage: {
                follow: null,
                get ColumnID() {
                    const i = Math.floor(Math.random() * this.follow.length);
                    return this.follow[i].columnID;
                },
                get initial() {
                    this.follow = GM_getValue("follow");
                    if (
                        !this.follow ||
                        !Array.isArray(this.follow) ||
                        this.follow.length === 0
                    )
                        return false;
                    return this.follow;
                },
                add(follow, direction, page) {
                    if (!follow) return;
                    setTimeout(() => {
                        const html = [];
                        const k = follow.length;
                        if (this.isReverse) {
                            this.followCursor = (this.index - 1) * 10 - 1;
                        } else {
                            if (page) {
                                this.followCursor = k - (this.index - 1) * 10;
                            } else {
                                if (this.followCursor === null) {
                                    this.followCursor = k;
                                } else {
                                    if (direction === "r")
                                        this.followCursor += 20;
                                    if (this.followCursor > k)
                                        this.followCursor = k;
                                }
                            }
                        }
                        const className = "list_date_follow";
                        const title = "follow&nbsp;date";
                        let id = 1;
                        const prefix = "https://www.zhihu.com/column/";
                        const methods = {
                            r() {
                                this.followCursor += 1;
                                return this.followCursor < k;
                            },
                            f() {
                                this.followCursor -= 1;
                                return this.followCursor > -1;
                            },
                        };
                        const func = this.isReverse ? "r" : "f";
                        while (methods[func].call(this)) {
                            const e = follow[this.followCursor];
                            const info = {};
                            info.ctitle = e.columnName;
                            info.title = escapeHTML(e.columnName);
                            info.updated = this.timeStampconvertor(e.update);
                            info.excerpt = e.tags.join(";&nbsp;");
                            info.url = prefix + e.columnID;
                            info.id = id;
                            html.push(this.liTagRaw(info, title, className));
                            id++;
                            if (id > 10) break;
                        }
                        this.appendNode(html, k, true);
                        this.previous = this.isReverse
                            ? this.index - 1
                            : !(this.followCursor + id - 1 === k);
                        this.next = this.isReverse
                            ? this.index === Math.ceil(k / 10)
                                ? 0
                                : 1
                            : this.followCursor > -1;
                        this.homePage.follow = null;
                    }, 0);
                },
            },
            home_Module: {
                loaded_list: null,
                current_Column_id: null,
                home_request: {
                    pre: null,
                    next: null,
                    is_loaded: true,
                    index: 0,
                    firstly: true,
                    request(url, node) {
                        return new Promise((resolve, reject) => {
                            this.is_loaded = false;
                            xmlHTTPRequest(url).then(
                                (json) => {
                                    const data = json.data;
                                    const arr = [];
                                    for (const d of data) {
                                        const type = d.type;
                                        if (
                                            !(
                                                type === "article" ||
                                                type === "answer"
                                            )
                                        )
                                            continue;
                                        const info = {};
                                        info.url =
                                            d.url ||
                                            "https://www.zhihu.com/question/" +
                                                d.question.id +
                                                "/answer/" +
                                                d.id;
                                        info.title =
                                            d.title || d.question.title;
                                        arr.push(
                                            column_Home.item_Raw(
                                                d,
                                                this.index,
                                                info
                                            )
                                        );
                                        this.index += 1;
                                        column_Home.item_index += 1;
                                    }
                                    this.firstly
                                        ? (node.innerHTML = arr.join(""))
                                        : node.insertAdjacentHTML(
                                              "afterbegin",
                                              arr.join("")
                                          );
                                    this.firstly = false;
                                    const p = json.paging;
                                    this.pre = p.is_start ? null : p.previous;
                                    this.next = p.is_end ? null : p.next;
                                    this.is_loaded = true;
                                    resolve(true);
                                },
                                (err) => {
                                    console.log(err);
                                    this.is_loaded = true;
                                    reject(null);
                                }
                            );
                        });
                    },
                },
                home_nextButton() {
                    if (
                        !this.home_request.is_loaded ||
                        Reflect.get(zhihu.autoScroll, "scrollState")
                    )
                        return;
                    if (!this.home_request.next) {
                        Notification("no more data", "Tips");
                        return;
                    }
                    this.home_request
                        .request(this.home_request.next, this.Node)
                        .then(() => zhihu.scroll.toTop());
                },
                home_DB_initial() {
                    !this.loaded_list && (this.loaded_list = []);
                    this.loaded_qlist = [];
                    dataBaseInstance.initial(["collection"], true).then(
                        () => {},
                        () =>
                            Notification(
                                "database initialization failed",
                                "Warning"
                            )
                    );
                },
                is_create_button: false,
                create_home_button() {
                    createButton("Next", "", "", "right");
                    setTimeout(() => {
                        let button = document.getElementById(
                            "assist-button-container"
                        );
                        button.onclick = () => this.home_nextButton();
                        button = null;
                    }, 0);
                    this.is_create_button = true;
                },
                get Node() {
                    return document.getElementsByClassName("ColumnHomeTop")[0];
                },
                get_full_title(a) {
                    const attributes = a.attributes;
                    for (const e of attributes)
                        if (e.name === "ctitle") return e.value;
                    return "";
                },
                loaded_qlist: null,
                li_set_blod(t) {
                    this.cancel_li_bold(t);
                    t.parentNode.style.fontWeight = "bold";
                },
                get_article_list(t) {
                    let p = t.parentNode;
                    let cn = p.className;
                    if (cn === "article_lists") return p;
                    else {
                        p = p.parentNode;
                        cn = p.className;
                        let ic = 0;
                        while (cn !== "article_lists") {
                            p = p.previousElementSibling;
                            if (!p || ic > 4) return null;
                            cn = p.className;
                            ic++;
                        }
                        return p;
                    }
                },
                cancel_li_bold(t) {
                    const p = this.get_article_list(t.parentNode);
                    if (!p) return;
                    const cs = p.children;
                    for (const c of cs) {
                        if (c.style.fontWeight === "bold") {
                            c.style.fontWeight = "normal";
                            break;
                        }
                    }
                },
                home_click(href, target, mode) {
                    if (!this.home_request.is_loaded) return;
                    const id = href.slice(href.lastIndexOf("/") + 1);
                    const pos = ["answer", "zhuanlan", "column", "question"];
                    const index = pos.findIndex((e) => href.includes(e));
                    if (index === 2) {
                        if (this.current_Column_id === id) return;
                        this.home_request.pre = null;
                        this.home_request.next = null;
                        this.home_request.firstly = true;
                        this.current_Column_id = id;
                        const url = `https://www.zhihu.com/api/v4/columns/${id}/items?limit=5&offset=0`;
                        this.home_request.request(url, this.Node);
                        !this.is_create_button && this.create_home_button();
                        column_Home.item_index = 0;
                        this.home_request.index = 0;
                        this.loaded_list.length = 0;
                        this.loaded_qlist.length = 0;
                        mode
                            ? this.li_set_blod(target)
                            : this.cancel_li_bold(target);
                    } else if (index < 2) {
                        if (this.loaded_list.includes(id)) return;
                        this.current_article_id = id;
                        this.current_Column_id = null;
                        const info = {};
                        info.title = this.get_full_title(target);
                        info.url = href;
                        this.loaded_list.push(id);
                        column_Home.single_Content_request(
                            index === 0 ? 0 : 2,
                            id,
                            this.Node,
                            info
                        );
                    } else {
                        if (this.loaded_qlist.includes(id)) return;
                        this.loaded_qlist.push(id);
                        const index = collect_Answers.findIndex(
                            (e) => e.qid === id
                        );
                        if (index > -1) {
                            const c = collect_Answers[index];
                            const title = c.title;
                            const data = c.data;
                            for (const d of data) {
                                if (this.loaded_list.includes(d.aid)) continue;
                                this.loaded_list.push(d.aid);
                                const info = {};
                                info.title = title;
                                info.url =
                                    "https://www.zhihu.com/question/" +
                                    id +
                                    "/answer/" +
                                    d.aid;
                                column_Home.single_Content_request(
                                    0,
                                    d.aid,
                                    this.Node,
                                    info
                                );
                            }
                        }
                    }
                },
            },
            targetIndex: 0,
            clickEvent(node, mode = false) {
                let buttons =
                    node.getElementsByClassName("nav button")[0].children;
                let article = node.getElementsByTagName("ul")[0];
                let aid = 0;
                //prevent click too fast
                let isReady = false;
                //show content in current page;
                let article_time_id = null;
                article.onclick = (e) => {
                    //if under autoscroll mode, => not allow to click
                    article_time_id && clearTimeout(article_time_id);
                    article_time_id = setTimeout(() => {
                        article_time_id = null;
                        if (
                            Reflect.get(zhihu.autoScroll, "scrollState") ||
                            Reflect.get(zhihu.noteHighlight, "editable")
                        )
                            return;
                        if (isReady) {
                            Notification("please operate slowly...", "Tips");
                            return;
                        }
                        const t = e.target;
                        const href = t.previousElementSibling.href;
                        if (location.href === href) return;
                        const className = t.className;
                        if (className === "list_date_follow") {
                            if (this.is_column_home) {
                                this.home_Module.home_click(
                                    href,
                                    t.previousElementSibling,
                                    true
                                );
                                return;
                            }
                            sessionStorage.clear();
                            window.open(href, "_self");
                        } else if (className !== "list_date") return;
                        const content = document.getElementsByClassName(
                            "RichText ztext Post-RichText"
                        );
                        if (content.length === 0) return;
                        const p = e.path;
                        let ic = 0;
                        for (const e of p) {
                            if (e.localName === "li") {
                                let id = e.children[0].innerText;
                                id *= 1;
                                if (id === aid) return;
                                aid = id;
                                break;
                            }
                            if (ic > 2) return;
                            ic++;
                        }
                        isReady = true;
                        const i = aid - 1;
                        const title =
                            document.getElementsByClassName("Post-Title");
                        title.length > 0 &&
                            (title[0].innerText = this.backupInfo[i].title);
                        content[0].innerHTML = this.backupInfo[i].content;
                        zhihu.colorAssistant.main();
                        window.history.replaceState(null, null, href);
                        document.title = `${this.backupInfo[i].title} - 知乎`;
                        //refresh the menu
                        const toc = document.getElementById("toc-bar");
                        if (toc) {
                            const refresh = toc.getElementsByClassName(
                                "toc-bar__refresh toc-bar__icon-btn"
                            );
                            refresh.length > 0 && refresh[0].click();
                        }
                        const author = this.backupInfo[i].author;
                        if (author && this.authorID !== author.url_token)
                            this.updateAuthor(author);
                        let pnode = e.target.parentNode;
                        let j = 0;
                        while (pnode.localName !== "li") {
                            pnode = pnode.parentNode;
                            j++;
                            if (j > 2) break;
                        }
                        j < 3 && (pnode.style.fontWeight = "bold");
                        if (this.targetIndex > 0) {
                            pnode.parentNode.children[
                                this.targetIndex - 1
                            ].style.fontWeight = "normal";
                        }
                        this.targetIndex = aid;
                        this.reInject();
                        isReady = false;
                        const links = content[0].getElementsByTagName("a");
                        for (const link of links) {
                            const href = decodeURIComponent(link.href).split(
                                "link.zhihu.com/?target="
                            );
                            if (href.length > 1) link.href = href[1];
                        }
                    }, 300);
                };
                article = null;
                //last page
                let isCollapsed = false;
                let change_time_id = null;
                buttons[0].onclick = () => {
                    change_time_id && clearTimeout(change_time_id);
                    change_time_id = setTimeout(() => {
                        change_time_id = null;
                        !isCollapsed &&
                            (this.previous
                                ? (mode
                                      ? Reflect.apply(this.homePage.add, this, [
                                            this.homePage.initial,
                                            "r",
                                            false,
                                        ])
                                      : this.requestData(this.previous),
                                  (aid = 0),
                                  this.changeSelect(false))
                                : this.showTips("no more content"));
                    }, 300);
                };
                //next page
                let change_time_id_a;
                buttons[1].onclick = () => {
                    change_time_id_a && clearTimeout(change_time_id_a);
                    change_time_id_a = setTimeout(() => {
                        change_time_id_a = null;
                        !isCollapsed &&
                            (this.next
                                ? (mode
                                      ? Reflect.apply(this.homePage.add, this, [
                                            this.homePage.initial,
                                            "f",
                                            false,
                                        ])
                                      : this.requestData(this.next),
                                  (aid = 0),
                                  this.changeSelect(true))
                                : this.showTips("no more content"));
                    }, 300);
                };
                //hide the sidebar
                buttons[2].onclick = function () {
                    const [style, text, title] = isCollapsed
                        ? ["block", "Hide", "hide the menu"]
                        : ["none", "Expand", "show the menu"];
                    this.parentNode.parentNode.children[1].style.display =
                        style;
                    const more = this.parentNode.nextElementSibling;
                    if (more) {
                        more.style.display = style;
                        more.nextElementSibling.style.display = style;
                    }
                    this.innerText = text;
                    this.title = title;
                    isCollapsed = !isCollapsed;
                };
                let addnew = true;
                const createModule = (button) => {
                    const sub = GM_getValue("subscribe");
                    if (addnew) {
                        this.columnsModule.liTagRaw = this.liTagRaw;
                        this.columnsModule.isZhuanlan = this.isZhuanlan;
                        this.columnsModule.is_column_home = this.is_column_home;
                        this.columnsModule.timeStampconvertor =
                            this.timeStampconvertor;
                        if (this.is_column_home)
                            this.columnsModule.home = this.home_Module;
                    }
                    let html = null;
                    let text = "";
                    if (sub && Array.isArray(sub)) {
                        let id = 1;
                        const prefix = "https://www.zhihu.com/column/";
                        const title = "subscribe&nbsp;time";
                        html = sub.map((e) => {
                            const info = {};
                            info.id = id;
                            info.url = prefix + e.columnID;
                            info.updated = this.timeStampconvertor(e.update);
                            info.title = e.columnName;
                            info.ctitle = escapeHTML(e.columnName);
                            info.excerpt = "";
                            id++;
                            return this.liTagRaw(info, title);
                        });
                        text = "Subscribe";
                    } else text = "no more data";
                    addnew
                        ? this.columnsModule.main(
                              button,
                              html,
                              text,
                              escapeBlank(
                                  this.isZhuanlan
                                      ? "column/article search"
                                      : this.is_column_home
                                      ? "column/article/answer search"
                                      : "column/article search"
                              )
                          )
                        : this.columnsModule.appendNewNode(html, text);
                    addnew = false;
                };
                buttons[3].onclick = function () {
                    !isCollapsed && createModule(this);
                };
                //if webpage is column home, expand this menu
                mode && createModule(buttons[3]);
                buttons = null;
            },
            columnsModule: {
                recentModule: {
                    log(type, config) {
                        //history, pocket, recent collect: h, c, p
                        const href = (config && config.url) || location.href;
                        let r = GM_getValue("recent");
                        if (r && Array.isArray(r)) {
                            const index = r.findIndex((e) => e.url === href);
                            if (index > -1) {
                                if (index === 0 && r[index].type === type)
                                    return;
                                r.splice(index, 1);
                            }
                        } else r = [];
                        let info = {};
                        if (config) info = config;
                        else {
                            info.title = get_Title();
                            info.update = Date.now();
                            info.type = type;
                            info.url = href;
                        }
                        const i = r.length;
                        if (i === 0) {
                            r.push(info);
                        } else {
                            if (i === 10) r.pop();
                            r.unshift(info);
                        }
                        GM_setValue("recent", r);
                        type === "p" &&
                            Notification(
                                "add current article to read later successfully",
                                "Tips"
                            );
                    },
                    remove(type, url) {
                        const href = url || location.href;
                        const r = GM_getValue("recent");
                        if (r && Array.isArray(r)) {
                            const index = r.findIndex((e) => e.url === href);
                            if (index > -1 && r[index].type === type) {
                                r.splice(index, 1);
                                GM_setValue("recent", r);
                            }
                        }
                    },
                    addnew: true,
                    read(node, liTagRaw, timeStampconvertor, home) {
                        const r = GM_getValue("recent");
                        if (!r || !Array.isArray(r) || r.length === 0) return;
                        const html = r.map((e) => {
                            const info = {};
                            const type = e.type;
                            info.updated = timeStampconvertor(e.update);
                            info.id = type;
                            info.excerpt = "";
                            info.url = e.url;
                            info.ctitle = escapeHTML(e.title);
                            info.title = titleSlice(e.title);
                            const title =
                                "recent&nbsp;" +
                                (type === "h"
                                    ? "read&nbsp;history"
                                    : type === "c"
                                    ? "collection"
                                    : "pocket");
                            return liTagRaw(info, title);
                        });
                        let ul = node.getElementsByTagName("ul")[0];
                        const pre = ul.previousElementSibling;
                        pre.style.display =
                            html.length === 0 ? "block" : "none";
                        ul.innerHTML = html.join("");
                        if (this.addnew) {
                            ul.onclick = (e) => {
                                if (e.target.className === "list_date") {
                                    if (
                                        Reflect.get(
                                            zhihu.autoScroll,
                                            "scrollState"
                                        )
                                    )
                                        return;
                                    const a =
                                        e.target.previousElementSibling.href;
                                    if (home) {
                                        home.home_click(
                                            a,
                                            e.target.previousElementSibling
                                        );
                                        return;
                                    }
                                    location.href !== a &&
                                        (sessionStorage.clear(),
                                        window.open(a, "_self"));
                                }
                            };
                        }
                        ul = null;
                    },
                    main(node, liTagRaw, timeStampconvertor, home) {
                        const n = node.nextElementSibling;
                        this.read(n, liTagRaw, timeStampconvertor, home);
                        let button =
                            n.getElementsByClassName("button refresh")[0];
                        button.onclick = () =>
                            this.read(n, liTagRaw, timeStampconvertor, home);
                        button = null;
                    },
                },
                database: null,
                node: null,
                liTagRaw: null,
                home: null,
                addNewModule(text, placeholder) {
                    const html = `
                        <div class="more columns">
                            <hr>
                            <div class="search module" style="margin-bottom: 10px">
                                <input
                                    type="text"
                                    placeholder=${placeholder}
                                    style="height: 24px; width: 250px"
                                />
                                <button class="button search" style="margin-left: 11px;">Search</button>
                            </div>
                            <hr>
                            <span class="header columns">${text}</span>
                            <ul class="columns list">
                            </ul>
                        </div>
                        <div class="recently-activity">
                            <hr>
                            <span class="header columns"
                                >Recent
                                <button
                                    class="button refresh"
                                    style="margin-top: 5px; margin-left: 210px; font-weight: normal"
                                    title="refresh recent content"
                                >
                                    refresh
                                </button>
                            </span>
                            <hr>
                            <span class="header columns">no recent data</span>
                            <ul style="height: 120px; overflow: auto;"></ul>
                        </div>`;
                    const column = document.getElementById("column_lists");
                    if (column)
                        column.children[1].insertAdjacentHTML(
                            "beforeend",
                            html
                        );
                },
                appendNewNode(html, text = "", node) {
                    let pnode = node || this.node;
                    pnode = pnode.parentNode.nextElementSibling;
                    const ul = pnode.getElementsByTagName("ul")[0];
                    ul.innerHTML = html.join("");
                    text && (pnode.children[3].innerText = text);
                },
                checkInlcudes(e, key) {
                    if (e.columnName.includes(key)) return true;
                    return e.tags.some((t) =>
                        key.length > t.length
                            ? key.includes(t)
                            : t.includes(key)
                    );
                },
                timeStampconvertor: null,
                commandFormat(str) {
                    const treg = /(?<=\$)[dmhyw][<>=][0-9]+/g;
                    const areg = /(?<=\$)a=\(.+\)/g;
                    const preg = /(?<=\$)p=[0-9]{5,}/g;
                    const t = str.match(treg);
                    const p = str.match(preg);
                    if (t && p) return null;
                    const a = str.match(areg);
                    if (p && a) return null;
                    if (!(a || p || t)) return null;
                    const sigs = ["=", ">", "<"];
                    const sign = {
                        0: "equal",
                        1: "great",
                        2: "less",
                    };
                    const type = {};
                    if (t) {
                        let cm = "";
                        let ecount = 0;
                        let lcount = 0;
                        let gcount = 0;
                        for (const e of t) {
                            if (cm && cm !== e[0]) return null;
                            if (!type[e[0]]) type[e[0]] = {};
                            if (type[e[0]]) {
                                const sg = e[1];
                                const index = sigs.findIndex((e) => e === sg);
                                if (index === 0) {
                                    ecount++;
                                    if (ecount > 1) return null;
                                } else if (index === 1) {
                                    gcount++;
                                    if (gcount > 1) return null;
                                } else {
                                    lcount++;
                                    if (lcount > 1) return null;
                                }
                                const n = sign[index];
                                if (type[e[0]][n]) return null;
                                type[e[0]][n] = e.slice(2);
                            }
                        }
                        if (a) {
                            if (a.length > 1) return null;
                            const tmp = a[0];
                            type[tmp[0]] = tmp
                                .slice(3, tmp.length - 1)
                                .split(" ");
                        }
                    } else if (p) {
                        if (p.length > 1) return null;
                        const tmp = p[0];
                        type[tmp[0]] = tmp.slice(2);
                    } else if (a) {
                        if (a.length > 1) return null;
                        const tmp = a[0];
                        type[tmp[0]] = tmp.slice(3, tmp.length - 1).split(" ");
                    }
                    return type;
                },
                //search box query command execute
                ExecuteFunc: {
                    Resultshow(e, i, liTagRaw, timeStampconvertor) {
                        const info = {};
                        info.id = i;
                        info.ctitle = escapeHTML(e.title);
                        info.title = titleSlice(e.title);
                        info.excerpt = escapeHTML(e.excerpt);
                        info.updated = timeStampconvertor(e.update);
                        const title = escapeBlank("collect time");
                        const prefix = "/p/";
                        info.url = prefix + e.pid;
                        return liTagRaw(info, title);
                    },
                    less(a, b) {
                        return a < b;
                    },
                    equal(a, b) {
                        return a === b;
                    },
                    great(a, b) {
                        return a > b;
                    },
                    get now() {
                        return Date.now();
                    },
                    get nowDate() {
                        return new Date();
                    },
                    getUpdate(update) {
                        return new Date(update);
                    },
                    w(value, mode, info) {
                        const now = this.now;
                        const time = info.update;
                        const week = (now - time) / (86400000 * 7);
                        return this[mode](week, value * 1);
                    },
                    d(value, mode, info) {
                        const now = this.now;
                        const time = info.update;
                        const day = (now - time) / 86400000;
                        return this[mode](day, value * 1);
                    },
                    m(value, mode, info) {
                        const endDate = this.nowDate;
                        const beginDate = this.getUpdate(info.update);
                        const month =
                            endDate.getFullYear() * 12 +
                            endDate.getMonth() -
                            (beginDate.getFullYear() * 12 +
                                beginDate.getMonth());
                        return this[mode](month, value * 1);
                    },
                    y(value, mode, info) {
                        const endDate = this.nowDate;
                        const beginDate = this.getUpdate(info.update);
                        const year =
                            endDate.getFullYear() * 12 +
                            endDate.getMonth() -
                            (beginDate.getFullYear() * 12 +
                                beginDate.getMonth()) /
                                12;
                        return this[mode](year, value * 1);
                    },
                    h(value, mode, info) {
                        const now = this.now;
                        const time = info.update;
                        const hour = (now - time) / (1000 * 60 * 60);
                        return this[mode](hour, value * 1);
                    },
                    a(value, info) {
                        const content =
                            info.title +
                            " | " +
                            info.tags.join("") +
                            info.excerpt;
                        return value.some((v) => content.includes(v));
                    },
                    search(table, fs, liTagRaw, timeStampconvertor) {
                        return new Promise((resolve, reject) => {
                            let ic = 0;
                            const html = [];
                            const cur = table.openCursor(null, "next");
                            cur.onsuccess = (e) => {
                                const cursor = e.target.result;
                                if (cursor) {
                                    const info = cursor.value;
                                    const result = fs.every((f) => f(info));
                                    if (result) {
                                        ic++;
                                        info.ctitle = escapeHTML(info.title);
                                        info.title = titleSlice(info.title);
                                        html.push(
                                            this.Resultshow(
                                                info,
                                                ic,
                                                liTagRaw,
                                                timeStampconvertor
                                            )
                                        );
                                    }
                                    if (ic === 10) {
                                        resolve(html);
                                    } else {
                                        cursor.continue();
                                    }
                                } else {
                                    resolve(html);
                                }
                            };
                            cur.onerror = (e) => {
                                console.log(e);
                                reject("open db cursor fail");
                            };
                        });
                    },
                    funcs(
                        type,
                        liTagRaw,
                        timeStampconvertor,
                        node,
                        appendNewNode
                    ) {
                        const fs = [];
                        const keys = Object.keys(type);
                        for (const f of keys) {
                            const tmp = type[f];
                            if (Array.isArray(tmp) || typeof tmp !== "object") {
                                if (f === "p") {
                                    dataBaseInstance.check(tmp).then(
                                        (r) => {
                                            const html = [];
                                            if (r) {
                                                r.ctitle = escapeHTML(r.title);
                                                r.title = titleSlice(r.title);
                                                html.push(
                                                    this.Resultshow(
                                                        r,
                                                        1,
                                                        liTagRaw,
                                                        timeStampconvertor
                                                    )
                                                );
                                            }
                                            appendNewNode(
                                                html,
                                                html.length === 0
                                                    ? "no search result"
                                                    : "article search results",
                                                node
                                            );
                                        },
                                        (err) => console.log(err)
                                    );
                                    return;
                                }
                                fs.push(this[f].bind(this, tmp));
                                continue;
                            }
                            const tk = Object.keys(tmp);
                            for (const e of tk)
                                fs.push(this[f].bind(this, tmp[e], e));
                        }
                        return fs;
                    },
                    main(
                        table,
                        type,
                        liTagRaw,
                        timeStampconvertor,
                        node,
                        appendNewNode
                    ) {
                        const fs = this.funcs(
                            type,
                            liTagRaw,
                            timeStampconvertor,
                            node,
                            appendNewNode
                        );
                        if (!fs || fs.length === 0) return;
                        this.search(
                            table,
                            fs,
                            liTagRaw,
                            timeStampconvertor
                        ).then(
                            (html) => {
                                appendNewNode(
                                    html,
                                    html.length === 0
                                        ? "no search result"
                                        : "article search results",
                                    node
                                );
                            },
                            (err) => console.log(err)
                        );
                    },
                },
                initialDatabase(type) {
                    if (!type) return;
                    dataBaseInstance.TableName = "collection";
                    this.ExecuteFunc.main(
                        dataBaseInstance.Table,
                        type,
                        this.liTagRaw,
                        this.timeStampconvertor,
                        this.node,
                        this.appendNewNode
                    );
                },
                _answer_check(title, arr) {
                    return arr.some((e) =>
                        title.length > e.length
                            ? title.includes(e)
                            : e.includes(title)
                    );
                },
                search_Anwsers(key) {
                    const html = [];
                    if (
                        collect_Answers.length === 0 ||
                        !(key[1] === "=" || key[1] === " ")
                    )
                        return html;
                    key = key.slice(2).trim();
                    if (!key) return html;
                    const tmp = key.split(" ");
                    let i = 0;
                    const title = escapeBlank("last active time");
                    const pref = "https://www.zhihu.com/question/";
                    for (const e of collect_Answers) {
                        if (this._answer_check(e.title, tmp)) {
                            i++;
                            const info = {};
                            info.id = i;
                            info.ctitle = e.title;
                            info.title = titleSlice(e.title);
                            info.excerpt = e.data.length;
                            info.updated = this.timeStampconvertor(
                                e.data[0].update
                            );
                            info.url = pref + e.qid;
                            html.push(this.liTagRaw(info, title));
                            if (i === 10) break;
                        }
                    }
                    return html;
                },
                isZhuanlan: false,
                is_column_home: false,
                searchDatabase(key) {
                    if (
                        (key.charAt(0) === "$" && this.isZhuanlan) ||
                        (this.is_column_home && key.length > 2)
                    ) {
                        const cm = ["a", "p", "d", "m", "y", "h", "w", "q"];
                        const f = key.charAt(1).toLowerCase();
                        const index = cm.indexOf(f);
                        if (index === cm.length - 1) {
                            if (!collect_Answers)
                                collect_Answers = GM_getValue("collect_a");
                            if (
                                !(
                                    collect_Answers &&
                                    Array.isArray(collect_Answers)
                                )
                            )
                                collect_Answers = [];
                            const html = this.search_Anwsers(key);
                            this.appendNewNode(
                                html,
                                html.length === 0
                                    ? "no search result"
                                    : "answer search results"
                            );
                            return true;
                        } else if (index > -1) {
                            this.initialDatabase(this.commandFormat(key));
                            return true;
                        }
                    }
                    return false;
                },
                search(key) {
                    //search follow columns & collection of article
                    if (this.searchDatabase(key)) return;
                    let i = 0;
                    const html = [];
                    const prefix = "https://www.zhihu.com/column/";
                    const title = "follow&nbsp;time";
                    for (const e of this.database) {
                        if (this.checkInlcudes(e, key)) {
                            i++;
                            const info = {};
                            info.id = i;
                            info.ctitle = e.columnName;
                            info.title = titleSlice(e.columnName);
                            info.excerpt = e.tags.join(";&nbsp;");
                            info.updated = this.timeStampconvertor(e.update);
                            info.url = prefix + e.columnID;
                            html.push(this.liTagRaw(info, title));
                            if (i === 10) break;
                        }
                    }
                    this.appendNewNode(
                        html,
                        html.length === 0
                            ? "no search result"
                            : "column search results"
                    );
                },
                event() {
                    const p = this.node.parentNode.nextElementSibling;
                    const input = p.getElementsByTagName("input")[0];
                    input.onkeydown = (e) => {
                        if (e.keyCode !== 13) return;
                        const key = input.value.trim();
                        key.length > 1 && this.search(key);
                    };
                    let button = p.getElementsByClassName("button search")[0];
                    button.onclick = () => {
                        const key = input.value.trim();
                        key.length > 1 && this.search(key);
                    };
                    button = null;
                    let ul = p.getElementsByTagName("ul")[0];
                    ul.onclick = (e) => {
                        if (e.target.className === "list_date") {
                            const a = e.target.previousElementSibling.href;
                            if (this.home) {
                                this.home.home_click(
                                    a,
                                    e.target.previousElementSibling
                                );
                                return;
                            }
                            location.href !== a && window.open(a, "_self");
                        }
                    };
                    ul = null;
                    this.recentModule.main(
                        p,
                        this.liTagRaw,
                        this.timeStampconvertor,
                        this.home
                    );
                },
                main(node, html, text, placeholder) {
                    this.node = node;
                    this.addNewModule(text, placeholder);
                    html && this.appendNewNode(html);
                    this.database = GM_getValue("follow");
                    if (!this.database || !Array.isArray(this.database))
                        this.database = [];
                    this.event();
                    GM_addValueChangeListener(
                        "follow",
                        (name, oldValue, newValue, remote) =>
                            remote && (this.database = newValue)
                    );
                },
            },
            assistNewModule: {
                preferenceModule(ainfo, binfo) {
                    const html = `
                    <div class="article_attitude" style="margin-left: 23px;">
                        <i
                            class="like"
                            style="
                                display: ${ainfo.ldisplay};
                                width: 26px;
                                content: url();
                                height: 26px;
                                margin-bottom: 5px;
                            "
                            title=${ainfo.ltitle}
                        ></i>
                        <i
                            class="dislike"
                            style="
                                content: url();
                                display: ${ainfo.ddisplay};
                                width: 24px;
                                height: 24px;
                            "
                            title=${ainfo.dtitle}
                        ></i>
                    </div>
                    <button class="assist-button collect" style="color: black;" title=${binfo.title}>${binfo.name}</button>`;
                    return html;
                },
                create(module) {
                    const node = document.getElementById(
                        "assist-button-container"
                    );
                    node.children[0].insertAdjacentHTML("afterend", module);
                    return node;
                },
                //open db with r&w mode, if this db is not exist then create db
                main() {
                    return new Promise((resolve) => {
                        const tables = ["collection", "preference"];
                        dataBaseInstance.initial(tables, true).then(
                            (result) => {
                                const ainfo = {};
                                ainfo.ltitle = "like this article";
                                ainfo.ldisplay = "block";
                                ainfo.ddisplay = "block";
                                ainfo.dtitle = "dislike this article";
                                const binfo = {};
                                binfo.title = "add this article to collection";
                                binfo.name = "Collect";
                                if (result === 0) {
                                    resolve(
                                        this.create(
                                            this.preferenceModule(
                                                escapeBlank(ainfo),
                                                escapeBlank(binfo)
                                            )
                                        )
                                    );
                                } else {
                                    dataBaseInstance
                                        .batchCheck(tables)
                                        .then((results) => {
                                            const c = results[0];
                                            if (c) {
                                                binfo.title =
                                                    "remove this acticle from collection list";
                                                binfo.name = "Remove";
                                            }
                                            if (results[1]) {
                                                const pref = results[1].value;
                                                if (pref === 1) {
                                                    ainfo.ltitle =
                                                        "cancel like this article";
                                                    ainfo.ddisplay = "none";
                                                } else if (pref === 0) {
                                                    ainfo.dtitle =
                                                        "cancel dislike this article";
                                                    ainfo.ldisplay = "none";
                                                }
                                            }
                                            resolve(
                                                this.create(
                                                    this.preferenceModule(
                                                        escapeBlank(ainfo),
                                                        escapeBlank(binfo)
                                                    )
                                                )
                                            );
                                        });
                                }
                            },
                            (err) => {
                                console.log(err);
                                resolve(null);
                            }
                        );
                    });
                },
            },
            syncData(mode, newValue) {
                dataBaseInstance.TableName = "preference";
                mode
                    ? dataBaseInstance.update(newValue[0])
                    : dataBaseInstance.dele(false, newValue[0]);
            },
            communication() {
                const monitor = () => {
                    GM_addValueChangeListener(
                        "blockarticleB",
                        (name, oldValue, newValue, remote) => {
                            if (remote) {
                                this.syncData(true, newValue);
                                GM_setValue("blockarticleB", "");
                            }
                        }
                    );
                    GM_addValueChangeListener(
                        "removearticleB",
                        (name, oldValue, newValue, remote) => {
                            if (remote) {
                                this.syncData(false, newValue);
                                GM_setValue("removearticleB", "");
                            }
                        }
                    );
                };
                const r = GM_getValue("removearticleB");
                const b = GM_getValue("blockarticleB");
                if (r || b) {
                    dataBaseInstance.TableName = "preference";
                    if (r && Array.isArray(r) && r.length > 0) {
                        for (const e of r) dataBaseInstance.dele(false, e);
                        GM_setValue("removearticleB", "");
                    }
                    if (b && Array.isArray(b) && b.length > 0) {
                        for (const e of b) dataBaseInstance.update(e);
                        GM_setValue("blockarticleB", "");
                    }
                    monitor();
                } else monitor();
            },
            creatAssistantEvent(node, mode) {
                const Button = (button, m) => {
                    if (m) {
                        if (!this.readerMode) return;
                        this.createFrame();
                        button.style.display = "none";
                        mode = false;
                    } else {
                        let text = "";
                        let style = "";
                        const column = document.getElementById("column_lists");
                        let i = 0;
                        let title = "";
                        if (this.readerMode) {
                            text = "Reader";
                            style = "none";
                            title = "enter";
                            if (column) column.style.display = style;
                            i = 1;
                            //show content;
                        } else {
                            style = "block";
                            text = "Exit";
                            title = "exit";
                            if (column) {
                                column.style.display = style;
                            } else {
                                //hide content
                                this.Tabs.check(this.columnID).then(
                                    (result) => !result && this.createFrame()
                                );
                            }
                            i = 2;
                        }
                        !this.modePrint && this.clearPage(i);
                        button.title = `${title} the reader mode`;
                        button.innerText = text;
                        this.readerMode = !this.readerMode;
                        GM_setValue("reader", this.readerMode);
                        mode &&
                            (button.previousElementSibling.style.display =
                                style);
                    }
                };
                let i = node.children.length;
                //reder
                node.children[--i].onclick = function () {
                    Button(this, false);
                };
                //menu
                if (mode) {
                    node.children[--i].onclick = function () {
                        Button(this, true);
                    };
                }
                //collection
                let cReady = false;
                const collectionClick = (button) => {
                    if (cReady) return;
                    cReady = true;
                    dataBaseInstance.TableName = "collection";
                    const text = button.innerText;
                    let s = "",
                        t = "";
                    if (text === "Remove") {
                        s = "Collect";
                        t = "add this article to collection list";
                        dataBaseInstance.dele(true);
                        this.columnsModule.recentModule.remove("c");
                    } else {
                        s = "Remove";
                        t = "remove this article frome collection list";
                        dataBaseInstance.additem(this.columnID);
                        this.columnsModule.recentModule.log("c");
                    }
                    button.innerText = s;
                    button.title = t;
                    cReady = false;
                };
                node.children[--i].onclick = function () {
                    collectionClick(this);
                };
                //pref
                const cm = (mode) => {
                    const bid = location.pathname.slice(3);
                    if (mode) {
                        let b = GM_getValue("blockarticleA");
                        if (b && Array.isArray(b)) {
                            for (const e of b) if (e.name === bid) return;
                        } else b = [];
                        const r = GM_getValue("removearticleA");
                        if (r && Array.isArray(r)) {
                            const i = r.indexOf(bid);
                            if (i > -1) {
                                r.splice(i, 1);
                                GM_setValue("removearticleA", r);
                            }
                        }
                        const info = {};
                        info.name = bid;
                        info.type = "article";
                        info.from = this.columnID;
                        info.update = Date.now();
                        info.userID = this.authorID;
                        info.userName = this.authorName;
                        b.push(info);
                        GM_setValue("blockarticleA", b);
                    } else {
                        let r = GM_getValue("removearticleA");
                        if (r && Array.isArray(r) && r.length > 0) {
                            if (r.includes(bid)) return;
                        } else r = [];
                        const b = GM_getValue("blockarticleA");
                        if (b && Array.isArray(b)) {
                            const i = b.findIndex((e) => e.name === bid);
                            if (i > -1) {
                                b.splice(i, 1);
                                GM_setValue("blockarticleA", b);
                            }
                        }
                        r.push(bid);
                        GM_setValue("removearticleA", r);
                    }
                };
                let pReady = false;
                const prefclick = (button, other, mode) => {
                    if (pReady) return;
                    pReady = true;
                    dataBaseInstance.TableName = "preference";
                    let title = "";
                    let f = false;
                    let cl = false;
                    if (other.style.display === "none") {
                        dataBaseInstance.dele(false);
                        title = `${mode ? "like" : "dislike"} this article`;
                        other.style.display = "block";
                        cl = mode;
                    } else {
                        const info = {};
                        info.pid = dataBaseInstance.pid;
                        info.update = Date.now();
                        info.userID = this.authorID;
                        info.from = this.columnID;
                        info.value = mode ? 1 : 0;
                        dataBaseInstance.update(info);
                        title = `cancel ${
                            mode ? "like" : "dislike"
                        } this article`;
                        other.style.display = "none";
                        f = !mode;
                    }
                    button.title = title;
                    !cl && cm(f);
                    pReady = false;
                };
                node.children[--i].onclick = (e) => {
                    const target = e.target;
                    const className = target.className;
                    if (className === "like")
                        prefclick(target, target.nextElementSibling, true);
                    else if (className === "dislike")
                        prefclick(target, target.previousElementSibling, false);
                };
            },
            recordID: null,
            dataRecord() {
                this.recordID && clearTimeout(this.recordID);
                this.recordID = setTimeout(() => {
                    this.recordID = null;
                    const pid = dataBaseInstance.pid;
                    dataBaseInstance.TableName = "collection";
                    dataBaseInstance.updateRecord(pid);
                    if (this.columnID) {
                        const f = GM_getValue("follow");
                        if (f && Array.isArray(f) && f.length > 0) {
                            for (const e of f) {
                                if (e.columnID === this.columnID) {
                                    e.visitTime = Date.now();
                                    let times = e.visitTimes;
                                    e.visitTimes = times ? ++times : 2;
                                    GM_setValue("follow", f);
                                    break;
                                }
                            }
                        }
                    }
                }, 15000);
            },
            reInject() {
                let assist = document.getElementById("assist-button-container");
                if (assist) {
                    assist.remove();
                    assist = null;
                }
                this.injectButton();
            },
            injectButton(mode) {
                const name = this.readerMode ? "Exit" : "Reader";
                const mbutton =
                    '<button class="assist-button siderbar" style="color: black;" title="show the siderbar">Menu</button>';
                createButton(
                    name,
                    (this.readerMode ? "exit " : "enter ") + "the reader mode",
                    mode ? mbutton : "",
                    "right"
                );
                this.assistNewModule
                    .main()
                    .then(
                        (node) => node && this.creatAssistantEvent(node, mode)
                    );
                this.dataRecord();
            },
            readerMode: false,
            createFrame() {
                if (this.columnID) {
                    this.Framework();
                    //const pinURL = `https://www.zhihu.com/api/v4/columns/${this.columnID}/pinned-items`;
                    const url = `https://www.zhihu.com/api/v4/columns/${this.columnID}/items`;
                    this.requestData(url);
                    this.Tabs.save(this.columnID);
                }
            },
            columnID: null,
            columnName: null,
            no_Article(mode) {
                const f = this.homePage.initial;
                if (f) {
                    this.columnID = this.homePage.ColumnID;
                    this.columnName = "Follow";
                    this.Framework();
                    Reflect.apply(this.homePage.add, this, [f, "f", false]);
                    mode === 1 && this.home_Module.home_DB_initial();
                } else
                    colorful_Console.main(
                        {
                            title: "warning",
                            content: "sidebar initiation failed",
                        },
                        colorful_Console.colors.warning
                    );
                return f;
            },
            main(mode = 0) {
                if (mode > 0) {
                    let f = false;
                    mode === 2
                        ? (window.onload = () => (
                              this.no_Article(mode), this.subscribeOrfollow()
                          ))
                        : (f = this.no_Article(mode));
                    return f;
                }
                if (this.ColumnDetail) {
                    if (this.readerMode)
                        this.Tabs.check(this.columnID).then((result) =>
                            result
                                ? this.injectButton(true)
                                : (this.createFrame(), this.injectButton())
                        );
                    else this.injectButton();
                } else this.injectButton();
                setTimeout(() => this.communication(), 5000);
            },
        },
        colorAssistant: {
            index: 0,
            arr: null,
            blue: true,
            grad: 5,
            get rgbRed() {
                this.index -= this.grad;
                if (this.index < 0 || this.index > 255) {
                    this.index = 0;
                    this.blue = true;
                    this.grad < 0 && (this.grad *= -1);
                }
                const s =
                    this.index > 233
                        ? 5
                        : this.index > 182
                        ? 4
                        : this.index > 120
                        ? 3
                        : this.index > 88
                        ? 2
                        : this.index > 35
                        ? 1
                        : 0;
                return `rgb(${this.index}, ${s}, 0)`;
            },
            redc: false,
            get rgbBlue() {
                this.index += this.grad;
                if (this.index > 255) {
                    this.blue = false;
                    if (this.redc) {
                        this.index = 0;
                        this.grad *= -1;
                        this.redc = false;
                    } else {
                        this.grad < 0 && (this.grad *= -1);
                        this.index = 255;
                        this.redc = true;
                    }
                }
                return `rgb(0, 0, ${this.index})`;
            },
            get textColor() {
                return this.blue ? this.rgbBlue : this.rgbRed;
            },
            setColor(text, color) {
                return `<colorspan class="color-node" style="color: ${color} !important;">${text}</colorspan>`;
            },
            setcolorGrad(tlength) {
                // according the length of string to set the color's gap
                this.grad =
                    tlength > 500
                        ? 1
                        : tlength > 300
                        ? 2
                        : tlength > 180
                        ? 3
                        : tlength > 120
                        ? 4
                        : tlength > 80
                        ? 5
                        : 6;
            },
            num: 0,
            textDetach(text) {
                // detach the text to single charactor
                this.setcolorGrad(text.length);
                const reg = /((\d+[\.-\/]\d+([\.-\/]\d+)?)|\d{2,}|[a-z]{2,})/gi;
                let result = null;
                let start = 0;
                let end = 0;
                let tmp = "";
                result = reg.exec(text);
                const numaColors = ["green", "#8B008B"];
                if (result) {
                    while (result) {
                        tmp = result[0];
                        end = reg.lastIndex - tmp.length;
                        for (start; start < end; start++)
                            this.arr.push(
                                this.setColor(text[start], this.textColor)
                            );
                        start = reg.lastIndex;
                        this.arr.push(this.setColor(tmp, numaColors[this.num]));
                        this.num = this.num ^ 1;
                        result = reg.exec(text);
                    }
                    end = text.length;
                    for (start; start < end; start++)
                        this.arr.push(
                            this.setColor(text[start], this.textColor)
                        );
                } else {
                    for (let t of text)
                        this.arr.push(this.setColor(t, this.textColor));
                }
            },
            nodeCount: 0,
            getItem(node) {
                // those tags will be ignored
                const localName = node.localName;
                const tags = [
                    "a",
                    "br",
                    "b",
                    "span",
                    "code",
                    "strong",
                    "u",
                    "sup",
                    "br",
                ];
                if (localName && tags.includes(localName)) {
                    this.arr.push(node.outerHTML);
                    this.nodeCount += 1;
                    return;
                } else {
                    const className = node.className;
                    if (className && className === "UserLink") {
                        this.arr.push(node.outerHTML);
                        this.nodeCount += 1;
                        return;
                    }
                }
                if (node.childNodes.length === 0) {
                    const text = node.nodeValue;
                    text && this.textDetach(text);
                } else {
                    // this is a trick, no traversal of textnode, maybe some nodes will lost content, take care
                    for (const item of node.childNodes) this.getItem(item);
                    this.arr.length > 0 &&
                        node.childNodes.length - this.nodeCount <
                            this.nodeCount + 2 &&
                        (node.innerHTML = this.arr.join(""));
                    this.arr = [];
                }
            },
            resetColor() {
                this.blue = !this.blue;
                this.index = this.blue ? 0 : 255;
            },
            codeHighlight(node) {
                const keywords = [
                    "abstract",
                    "arguments",
                    "await",
                    "boolean",
                    "break",
                    "byte",
                    "case",
                    "catch",
                    "char",
                    "class",
                    "const",
                    "continue",
                    "debugger",
                    "default",
                    "delete",
                    "do",
                    "double",
                    "else",
                    "enum",
                    "eval",
                    "export",
                    "extends",
                    "false",
                    "final",
                    "finally",
                    "float",
                    "for",
                    "function",
                    "goto",
                    "if",
                    "implements",
                    "import",
                    "in",
                    "instanceof",
                    "int",
                    "interface",
                    "let",
                    "long",
                    "native",
                    "new",
                    "null",
                    "package",
                    "private",
                    "protected",
                    "public",
                    "return",
                    "short",
                    "static",
                    "super",
                    "switch",
                    "synchronized",
                    "this",
                    "throw",
                    "throws",
                    "transient",
                    "true",
                    "try",
                    "typeof",
                    "var",
                    "void",
                    "volatile",
                    "while",
                    "with",
                    "yield",
                ];
                node.title = "Ctrl + Right mouse button to copy this code";
                const code = node.getElementsByClassName("language-text");
                if (code.length === 0 || code[0].childNodes.length > 1) return;
                let html = code[0].innerHTML;
                const reg = /(["'])(.+?)(["'])/g;
                const keyReg = /([a-z]+(?=[\s\(]))/g;
                const i = html.length;
                const h = (match, color) =>
                    `<hgclass class="hgColor" style="color:${color} !important;">${match}</hgclass>`;
                html = html.replace(
                    reg,
                    (e) => e[0] + h(e.slice(1, -1), "#FA842B") + e.slice(-1)
                );
                html = html.replace(keyReg, (e) => {
                    if (e && keywords.includes(e))
                        return h(e, "rgb(0, 0, 252)");
                    return e;
                });
                const Reg = /\/\//g;
                html = html.replace(Reg, (e) => h(e, "green"));
                i !== html.length && (code[0].innerHTML = html);
            },
            rightClickCopyCode: {
                checkCodeZone(target) {
                    if (target.className === "highlight") {
                        return target;
                    }
                    let p = target.parentNode;
                    let className = p.className;
                    let i = 0;
                    while (className !== "highlight") {
                        p = p.parentNode;
                        if (!p || i > 2) return null;
                        className = p.className;
                        i++;
                    }
                    return p;
                },
                exe(e) {
                    const code = this.checkCodeZone(e.target);
                    if (code) {
                        e.preventDefault();
                        zhihu.clipboardClear.clear(code.innerText);
                        Notification(
                            "this code has been copied to clipboard",
                            "clipboard"
                        );
                    }
                },
                main(node) {
                    //if directly use oncontextmenu, which will corrupt the mouse contextmenus
                    node.addEventListener(
                        "contextmenu",
                        (e) => e.button === 2 && e.ctrlKey && this.exe(e),
                        true
                    );
                },
            },
            main() {
                if (GM_getValue("nocolofultext")) return;
                let holder = document.getElementsByClassName(
                    "RichText ztext Post-RichText"
                );
                if (holder.length === 0) {
                    console.log("get content fail");
                    return;
                }
                holder = holder[0];
                // text is too long
                const t_length = holder.innerText.length;

                if (t_length < 9000) {
                    let code = false;
                    this.blue = Math.ceil(Math.random() * 100) % 2 === 0;
                    !this.blue && (this.index = 255);
                    const tags = ["p", "ul", "li", "ol", "blockquote"];
                    const textNode = [];
                    let i = -1;
                    for (const node of holder.childNodes) {
                        i++;
                        const type = node.nodeType;
                        //text node deal with separately
                        if (type === 3) {
                            textNode.push(i);
                            continue;
                        } else if (!tags.includes(node.tagName.toLowerCase())) {
                            //the continuity of content is interrupted, reset the color;
                            this.resetColor();
                            if (node.className === "highlight") {
                                this.codeHighlight(node);
                                code = true;
                            }
                            continue;
                        }
                        this.arr = [];
                        this.nodeCount = 0;
                        this.getItem(node);
                    }
                    i = textNode.length;
                    if (i > 0) {
                        for (i; i--; ) {
                            let node = holder.childNodes[textNode[i]];
                            const text = node.nodeValue;
                            if (text) {
                                this.arr = [];
                                this.textDetach(text);
                                const iNode =
                                    document.createElement("colorspan");
                                holder.insertBefore(iNode, node);
                                iNode.outerHTML = this.arr.join("");
                                node.remove();
                            }
                        }
                    }
                    this.arr = null;
                    code && this.rightClickCopyCode.main(holder);
                } else {
                    const codes = holder.getElementsByClassName("highlight");
                    if (codes.length > 0) {
                        for (const node of codes) this.codeHighlight(node);
                        this.rightClickCopyCode.main(holder);
                    }
                }
            },
        },
        userPage: {
            username: null,
            changeTitle(mode) {
                const title = document.getElementsByClassName(
                    "ProfileHeader-contentHead"
                );
                if (title.length === 0) return;
                title[0].style.textDecoration =
                    mode === "Block" ? "none" : "line-through";
            },
            userManage(mode) {
                let text = "";
                const info = {};
                if (mode === "Block") {
                    info.mode = "block";
                    blackName.push(this.username);
                    text = "add user to blackname successfully";
                } else {
                    info.mode = "unblock";
                    const i = blackName.indexOf(this.username);
                    i > -1 && blackName.splice(i, 1);
                    text = "remove user from blackname successfully";
                }
                info.username = this.username;
                GM_setValue("blacknamechange", info);
                GM_setValue("blackname", blackName);
                Notification(text, "blackName");
            },
            injectButton(name) {
                createButton(name, name + " this user");
                let assist = document.getElementById("assist-button-container");
                assist.children[1].onclick = (e) => {
                    const button = e.target;
                    const n = button.innerText;
                    this.userManage(n);
                    button.innerText = n === "Block" ? "unBlock" : "Block";
                    this.changeTitle(button.innerText);
                    this.title = button.innerText + " this user";
                };
                assist = null;
                name === "unBlock" && this.changeTitle(name);
            },
            changeButton(mode) {
                const button = document.getElementById(
                    "assist-button-container"
                );
                if (!button) return;
                const name = button.children[1].innerText;
                if ((mode && name === "unBlock") || (!mode && name === "Block"))
                    return;
                button.children[1].innerText = blackName.includes(this.name)
                    ? "unBlock"
                    : "Block";
            },
            main() {
                const profile =
                    document.getElementsByClassName("ProfileHeader-name");
                if (profile.length === 0) {
                    console.log("get usename id fail");
                    return;
                }
                const user = profile[0].innerText;
                if (!user) {
                    console.log("failed to get the user name");
                    return;
                }
                this.username = clear_zero_width(user);
                this.username &&
                    this.injectButton(
                        blackName.includes(this.username) ? "unBlock" : "Block"
                    );
            },
        },
        click_Highlight() {
            let h = GM_getValue("highlight");
            h = !h;
            GM_setValue("highlight", h);
            Notification(
                `the feature of "click to highlight content" has been ${
                    h ? "enable" : "disable"
                }`,
                "Tips"
            );
            this.Filter.colorIndicator.stat = h;
            !h && this.Filter.colorIndicator.restore();
        },
        key_ctrl_clip() {
            let c = GM_getValue("clipboard");
            c = !(c === false ? false : true);
            GM_setValue("clipboard", c);
            Notification(
                `the feature of "clipboard clear" has been ${
                    c ? "enable" : "disable"
                }`,
                "Tips"
            );
            this.clipboardClear.replace_ZH = c;
            return true;
        },
        _top_Picture: {
            display(p) {
                const i = document.getElementsByClassName("TitleImage");
                if (i.length === 0) return;
                i[0].style.display = p ? "none" : "block";
            },
            main() {
                let p = GM_getValue("topnopicture");
                p = !p;
                GM_setValue("topnopicture", p);
                this.display(p);
                Notification(
                    `${p ? "disable" : "enable"} top picture of article`,
                    "Tips"
                );
            },
        },
        key_ctrl_sync(mode) {
            GM_addValueChangeListener(
                "clipboard",
                (name, oldValue, newValue, remote) =>
                    remote && (this.clipboardClear.replace_ZH = newValue)
            );
            GM_addValueChangeListener(
                "clear_close",
                (name, oldValue, newValue, remote) =>
                    remote && newValue && window.close()
            );
            mode
                ? GM_addValueChangeListener(
                      "topnopicture",
                      (name, oldValue, newValue, remote) =>
                          remote && this._top_Picture.display(newValue)
                  )
                : GM_addValueChangeListener(
                      "highlight",
                      (name, oldValue, newValue, remote) =>
                          remote &&
                          ((this.Filter.colorIndicator.stat = newValue),
                          !newValue && this.Filter.colorIndicator.restore())
                  );
        },
        key_open_Reader(mode = false) {
            if (this.autoScroll.scrollState) return;
            const items = document.getElementsByClassName(
                "ContentItem AnswerItem"
            );
            const e = elementVisible.main(document.documentElement, items);
            e && this.qaReader.main(e, this.Filter.foldAnswer.getid(e), mode);
        },
        show_Total: {
            _common() {
                const c = colorful_Console.colors.info;
                let date = GM_getValue("installeddate");
                const script = GM_info.script;
                if (!date) {
                    date = Date.now();
                    GM_setValue("installeddate", date);
                }
                colorful_Console.main(
                    {
                        title: "JS Author:",
                        content: script.author,
                    },
                    c
                );
                colorful_Console.main(
                    {
                        title: "Current Version:",
                        content: script.version,
                    },
                    c
                );
                colorful_Console.main(
                    {
                        title: "Installed Date:",
                        content: new Date(date).toString(),
                    },
                    c
                );
                colorful_Console.main(
                    {
                        title: "LastModified:",
                        content: new Date(script.lastModified).toString(),
                    },
                    c
                );
                colorful_Console.main(
                    {
                        title: "Service Time:",
                        content:
                            "This JS has provided you with more than " +
                            ((Date.now() - date) / 1000 / 60 / 60 / 24).toFixed(
                                0
                            ) +
                            " Days of service",
                    },
                    c
                );
            },
            _zhuanlan(result) {
                const c = colorful_Console.colors.info;
                const title =
                    result.name === "collection"
                        ? "collected articles: "
                        : "like/dislike articles:";
                colorful_Console.main(
                    {
                        title: title,
                        content: result.count,
                    },
                    c
                );
            },
            _other(result) {
                const c = colorful_Console.colors.warning;
                colorful_Console.main(
                    { title: "Black Keywords:", content: blackKey.length },
                    c
                );
                colorful_Console.main(
                    { title: "Blocked Users:", content: blackName.length },
                    c
                );
                colorful_Console.main(
                    {
                        title: "Blocked Questions/Topics:",
                        content: blackTopicAndQuestion.length,
                    },
                    c
                );
                colorful_Console.main(
                    {
                        title: "Blocked Answers/Articles:",
                        content: result.count,
                    },
                    c
                );
            },
            main(mode) {
                if (!dataBaseInstance.db) return;
                dataBaseInstance
                    .getdataCount(
                        mode ? ["collection", "preference"] : ["foldedAnswer"]
                    )
                    .then((results) => {
                        results.forEach((e) =>
                            e.status !== "rejected" && mode
                                ? this._zhuanlan(e.value)
                                : this._other(e.value)
                        );
                        this._common();
                    });
            },
        },
        commander: {
            bgi: {
                bing_image(bgi, mode) {
                    return new Promise((resolve, reject) => {
                        let index = bgi.index || 0;
                        index += mode ? 1 : 0;
                        bgi.index = index > 7 ? 0 : index;
                        bgi.atime = Date.now();
                        const api = `https://cn.bing.com/HPImageArchive.aspx?format=js&idx=${bgi.index}&n=1&pid=hp`;
                        xmlHTTPRequest(api).then(
                            (json) => resolve(json.images[0].url),
                            (err) => {
                                reject(null);
                                console.log(err);
                                Notification(
                                    "failed to get image from bing",
                                    "Warning"
                                );
                            }
                        );
                    });
                },
                timeID: null,
                auto_update(bgi) {
                    const a = bgi.atime;
                    if (a && Date.now() - a < 1000 * 60 * 60 * 24) {
                        bgi = null;
                        return;
                    }
                    this.timeID = setTimeout(() => {
                        bgi.index = 0;
                        this.timeID = null;
                        this._i(bgi, 0);
                    }, 3000);
                },
                name: "bgi_image",
                get get() {
                    return GM_getValue(this.name);
                },
                /**
                 * @param {object} v
                 */
                set set(v) {
                    GM_setValue(this.name, v);
                },
                get_base64_fileType(url) {
                    return url.slice(url.indexOf("/") + 1, url.indexOf(";"));
                },
                d() {
                    const bgi = this.get;
                    if (!bgi) return;
                    const url = Reflect.get(bgi, "url");
                    if (url) {
                        const reg = /\.(ico|gif|png|jpe?g)/;
                        let r = null;
                        let filename = "";
                        let m = false;
                        if (
                            url.startsWith("data:image") ||
                            (r = url.match(reg))
                        ) {
                            if (r) {
                                filename =
                                    "zhihu_bgi_image_" + Date.now() + r[0];
                            } else {
                                m = true;
                                filename =
                                    "zhihu_bgi_image_" +
                                    Date.now() +
                                    this.get_base64_fileType(url);
                            }
                        } else {
                            xmlHTTPRequest(url, 30000, "blob").then(
                                (blob) => {
                                    const file = new FileReader();
                                    file.readAsDataURL(blob);
                                    file.onload = (result) => {
                                        if (result.target.readyState === 2) {
                                            filename =
                                                "zhihu_bgi_image_" +
                                                Date.now() +
                                                this.get_base64_fileType(url);
                                            image_base64_download(
                                                result.target.result,
                                                filename
                                            );
                                        }
                                    };
                                    file.onerror = (e) => {
                                        console.log(e);
                                        Notification(
                                            "failed to convert file to base64",
                                            "Warning"
                                        );
                                    };
                                },
                                (err) => {
                                    console.log(err);
                                    Notification(
                                        "failed to donwload background image"
                                    );
                                }
                            );
                            return;
                        }
                        m
                            ? image_base64_download(url, filename)
                            : Download_module(url, filename, 30000).then(
                                  () =>
                                      colorful_Console.main(
                                          {
                                              title: "download:",
                                              content:
                                                  "download image successfully",
                                          },
                                          colorful_Console.colors.info
                                      ),
                                  (err) =>
                                      Notification(
                                          `fialed to download image; ${
                                              err ? `, ${err}` : ""
                                          }`,
                                          "Warning"
                                      )
                              );
                    }
                },
                /**
                 * @param {String} url
                 */
                set body_img(url) {
                    document.body.style.backgroundImage =
                        url === "none" ? "none" : `url(${url})`;
                },
                get is_body_image() {
                    return document.body.style.backgroundImage;
                },
                /**
                 * @param {string} base64
                 */
                set image_cache(base64) {
                    GM_setValue("fixed_image", base64);
                },
                dele_image_cache() {
                    GM_deleteValue("fixed_image");
                },
                f(cm) {
                    const bgi = this.get;
                    if (bgi && bgi.url && this.is_body_image) {
                        bgi.status = "fixed";
                        const reg = /\.(jpe?g|webp)/;
                        const m = bgi.url.match(reg);
                        if (m) {
                            const reg = /(?<=-f\s?)(0\.)?\d+(\.\d+)?/;
                            const mc = cm.match(reg);
                            let ro = 0;
                            const c_ratio = mc
                                ? (ro = mc[0] * 1) > 1
                                    ? 1
                                    : ro < 0.3
                                    ? 0.3
                                    : ro
                                : 0.3;
                            imageConvertor
                                .main(bgi.url, "webp", c_ratio)
                                .then((base64) => {
                                    this.body_img = base64;
                                    this.image_cache = base64;
                                    this.set = bgi;
                                    colorful_Console.main(
                                        {
                                            title: "cached image",
                                            content:
                                                "cached image successfully",
                                        },
                                        colorful_Console.colors.info
                                    );
                                    Notification(
                                        "change setup successfully",
                                        "Tips"
                                    );
                                }),
                                () => {
                                    colorful_Console.main(
                                        {
                                            title: "cached image",
                                            content:
                                                "failed to cache background iamge",
                                        },
                                        colorful_Console.colors.warning
                                    );
                                    Notification(
                                        "failed to set up image",
                                        "Warning",
                                        3500
                                    );
                                };
                        } else
                            Notification(
                                "compression does not support the format of current image",
                                "Tips",
                                3500
                            );
                    } else
                        Notification(
                            "you need set up background iamge firstly",
                            "Tips"
                        );
                },
                t() {
                    let bgi = this.get;
                    if (bgi) {
                        const at = bgi.atime;
                        const st = bgi.status;
                        if (at && bgi.url && st) {
                            if (
                                st === "auto" &&
                                Date.now() - at < 1000 * 60 * 60 * 24
                            ) {
                                bgi.status = "tmp";
                                !this.is_body_image &&
                                    (this.body_img = bgi.url);
                                this.set = bgi;
                                this.dele_image_cache();
                                return;
                            }
                        }
                    } else bgi = {};
                    this.bing_image(bgi).then((url) => {
                        this.body_img = bgi.url = `https://cn.bing.com${url}`;
                        bgi.status = "tmp";
                        this.set = bgi;
                        this.dele_image_cache();
                    });
                },
                s() {
                    const bgi = this.get;
                    if (!bgi) {
                        Notification(
                            "you need setup background image firstly",
                            "Tips"
                        );
                        return;
                    }
                    this.bing_image(bgi, true).then((url) => {
                        bgi.status = "auto";
                        this.body_img = bgi.url = `https://cn.bing.com${url}`;
                        this.set = bgi;
                        this.dele_image_cache();
                    });
                },
                r(type) {
                    const bgp = GM_getValue("bgpreader");
                    if (!bgp) {
                        Notification(
                            "you have not set up the backgorund iamge of reader"
                        );
                        return;
                    }
                    const bgi = this.get || {};
                    bgi.status = type === true ? "tmp" : "reader";
                    this.set = bgi;
                    this.body_img = bgp;
                    this.dele_image_cache();
                },
                o() {
                    this.body_img = "none";
                    GM_deleteValue(this.name);
                    this.dele_image_cache();
                },
                //type -tmp
                u(cm, type) {
                    const m = cm.match(/https.+(?=\s)?/g);
                    if (!m) {
                        Notification(
                            "your image url does not match rule",
                            "Warning"
                        );
                        return;
                    }
                    const url = m[0];
                    const bgi = this.get || {};
                    bgi.status = type ? "tmp" : "self";
                    bgi.url = url;
                    this.set = bgi;
                    this.body_img = url;
                    this.dele_image_cache();
                },
                _i(bgi, index) {
                    this.bing_image(bgi, index).then((url) => {
                        bgi.status = "auto";
                        this.body_img = bgi.url = `https://cn.bing.com${url}`;
                        this.set = bgi;
                        bgi = null;
                        this.dele_image_cache();
                    });
                },
                i() {
                    let bgi = this.get;
                    if (bgi) {
                        const at = bgi.atime;
                        if (at) {
                            const url = bgi.url;
                            if (url && Date.now() - at > 1000 * 60 * 60 * 24) {
                                bgi.status = "auto";
                                !this.is_body_image && (this.body_img = url);
                                this.set = bgi;
                                this.dele_image_cache();
                                return;
                            }
                        }
                    } else bgi = {};
                    bgi.index = 0;
                    this._i(bgi, 0);
                },
                no_match_tips() {
                    Notification(
                        "please check the command, which does not match the rule",
                        "Warning"
                    );
                },
                cm_format(cm) {
                    this.timeID && clearTimeout(this.timeID);
                    cm = cm.slice(3);
                    if (cm) {
                        const reg = /(?<=\s-)[tosdfur]/g;
                        const m = cm.match(reg);
                        if (!m) {
                            this.no_match_tips();
                            return;
                        }
                        if (m.length === 1) this[m[0]](cm);
                        else {
                            const arr = [...new Set(m)];
                            const i = arr.length;
                            if (i === 1) this[arr[0]](cm);
                            else if (i > 2) this.no_match_tips();
                            else {
                                //only use -t this paramter with -u or -t
                                let f = arr.indexOf("t");
                                if (f < 0) {
                                    this.no_match_tips();
                                    return;
                                }
                                const cn = f === 0 ? arr[1] : arr[0];
                                if (!(cn === "u" || cn === "r")) {
                                    this.no_match_tips();
                                    return;
                                }
                                this[cn](cm, true);
                            }
                        }
                    } else this.i();
                },
                main(cm, index) {
                    index > 3
                        ? Notification(
                              "current webpage does not support this feature",
                              "Tips"
                          )
                        : this.cm_format(cm);
                },
            },
            fold: {
                main() {
                    return "Button ContentItem-action ContentItem-rightButton Button--plain";
                },
            },
            expand: {
                main(index) {
                    return index > 1
                        ? "Button ContentItem-more Button--plain"
                        : "Button ContentItem-rightButton ContentItem-expandButton Button--plain";
                },
            },
            f_e: {
                c(item, name) {
                    const button = item.getElementsByClassName(name);
                    button.length > 0 && button[0].click();
                },
                main(name) {
                    const buttons = document.body.getElementsByClassName(name);
                    if (buttons.length < 6) {
                        let i = buttons.length;
                        for (i; i--; ) buttons[i].click();
                        return;
                    }
                    const items = document.body.getElementsByClassName(
                        "ContentItem AnswerItem"
                    );
                    let i = items.length;
                    if (i === 0) return;
                    else if (i < 6) {
                        for (i; i--; ) this.c(items[i], name);
                    } else {
                        let index = 0;
                        for (const item of items) {
                            if (
                                elementVisible.main(
                                    document.documentElement,
                                    item
                                )
                            )
                                break;
                            index++;
                        }
                        let start = index - 2;
                        start < 0 && (start = 0);
                        let end = start + 4;
                        if (end > --i) {
                            start = start - (end - i);
                            end = i;
                        }
                        start--;
                        for (end; end > start; end--) this.c(items[end], name);
                    }
                },
            },
            light: {
                get cover() {
                    return document.getElementById("screen_shade_cover");
                },
                set_opacity(co, v) {
                    co.style.opacity = v;
                },
                set_color(co, c) {
                    co.style.background = c;
                },
                /**
                 * @param {any} v
                 */
                set set(v) {
                    GM_setValue(this.name, v);
                },
                name: "tmp_cover",
                get get() {
                    return GM_getValue(this.name);
                },
                a(cm, tc) {
                    tc.status = "a";
                    return true;
                },
                item_no_match(name) {
                    Notification(
                        `the format of ${name} does not match rule`,
                        "Tips"
                    );
                },
                c(cm, tc, co) {
                    const reg = /#\w{3,6}/;
                    const m = cm.match(reg);
                    if (!m) {
                        this.item_no_match("color");
                        return false;
                    }
                    this.set_color(co, m[0]);
                    tc.color = m[0];
                    return true;
                },
                o(cm, tc, co) {
                    const reg = /(?<=-o\s?)(0\.)?\d+(\.\d+)?/;
                    const m = cm.match(reg);
                    if (!m) {
                        this.item_no_match("opacity");
                        return false;
                    }
                    let v = m[0] * 1;
                    v > 1 && (v = 1);
                    this.set_opacity(co, v);
                    tc.opacity = v;
                    return true;
                },
                t(cm, tc) {
                    const reg = /(?<=-t\s?)(0\.)?\d+(\.\d+)?/;
                    const m = cm.match(reg);
                    if (!m) {
                        this.item_no_match("time");
                        return false;
                    }
                    let v = m[0] * 1;
                    v > 24 && (v = 24);
                    tc.rtime = v * 60 * 60 * 1000;
                    return true;
                },
                _s(funcs, cm, type) {
                    const c = this.cover;
                    if (!c) {
                        Notification(
                            "the dom of cover has been removed",
                            "Tips"
                        );
                        return;
                    }
                    const tc = {};
                    tc.status = type ? "a" : "s";
                    tc.update = Date.now();
                    funcs.forEach((e) => this[e](cm, tc, c));
                    type && this.set_sync(funcs, tc);
                },
                set_o(tc) {
                    GM_setValue("opacity", tc.opacity);
                },
                set_c(tc) {
                    GM_setValue("color", tc.color);
                },
                set_sync(funcs, tc) {
                    funcs.forEach((e) => {
                        const n = this[`set_${e}`];
                        n && n(tc);
                    });
                    this.set = tc;
                },
                no_match_tips() {
                    Notification(
                        "please check the command, which does not match the rule",
                        "Warning"
                    );
                },
                cm_format(cm) {
                    cm = cm.slice("light".length);
                    this._is_single = true;
                    if (cm) {
                        const reg = /(?<=\s-)[acot]/g;
                        const m = cm.match(reg);
                        if (!m) {
                            this.no_match_tips();
                            return;
                        }
                        const arr = [...new Set(m)];
                        const a = arr.includes("a");
                        const t = arr.includes("t");
                        const i = arr.length;
                        if ((t && !a) || i > 3 || (i === 1 && (a || t))) {
                            this.no_match_tips();
                            return;
                        }
                        this._s(arr, cm, a);
                        this._is_single = !a;
                    }
                },
                _is_single: false,
                main(cm) {
                    this.cm_format(cm);
                },
            },
            last_time_cm: "",
            help: {
                a() {
                    const ts = Object.values(Assist_info_URL);
                    ts.forEach(
                        (a, index) =>
                            (a.includes("meituan") || a.endsWith(".md")) &&
                            setTimeout(() =>
                                GM_openInTab(
                                    a,
                                    { insert: true, active: true },
                                    index * 300
                                )
                            )
                    );
                },
                i() {
                    GM_openInTab(Assist_info_URL.cmd_help, {
                        insert: true,
                        active: true,
                    });
                },
                main(cm) {
                    cm = cm.slice(4);
                    if (cm) {
                        const reg = /\s-a/;
                        cm.match(reg) && this.a();
                    }
                    this.i();
                },
            },
            reset: {
                get_confirm(name) {
                    return confirm(`are you sure want to clear the ${name}`);
                },
                get_warning(name) {
                    if (!this.get_confirm(name)) return false;
                    return confirm(
                        "this operation will clear your important data, continue?"
                    );
                },
                get close_confirm() {
                    return confirm(
                        "this operation needs close other tab(page) of zhihu, continue?"
                    );
                },
                success() {
                    Notification("operation is completed", "Tips");
                },
                c() {
                    if (!this.get_confirm("cookie")) return;
                    const c = document.cookie;
                    if (c) {
                        const keys = c.match(/[^ =;]+(?=\=)/g);
                        if (keys) {
                            const domains = [
                                ".zhihu.com",
                                "www.zhihu.com",
                                "zhuanlan.zhihu.com, api.zhihu.com",
                            ];
                            keys.forEach((e) =>
                                domains.forEach(
                                    (d) =>
                                        (document.cookie =
                                            e +
                                            `=0; expires=Sat, 4 Jun 1989 12:00:00 GMT; path=/; domain=${d}`)
                                )
                            );
                        }
                    }
                    this.success();
                },
                s() {
                    if (!this.get_confirm("sessionStorage")) return;
                    sessionStorage.clear();
                    this.success();
                },
                l() {
                    if (!this.get_confirm("localStorage")) return;
                    localStorage.clear();
                    this.success();
                },
                alert() {
                    alert("current webpage need restart");
                    GM_setValue("clear_close", false);
                    setTimeout(() => window.close(), 50);
                },
                is_d: false,
                is_close: false,
                close_sync() {
                    GM_setValue("clear_close", true);
                    this.is_close = true;
                },
                d() {
                    if (
                        !this.get_warning("IndexedDB") ||
                        (!this.is_close && !this.close_confirm)
                    )
                        return;
                    !this.is_close && this.close_sync();
                    this.is_d = true;
                    setTimeout(() => {
                        dataBaseInstance.close();
                        setTimeout(() => Database.deleDB("zhihuDatabase"), 300);
                    }, 300);
                    this.success();
                },
                is_t: false,
                t() {
                    if (
                        !this.get_warning("Tampermonkey") ||
                        (!this.is_close && !this.close_confirm)
                    )
                        return;
                    !this.is_close && this.close_sync();
                    this.is_t = true;
                    const items = GM_listValues();
                    items &&
                        setTimeout(
                            () => items.forEach((e) => GM_deleteValue(e)),
                            300
                        );
                    this.success();
                },
                fail() {
                    Notification("your cmd does not match the rule", "Tips");
                },
                main(cm) {
                    cm = cm.slice(5);
                    if (cm) {
                        const reg = /(?<=\s-)[sldtc]/g;
                        const m = cm.match(reg);
                        m
                            ? [...new Set(m)].forEach((e) => this[e]())
                            : this.fail();
                        (this.is_d || this.is_t) &&
                            setTimeout(
                                () => this.alert(),
                                this.is_d ? 1200 : 300
                            );
                        this.is_t = this.is_d = false;
                    } else this.fail();
                },
            },
            monkey: {
                get eid() {
                    const id = GM_getValue("monkey_id");
                    if (id) return id;
                    const ua = this.ua;
                    ua.includes("Edg")
                        ? "iikmkjmpaadaobahmlepeloendndfphd"
                        : ua.includes("Chrome")
                        ? "dhdgffkkebhmkfjojejmpbldmpobfkfo"
                        : "";
                },
                get ua() {
                    return navigator.userAgent;
                },
                get uuid() {
                    return GM_info.script.uuid;
                },
                set eid(eid) {
                    GM_setValue("monkey_id", eid);
                    Notification("set eid successfully", "Tips");
                },
                open(url, mode) {
                    GM_openInTab(
                        url,
                        mode ? { insert: true } : { insert: true, active: true }
                    );
                },
                notice() {
                    this.open("chrome://extensions/", true);
                    alert("you need set the id of tampermonkey firstly");
                },
                fail() {
                    Notification("your cmd does not match the rule", "Tips");
                },
                i() {
                    const eid = this.eid;
                    eid
                        ? this.open(`chrome-extension://${eid}/options.html`)
                        : this.notice();
                },
                e(cm) {
                    const reg = /[a-z]{32}/;
                    const m = cm.match(reg);
                    m ? (this.eid = m[0]) : this.fail();
                },
                s() {
                    const eid = this.eid;
                    eid
                        ? this.open(
                              `chrome-extension://${eid}/options.html#nav=${this.uuid}+storage`
                          )
                        : this.notice();
                },
                o() {
                    const eid = this.eid;
                    eid
                        ? this.open(
                              `chrome-extension://${eid}/options.html#nav=settings`
                          )
                        : this.notice();
                },
                main(cm) {
                    cm = cm.slice(6);
                    if (cm) {
                        const reg = /(?<=\s-)[soe]/g;
                        const m = cm.match(reg);
                        !m || m.length > 1
                            ? this.fail()
                            : m[0] === "e"
                            ? this.e(cm)
                            : this[m[0]]();
                    } else this.i();
                },
            },
            main(index) {
                let cm = prompt(
                    "please input cmd string, e.g.:$+ fold, ligth, expand, bgi, reset, help",
                    this.last_time_cm || "$"
                );
                if (!cm || cm.length < 2 || !(cm = cm.trim()) || cm[0] !== "$")
                    return true;
                cm = cm.slice(1).toLowerCase().trim();
                if (!cm) return true;
                const cms = [
                    "bgi",
                    "light",
                    "help",
                    "reset",
                    "monkey",
                    "fold",
                    "expand",
                ];
                const r = cms.findIndex((e) => cm.startsWith(e));
                r < 0
                    ? this.reset.fail()
                    : r < 5
                    ? this[cms[r]].main(cm, index)
                    : index > 3
                    ? Notification(
                          "current webpage does not support this feature",
                          "Tips"
                      )
                    : this.f_e.main(this[cms[r]].main(index));
                this.last_time_cm = `$${cm}`;
                return true;
            },
        },
        //the origin keyboard event of zhihu
        key_conflict(keyCode, shift) {
            return 68 === keyCode || (shift && [71, 85].includes(keyCode));
        },
        common_KeyEevnt(shift, keyCode, index = -1) {
            /*
            open user manual webpage;
            open shortcuts webpage;
            enable/disable convert chinese letter to english letter;
            multi search;
            white noise control
            */
            if (shift) {
                const url =
                    keyCode === 85
                        ? Assist_info_URL.usermanual
                        : keyCode === 75
                        ? Assist_info_URL.shortcuts
                        : keyCode === 69
                        ? this.key_ctrl_clip()
                        : keyCode === 77
                        ? this.multiSearch.site()
                        : keyCode === 87
                        ? index !== 9 && this.white_noise.destroy_audio()
                        : keyCode === 86
                        ? this.white_noise.audio_ctrl.play_pause()
                        : keyCode === 187
                        ? this.white_noise.audio_ctrl.voice_up_down(true)
                        : keyCode === 189
                        ? this.white_noise.audio_ctrl.voice_up_down(false)
                        : keyCode === 90
                        ? this.white_noise.change_music(true)
                        : keyCode === 88
                        ? "https://zhuanlan.zhihu.com/"
                        : keyCode === 81
                        ? (this.commander.main(index) &&
                              (this.shade.sing_protect =
                                  this.commander.light._is_single)) ||
                          true
                        : null;
                if (url) {
                    typeof url === "string" &&
                        GM_openInTab(url, { insert: true, active: true });
                    return true;
                }
            }
            return false;
        },
        auto_load_Reader: {
            get setup() {
                return GM_getValue("autoloadreader");
            },
            get nsetup() {
                return GM_getValue("nursereader");
            },
            main() {
                let a = this.setup;
                a = !a;
                GM_setValue("autoloadreader", a);
                Notification(
                    a
                        ? "enabled automatic reading mode successfully"
                        : "has diabled automatic reading mode",
                    "Tips"
                );
            },
            nurse() {
                let n = this.nsetup;
                n = !n;
                GM_setValue("nursereader", n);
                Notification(
                    n
                        ? "enabled nurse reading mode successfully"
                        : "has diabled automatic nurse reading mode",
                    "Tips"
                );
            },
            timeID: null,
            cancel() {
                clearTimeout(this.auto_load_Reader.timeID);
                this.Filter.auto_load_reader = false;
                this.auto_load_Reader.timeID = null;
            },
            start(mode) {
                Notification(
                    "5s, automatically load reader page; click to cancel",
                    "Tips",
                    5000,
                    this.auto_load_Reader.cancel.bind(this)
                );
                this.auto_load_Reader.timeID = setTimeout(() => {
                    this.Filter.auto_load_reader = false;
                    this.auto_load_Reader.timeID = null;
                    this.key_open_Reader(mode);
                }, 5300);
                this.Filter.auto_load_reader = true;
            },
        },
        QASkeyBoardEvent(index) {
            /*
            when in autoloaded reader mode, block keyevent if the autoloaded is progressing;
            */
            document.addEventListener(
                "keydown",
                (e) => {
                    if (
                        e.ctrlKey ||
                        e.altKey ||
                        e.target.localName === "input" ||
                        this.auto_load_Reader.timeID
                    )
                        return;
                    const className = e.target.className;
                    if (
                        className &&
                        typeof className === "string" &&
                        className.includes("DraftEditor")
                    )
                        return;
                    const shift = e.shiftKey;
                    const keyCode = e.keyCode;
                    if (this.key_conflict(keyCode, shift)) {
                        e.preventDefault();
                        e.stopImmediatePropagation();
                    }
                    if (this.common_KeyEevnt(shift, keyCode, index)) return;
                    const r = this.qaReader.readerMode;
                    r
                        ? this.qaReader.keyEvent(keyCode, shift)
                        : shift
                        ? keyCode === 72
                            ? this.click_Highlight()
                            : keyCode === 82
                            ? index < 2 && this.key_open_Reader(false)
                            : keyCode === 66
                            ? index < 9 &&
                              !this.autoScroll.scrollState &&
                              MangeData.exportData.main(true)
                            : keyCode === 73
                            ? index < 9 &&
                              !this.autoScroll.scrollState &&
                              MangeData.importData.main(false)
                            : keyCode === 68
                            ? index < 2 && this.auto_load_Reader.main()
                            : keyCode === 78
                            ? index < 2 && this.auto_load_Reader.nurse()
                            : null
                        : keyCode === 192
                        ? (this.autoScroll.start(),
                          (this.Filter.is_scroll_state =
                              this.autoScroll.scrollState))
                        : keyCode === 187
                        ? this.autoScroll.speedUP()
                        : keyCode === 189
                        ? this.autoScroll.slowDown()
                        : keyCode === 78
                        ? !this.autoScroll.scrollState &&
                          this.turnPage.start(true)
                        : keyCode === 84
                        ? !this.autoScroll.scrollState && this.scroll.toTop()
                        : keyCode === 82
                        ? index < 2 &&
                          location.pathname.includes("/answer/") &&
                          !this.autoScroll.scrollState &&
                          this.scroll.toBottom()
                        : keyCode === 85
                        ? !this.autoScroll.scrollState &&
                          this.turnPage.start(false)
                        : this.multiSearch.main(keyCode);
                },
                true
            );
        },
        right_click_F_E: {
            get_contentNode(node) {
                return node.className === "ContentItem AnswerItem"
                    ? node
                    : node.parentNode.parentNode;
            },
            fold_item(node, className) {
                const button =
                    this.get_contentNode(node).getElementsByClassName(
                        className
                    );
                button.length > 0 && button[0].click();
            },
            get_m_node(node) {
                const tmp = node.getElementsByClassName("RichContent-inner");
                return tmp.length === 0 ? null : tmp[0];
            },
            button_display(node, mode) {
                const b = this.get_button(node);
                b && (b.style.display = mode ? "block" : "none");
            },
            zhuanlan_fold(node) {
                const m = this.get_m_node(node);
                if (m) {
                    m.style.maxHeight = "400px";
                    m.parentNode.className =
                        "RichContent is-collapsed RichContent--unescapable";
                    this.button_display(node, true);
                }
            },
            load_lazy_img(node) {
                const imgs = node.getElementsByTagName("img");
                for (const i of imgs) {
                    if (i.currentSrc.startsWith("data:image/svg")) {
                        const src = i.dataset.original || i.dataset.actualsrc;
                        src && (i.src = src);
                    }
                }
            },
            get_button(node) {
                const button = node.getElementsByClassName(
                    "Button ContentItem-rightButton ContentItem-expandButton Button--plain"
                );
                return button.length === 0 ? null : button[0];
            },
            zhuanlan_expand(node, mode) {
                const m = this.get_m_node(node);
                if (m) {
                    m.removeAttribute("style");
                    m.parentNode.className =
                        "RichContent RichContent--unescapable";
                    !mode && this.load_lazy_img(node);
                    this.button_display(node, false);
                }
            },
            zhuanlan_E_F(p) {
                let node = this.get_contentNode(p);
                node.className !== "ContentItem AnswerItem" &&
                    (node = node.parentNode);
                const tmp = node.dataset.is_show;
                tmp
                    ? this.zhuanlan_fold(node)
                    : tmp === false
                    ? this.zhuanlan_expand(node, true)
                    : this.zhuanlan_expand(node, false);
                node.dataset.is_show = !tmp;
                tmp && node.parentNode.scrollIntoView();
            },
        },
        rightMouse_OpenQ(index) {
            //open the specific url in search webpage or topic webpage directly
            document.oncontextmenu = (e) => {
                if (e.shiftKey) {
                    const pt = e.path;
                    let i = 0;
                    for (const p of pt) {
                        let cn = "";
                        if (p.localName === "a") {
                            const pos = ["/answer/", "/topic/"];
                            const index = pos.findIndex((o) =>
                                p.pathname.includes(o)
                            );
                            if (index > -1) {
                                e.preventDefault();
                                const href = p.href;
                                GM_openInTab(
                                    index === 0
                                        ? href.slice(
                                              0,
                                              href.lastIndexOf(pos[index])
                                          )
                                        : `${href}/top-answers`,
                                    {
                                        insert: true,
                                        active: true,
                                    }
                                );
                            }
                            break;
                        } else if ((cn = p.className) && cn) {
                            if (this.qaReader.readerMode) return;
                            if (
                                cn ===
                                    "RichText ztext CopyrightRichText-richText" ||
                                cn === "ContentItem AnswerItem"
                            ) {
                                e.preventDefault();
                                index === 9
                                    ? this.right_click_F_E.zhuanlan_E_F(p)
                                    : this.right_click_F_E.fold_item(
                                          p,
                                          this.commander.fold.main()
                                      );
                                break;
                            } else if (
                                cn.endsWith(
                                    "Button--withIcon Button--withLabel"
                                )
                            ) {
                                e.preventDefault();
                                break;
                                if (index < 2) {
                                }
                            }
                        } else if (i > 6) break;
                        i++;
                    }
                } else if (e.ctrlKey)
                    this.colorAssistant.rightClickCopyCode.exe(e);
            };
        },
        original_status: false,
        visible_time_id: null,
        visibleChange(mode) {
            /*
            switch the webpage(tab), control(start or pause the scroll) the scroll state
            */
            this.visible_time_id && clearTimeout(this.visible_time_id);
            if (this.qaReader.readerMode) {
                if (mode) {
                    this.original_status = this.qaReader.autoScroll.scrollState;
                    this.original_status &&
                        this.qaReader.autoScroll.stopScroll(false);
                } else if (this.original_status) {
                    this.visible_time_id = setTimeout(() => {
                        this.visible_time_id = null;
                        this.qaReader.autoScroll.keyCount = 2;
                        this.qaReader.autoScroll.start();
                    }, 300);
                }
            } else {
                if (mode) {
                    this.original_status = this.autoScroll.scrollState;
                    this.original_status && this.autoScroll.stopScroll(false);
                } else if (this.original_status) {
                    this.visible_time_id = setTimeout(() => {
                        this.visible_time_id = null;
                        this.autoScroll.keyCount = 2;
                        this.autoScroll.start();
                    }, 600);
                }
            }
        },
        Filter_Reader_sync() {
            this.qaReader.readerMode && (this.qaReader.scroll_record += 3);
        },
        hide_search_sync(filter) {
            let b = this.searchPage.is_simple_search;
            Object.defineProperty(this.searchPage, "is_simple_search", {
                enumerable: true,
                configurable: true,
                set(v) {
                    filter.is_simple_search = b = v;
                },
                get() {
                    return b;
                },
            });
            return b;
        },
        add_reader_button() {
            const html = `
                <i
                    class="reader_mode_ico"
                    title="open current answer in reader mode"
                    style="
                        content: url();
                        right: 10%;
                        width: 45px;
                        height: 55px;
                        position: fixed;
                        bottom: 12%;
                    "
                ></i>`;
            document.body.insertAdjacentHTML("beforeend", html);
            setTimeout(
                () =>
                    (document.body.getElementsByClassName(
                        "reader_mode_ico"
                    )[0].onclick = () =>
                        !this.auto_load_Reader.timeID &&
                        this.key_open_Reader()),
                0
            );
        },
        pageOfQA(index, href) {
            //inject as soon as possible; maybe, need to concern about some conflict between eventlisteners and MutationObserver
            this.addStyle(index);
            this.clearStorage();
            window.onload = () => {
                if (index !== 8) {
                    this.getData();
                    this.blackUserMonitor(index);
                    (index < 4
                        ? !(index === 1 && href.endsWith("/waiting"))
                        : false) &&
                        setTimeout(() => {
                            this.Filter.is_simple_search =
                                this.searchPage.is_simple_search;
                            this.Filter.main(
                                index,
                                this.Filter_Reader_sync.bind(this)
                            );
                            this.QASkeyBoardEvent(index);
                            setTimeout(
                                () => this.show_Total.main(false),
                                30000
                            );
                            this.key_ctrl_sync(false);
                            this.rightMouse_OpenQ(index);
                            index < 2
                                ? setTimeout(() => {
                                      this.add_reader_button();
                                      let n = false;
                                      ((n = this.auto_load_Reader.nsetup) ||
                                          this.auto_load_Reader.setup) &&
                                          this.auto_load_Reader.start.call(
                                              this,
                                              n
                                          );
                                  }, 300)
                                : index === 3 &&
                                  (this.searchPage.main(),
                                  this.hide_search_sync(this.Filter));
                            this.body_img_update &&
                                this.commander.bgi.auto_update(
                                    this.body_img_update
                                );
                        }, 100);
                    (index === 6 || index === 7) && this.userPage.main();
                }
                this.inputBox.monitor(
                    index,
                    index < 4 ? this.visibleChange.bind(this) : null
                );
            };
        },
        blackUserMonitor(index) {
            GM_addValueChangeListener(
                "blackname",
                (name, oldValue, newValue, remote) => {
                    if (!remote) return;
                    //mode => add user to blockname
                    blackName = newValue;
                    if (index === 6 || index === 7) {
                        const mode =
                            !oldValue || oldValue.length < newValue.length;
                        this.userPage.changeButton(mode);
                    } else this.Filter.userChange(index);
                }
            );
        },
        pre_Check(content) {
            blackKey.some((e) => content.includes(e)) &&
                confirm(
                    "The link you are currently visiting contains rubbish, close the tab?"
                ) &&
                window.close();
        },
        remove_noise(href) {
            const tmp = href.slice(href.lastIndexOf("com/") + 4);
            if (tmp && !tmp.includes("/")) {
                const noise = ["?", "#"];
                const arr = noise.map((e) => href.lastIndexOf(e));
                const index = Math.min(...arr);
                if (index > 0) return href.slice(0, index);
            }
            return href;
        },
        start() {
            let href = location.href;
            const excludes = ["/write", "/api/", "/api"];
            if (excludes.some((e) => href.includes(e))) return;
            fetch_intercept.initial();
            this.anti_setInterval();
            const search = location.search;
            search &&
                search.length > 2 &&
                this.pre_Check(decodeURIComponent(search));
            const includes = [
                "/answer/",
                "/question/",
                "/topic/",
                "/search",
                "/column",
                "/zhuanlan",
                "/people/",
                "/org/",
                "/www",
            ];
            const index = includes.findIndex((e) => href.includes(e));
            let z = false;
            let f = false;
            (
                (z = index === 5) && (href = this.remove_noise(href))
                    ? (!(f = href.endsWith("zhihu.com/")) &&
                          this.pre_Check(document.title)) ||
                      z
                    : index === 4
            )
                ? this.zhuanlanStyle(z && href.includes("/p/") ? 0 : f ? 1 : 2)
                : (this.pre_Check(document.title), this.pageOfQA(index, href));
            this.antiRedirect();
            this.antiLogin();
            this.shade.start();
            setTimeout(() => !this.is_delayed && (this.hasLogin = true), 15000);
            this.clipboardClear.event();
            setTimeout(() => installTips.main(), 3000);
        },
    };
    zhihu.start();
})();