Greasy Fork is available in English.

巴哈黑名單、關鍵詞、字數過少隱藏顯示

在巴哈姆將黑名單、關鍵詞、字數過少過濾文章留言,在頂端列可以開關過濾器(一次性)

// ==UserScript==
// @name         巴哈黑名單、關鍵詞、字數過少隱藏顯示
// @namespace    https://home.gamer.com.tw/moontai0724
// @version      2.4.9
// @description  在巴哈姆將黑名單、關鍵詞、字數過少過濾文章留言,在頂端列可以開關過濾器(一次性)
// @author       moontai0724
// @match        https://forum.gamer.com.tw/B.php*
// @match        https://forum.gamer.com.tw/Bo.php*
// @match        https://forum.gamer.com.tw/C.php*
// @match        https://forum.gamer.com.tw/Co.php*
// @match        https://forum.gamer.com.tw/search.php*
// @grant        GM_xmlhttpRequest
// @connect      home.gamer.com.tw
// @require      https://ajax.googleapis.com/ajax/libs/jquery/3.3.1/jquery.min.js
// @supportURL   https://home.gamer.com.tw/moontai0724
// @license      MIT
// ==/UserScript==

(function () {
    'use strict';
    console.log('Start, varsion: ' + GM_info.script.version + ', setting: ', localStorage.getItem("BLH_Setting"));
    if (!localStorage.getItem("BLH_Setting") || JSON.parse(localStorage.getItem("BLH_Setting")) == null)
        (localStorage.setItem("BLH_Setting", JSON.stringify({
            switch: {
                keywordPostFilter: true,
                keywordCommentFilter: true,
                blacklistPostFilter: true,
                blacklistCommentFilter: true,
                contentLengthPostFilter: false,
                contentLengthCommentFilter: false,
            },
            lengthLimit: {
                contentLengthPostLimit: 2,
                contentLengthCommentLimit: 2
            },
            data: {
                forceShowList: [],
                forceHideList: [],
                blockKeywordsPC: [],
                postBlockKeywordsPC: [],
                commentBlockKeywordsPC: [],
                blockKeywordsFC: [],
                postBlockKeywordsFC: [],
                commentBlockKeywordsFC: []
            }
        })), console.log('New Setting'));
    var setting = JSON.parse(localStorage.getItem("BLH_Setting"));

    // 加入隱藏切換
    jQuery('head').append('<style type="text/css" id="BLH_BlockHideCSS">.BlockHide { display: none !important; }</style>');
    jQuery('.BH-menuE').append('<li><a id="BLH_ShowBlock" href="javascript:;" style="display: block;" onclick="BlockDisplay(true);">關閉過濾器</a></li>');
    jQuery('.BH-menuE').append('<li><a id="BLH_HideBlock" href="javascript:;" style="display: none;" onclick="BlockDisplay(false);">開啟過濾器</a></li>');
    jQuery('.BH-menuE').append('<li><a id="BLH_FilterSetting" href="javascript:;" style="display: block;">過濾器設定</a></li>');
    jQuery('body').append('<script type="text/javascript">' + function BlockDisplay(status) {
        document.getElementById('BLH_HideBlock').style.display = status ? 'block' : 'none';
        document.getElementById('BLH_ShowBlock').style.display = status ? 'none' : 'block';
        document.getElementById('BLH_BlockHideCSS').innerHTML = document.getElementById('BLH_BlockHideCSS').innerHTML.replace(status ? 'BlockHide' : 'noBlockHide', status ? 'noBlockHide' : 'BlockHide');
    } + '</script>');
    document.getElementById('BLH_FilterSetting').onclick = () => openSettingWindow();

    // 將 blockKeywords 加入 postBlockKeywords 和 commentBlockKeywords 中
    for (let i = 0; i < setting.data.blockKeywordsPC.length; i++) setting.data.postBlockKeywordsPC[setting.data.postBlockKeywordsPC.length] = setting.data.commentBlockKeywordsPC[setting.data.commentBlockKeywordsPC.length] = setting.data.blockKeywordsPC[i];
    for (let i = 0; i < setting.data.blockKeywordsFC.length; i++) setting.data.postBlockKeywordsFC[setting.data.postBlockKeywordsFC.length] = setting.data.commentBlockKeywordsFC[setting.data.commentBlockKeywordsFC.length] = setting.data.blockKeywordsFC[i];

    //BC頁分開
    switch (location.pathname) {
        case '/B.php':
            console.group('Filter log message');
            setTimeout(() => console.groupEnd(), 200);

            // blacklist post filter
            if (setting.switch.blacklistPostFilter) startFilter('blacklist', 'post', '.b-list__count__user>a', '.b-list__row');
            // keywords post title filter
            if (setting.switch.keywordPostFilter) (startFilter('postBlockKeywordsPC', 'post', '.b-list__main__title', '.b-list__row'), startFilter('postBlockKeywordsFC', 'post', '.b-list__main__title', '.b-list__row'));
            // popular recommend title filter
            if (setting.switch.keywordPostFilter) (startFilter('postBlockKeywordsPC', 'post', '.popular .name', '.popular__item'), startFilter('postBlockKeywordsFC', 'post', '.popular .name', '.popular__item'));
            break;
        case '/C.php': case '/Co.php':
            console.group('Filter log message');
            setTimeout(() => console.groupEnd(), 200);
            // 擷取展開按鈕事件:當展開留言按鈕被點擊,執行原生展開留言指令並處理內容
            jQuery('body').append('<button id="extendCommentListener" style="display: none;"></button>');
            jQuery('.more-reply').each((index, element) => element.setAttribute("onclick", element.getAttribute("onclick") + " [document.getElementById('extendCommentListener').dataset.bsn, document.getElementById('extendCommentListener').dataset.postid] = [" + element.getAttribute('onclick').replace('extendComment(', '').replace(');', '').split(', ') + "]; jQuery('#Commendlist_" + element.getAttribute('onclick').replace('extendComment(', '').replace(');', '').split(', ')[1] + "').append('<div id=\"extendCommentAreaListener\"></div>'); document.getElementById('extendCommentListener').click();"));
            // 當按鈕點擊就執行
            document.getElementById('extendCommentListener').onclick = function () {
                let [bsn, postid] = [document.getElementById('extendCommentListener').dataset.bsn, document.getElementById('extendCommentListener').dataset.postid], times = 0, ms = 0;
                setTimeout(function restartFilter(ms) {
                    setTimeout(function () {
                        if (!document.getElementById('extendCommentAreaListener')) {
                            jQuery('#Commendlist_' + postid).each((index, element) => {
                                if (setting.switch.blacklistCommentFilter) {
                                    getBlackList().then(BlackList => jQuery(element).find('.reply-content__user').each((index, value) => {
                                        if (BlackList.includes(value.href.replace('https://home.gamer.com.tw/', '').toLowerCase())) {
                                            jQuery(value).parents('.c-reply__item').addClass('BlockHide');
                                            console.log('Hid a comment with block user: ' + value.innerText.toLowerCase(), jQuery(value).parents('.c-reply__item'));
                                        }
                                    }));
                                }
                                if (setting.switch.keywordCommentFilter) {
                                    jQuery(element).find('.reply-content__article').each((index, value) => {
                                        setting.data.commentBlockKeywordsPC.forEach(data => {
                                            if (value.innerText.toLowerCase().includes(data)) {
                                                jQuery(value).parents('.c-reply__item').addClass('BlockHide');
                                                console.log('Hid a post includes keyword: ' + data, jQuery(value).parents('.c-reply__item'));
                                            }
                                        });
                                        setting.data.commentBlockKeywordsFC.forEach(data => {
                                            if (value.innerText.toLowerCase() == data) {
                                                jQuery(value).parents('.c-reply__item').addClass('BlockHide');
                                                console.log('Hid a post includes keyword: ' + data, jQuery(value).parents('.c-reply__item'));
                                            }
                                        });
                                    });
                                }
                                if (setting.switch.contentLengthCommentFilter) {
                                    jQuery(element).find('.reply-content__article').each((index, value) => {
                                        let data = value.innerText.replace(/\s/g, '').length;
                                        if (data < setting.lengthLimit.contentLengthCommentLimit) {
                                            jQuery(value).parents('.c-reply__item').addClass('BlockHide');
                                            console.log('Hid a comment less then setting text totalLength limit: ' + setting.lengthLimit.contentLengthCommentLimit, data, jQuery(value).parents('.c-reply__item'));
                                        }
                                    });
                                }
                            });
                        } else if (times++ < 50) restartFilter(100);
                    }, ms);
                });
            };

            // blacklist post filter
            if (setting.switch.blacklistPostFilter) startFilter('blacklist', 'post', '.c-post__header__author>.userid', '.c-section');
            // blacklist comment filter
            if (setting.switch.blacklistCommentFilter) startFilter('blacklist', 'comment', '.reply-content__user', '.c-reply__item');
            // keywords post filter
            if (setting.switch.keywordPostFilter) (startFilter('postBlockKeywordsPC', 'post', '.c-article.FM-P2', '.c-section'), startFilter('postBlockKeywordsFC', 'post', '.c-article.FM-P2', '.c-section'));
            // keywords comment filter
            if (setting.switch.keywordCommentFilter) (startFilter('commentBlockKeywordsPC', 'post', '.reply-content__article', '.c-reply__item'), startFilter('commentBlockKeywordsFC', 'post', '.reply-content__article', '.c-reply__item'));
            // post content length filter
            if (setting.switch.contentLengthPostFilter) startFilter('contentLengthPostLimit', 'post', '.c-article.FM-P2', '.c-section');
            // comment content length filter
            if (setting.switch.contentLengthCommentFilter) startFilter('contentLengthCommentLimit', 'comment', '.reply-content__article', '.c-reply__item');
            // popular recommend title filter
            if (setting.switch.keywordPostFilter) (startFilter('postBlockKeywordsPC', 'post', '.popular .name', '.popular__item'), startFilter('postBlockKeywordsFC', 'post', '.popular .name', '.popular__item'));
            break;
        case '/Bo.php':
            console.group('Filter log message');
            setTimeout(() => console.groupEnd(), 200);
            // blacklist post filter
            if (setting.switch.blacklistPostFilter) startFilter('blacklist', 'post', '.FM-blist6>a[href*="home.gamer.com.tw"]', 'tr');
            // keywords post title filter
            if (setting.switch.keywordPostFilter) (startFilter('postBlockKeywordsPC', 'post', '.FM-blist3', 'tr'), startFilter('postBlockKeywordsFC', 'post', '.FM-blist3', 'tr'));
            break;
        case '/search.php':
            setTimeout(function restartFilter(ms) {
                setTimeout(() => {
                    if (jQuery('.gsc-table-cell-snippet-close').length > 0) {
                        console.group('Filter log message');
                        setTimeout(() => console.groupEnd(), 100);
                        if (setting.switch.keywordPostFilter) (startFilter('postBlockKeywordsPC', 'post', '.gsc-table-cell-snippet-close', '.gsc-result'), startFilter('postBlockKeywordsFC', 'post', '.gsc-table-cell-snippet-close', '.gsc-result'));
                    } else restartFilter(100);
                }, ms);
            });
            break;
    }

    function startFilter(filterType, elementType, target, hideClass) {
        if (filterType.includes('Keyword') && filterType.includes('PC')) {
            console.log('Keyword part compare filter start.');
            jQuery(target).each((index, element) => setting.data[filterType].forEach(value => {
                if (element.innerText.toLowerCase().replace(/\s/g, '').includes(value)) {
                    jQuery(element).parents(hideClass).addClass('BlockHide');
                    console.log('Hid a ' + elementType + ' includes keyword: ' + value, jQuery(element).parents(hideClass));
                }
            }));
        } else if (filterType.includes('Keyword') && filterType.includes('FC')) {
            console.log('Keyword full compare filter start.');
            jQuery(target).each((index, element) => setting.data[filterType].forEach(value => {
                if (element.innerText.toLowerCase().replace(/\s/g, '') == value) {
                    jQuery(element).parents(hideClass).addClass('BlockHide');
                    console.log('Hid a ' + elementType + ' includes keyword: ' + value, jQuery(element).parents(hideClass));
                }
            }));
        } else if (filterType.includes('blacklist')) {
            console.log('Blacklist filter start.');
            getBlackList().then(BlackList => jQuery(target).each((index, element) => {
                if (BlackList.includes(element.href.replace('https://home.gamer.com.tw/', '').toLowerCase())) {
                    jQuery(element).parents(hideClass).addClass('BlockHide');
                    console.log('Hid a ' + elementType + ' with block user: ' + element.href.replace('https://home.gamer.com.tw/', '').toLowerCase(), jQuery(element).parents(hideClass));
                }
            }));
        } else if (filterType.includes('Length')) {
            console.log('content length filter start.');
            jQuery(target).each((index, element) => {
                let value = element.innerText.replace(/\s/g, '').length;
                if (value < setting.lengthLimit[filterType]) {
                    jQuery(element).parents(hideClass).addClass('BlockHide');
                    console.log('Hid a ' + elementType + ' less then setting text totalLength limit: ' + setting.lengthLimit[filterType], value, jQuery(element).parents(hideClass));
                }
            });
        }
    }

    function BlackListHandle(BlackList) {
        let removeNum = [];
        setting.data.forceHideList.forEach(value => BlackList.includes(value.replace(/\s/g, '').toLowerCase()) ? void (0) : BlackList[BlackList.length] = value.replace(/\s/g, '').toLowerCase());
        BlackList.forEach((value, index) => setting.data.forceShowList.includes(value.toLowerCase()) ? removeNum[removeNum.length] = index : void (0));
        for (let i = removeNum.length - 1; i >= 0; i--) BlackList.splice(removeNum[i], 1);
        console.log('BlackList handle process over, returned blacklist: ', BlackList);
        return BlackList;
    }

    function getBlackList(forceReload) {
        return new Promise(resolve => {
            console.log('Received get blacklist request!');
            if (localStorage.getItem('BHBlackList') && !forceReload) {
                let BHBlackList = JSON.parse(localStorage.getItem('BHBlackList')), today = new Date((new Date()).getFullYear(), (new Date()).getMonth(), (new Date()).getDate(), 0, 0, 0, 0);
                if (today.getTime() < BHBlackList.time && BHBlackList.time < today.getTime() + 86400000) {
                    console.log('Already requested blacklist today, resolve.');
                    resolve(BlackListHandle(BHBlackList.BlackList));
                } else {
                    console.log('Today not requested blacklist yet, start request.');
                    getBlackList(true).then(data => resolve(BlackListHandle(data)));
                }
            } else {
                console.log('Start get blacklist from Bahamut server.');
                GM_xmlhttpRequest({
                    method: "GET",
                    url: "https://home.gamer.com.tw/ajax/friend_getData.php?here=0",
                    onload: data => {
                        console.log(data);
                        let list = data.responseText.match(/data\-gamercard\-userid\=\"[\s\S]*?\"/g);
                        let BlackList = list != (null || undefined) ? list.map(value => value.replace(/data\-gamercard\-userid\=\"|\"/g, '').toLowerCase()) : [];
                        console.log('Get blacklist data from Bahamut server.');
                        localStorage.setItem('BHBlackList', JSON.stringify({ time: new Date().getTime(), BlackList: BlackList }));
                        resolve(BlackListHandle(BlackList));
                    },
                    onerror: err => console.error('Responsed Error: ', err),
                    ontimeout: err => console.error('Response timeout: ', err)
                });
            }
        });
    }

    function openSettingWindow() {
        // black background
        let BLH_SW_Background = document.createElement("div");
        BLH_SW_Background.id = "BLH_SW_Background";
        BLH_SW_Background.setAttribute("onmousedown", "javascript:if(!jQuery(this).hasClass('mouseenter')) jQuery('#BLH_SW_Background').remove();");
        BLH_SW_Background.style = "background-color: rgba(0, 0, 0, 0.5); z-index: 95; position: fixed; top: 0px; bottom: 0px; left: 0px; right: 0px; width: 100%; height: 100%; padding-top: 35px;" +
            " border: 1px solid #a7c7c8;" +
            " display: flex; align-items: center; justify-content: center;" +
            " -webkit-box-sizing: border-box; -moz-box-sizing: border-box; box-sizing: border-box;";
        document.body.appendChild(BLH_SW_Background);

        // window case
        let BLH_SW_Case = document.createElement("div");
        BLH_SW_Case.id = "BLH_SW_Case";
        BLH_SW_Case.setAttribute("onmouseenter", "javascipt:jQuery('#BLH_SW_Background').addClass('mouseenter');");
        BLH_SW_Case.setAttribute("onmouseleave", "javascipt:jQuery('#BLH_SW_Background').removeClass('mouseenter');");
        BLH_SW_Case.style = "position: absolute; height: 80%; width: 90%; overflow: hidden;" +
            " display: flex; align-item: stretch; flex-direction: column;" +
            " background-color: #FFFFFF; border: 1px solid #a7c7c8;";
        document.getElementById("BLH_SW_Background").appendChild(BLH_SW_Case);

        // title
        let BLH_SW_Title = document.createElement("div");
        BLH_SW_Title.setAttribute("style", "display: flex; align-items: center; justify-content: center; width: 100%; min-height: 35px;" +
            " background-color: #E5F7F8; color: #484b4b;" +
            " font-size: 22px; font-weight: bold; font-family: '微軟正黑體', 'Microsoft JhengHei', '黑體-繁', '蘋果儷中黑', 'sans-serif';");
        BLH_SW_Title.innerHTML = "過濾器設定";
        document.getElementById("BLH_SW_Case").appendChild(BLH_SW_Title);

        // content
        let BLH_SW_Content = document.createElement("div");
        BLH_SW_Content.id = "BLH_SW_Content";
        BLH_SW_Content.setAttribute("style", "display: flex; align-items: center; justify-content: center; flex-flow: row wrap; flex-grow: 1; overflow: auto; padding: 10px;" +
            " background-color: #FFFFFF;" +
            " word-break: break-all; font-size: 16px; line-height: 150%; text-align: center; font-family: 微軟正黑體, Microsoft JhengHei, 黑體-繁, 蘋果儷中黑, sans-serif;" +
            " -webkit-box-sizing: border-box; -moz-box-sizing: border-box; box-sizing: border-box;");
        document.getElementById("BLH_SW_Case").appendChild(BLH_SW_Content);

        // bottom element
        let BLH_SW_BottomArea = document.createElement("div");
        BLH_SW_BottomArea.id = "BLH_SW_BottomArea";
        BLH_SW_BottomArea.style = "display: flex; align-items: center; justify-content: center;" +
            " background-color: #E5F7F8;" +
            " width: 100%; min-height: 35px;";
        document.getElementById('BLH_SW_Case').appendChild(BLH_SW_BottomArea);

        // close button
        let BLH_SW_CloseButton = document.createElement('button');
        BLH_SW_CloseButton.innerHTML = '完成';
        BLH_SW_CloseButton.setAttribute("onclick", "jQuery('#BLH_SW_Background').remove();");
        document.getElementById('BLH_SW_BottomArea').appendChild(BLH_SW_CloseButton);

        // content
        jQuery('#BLH_SW_Content').append(`
    <div style="padding: 10px; border: 1px solid; border-radius: 10px; margin: 10px; background: antiquewhite;">
        切換
        <div style="display: flex; align-items: center; justify-content: center; flex-flow: row wrap;">
            <div style="border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px; background: beige;">
                黑名單過濾
                <div style="display: flex; align-items: center; border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px;">
                    黑名單文章過濾:
                    <span id="blacklistPostFilter"></span>
                    <button style="margin: 5px;" onclick="BLH_switch('blacklistPostFilter');">切換</button>
                </div>
                <div style="display: flex; align-items: center; border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px;">
                    黑名單留言過濾:
                    <span id="blacklistCommentFilter"></span>
                    <button style="margin: 5px;" onclick="BLH_switch('blacklistCommentFilter');">切換</button>
                </div>
            </div>
            <div style="border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px; background: beige;">
                關鍵詞過濾
                <div style="display: flex; align-items: center; border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px;">
                    關鍵詞文章過濾:
                    <span id="keywordPostFilter"></span>
                    <button style="margin: 5px;" onclick="BLH_switch('keywordPostFilter');">切換</button>
                </div>
                <div style="display: flex; align-items: center; border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px;">
                    關鍵詞留言過濾:
                    <span id="keywordCommentFilter"></span>
                    <button style="margin: 5px;" onclick="BLH_switch('keywordCommentFilter');">切換</button>
                </div>
            </div>
            <div style="border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px; background: beige;">
                字數過濾
                <div style="display: flex; align-items: center; border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px;">
                    文章字數過濾:
                    <span id="contentLengthPostFilter"></span>
                    <button style="margin: 5px;" onclick="BLH_switch('contentLengthPostFilter');">切換</button>
                </div>
                <div style="display: flex; align-items: center; border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px;">
                    留言字數過濾:
                    <span id="contentLengthCommentFilter"></span>
                    <button style="margin: 5px;" onclick="BLH_switch('contentLengthCommentFilter');">切換</button>
                </div>
            </div>
            <div style="border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px; background: beige;">
                字數過濾
                <div style="display: flex; align-items: center; border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px;">
                    文章字數下限:
                    <span id="contentLengthPostLimit"></span>
                    <button style="margin: 5px;" onclick="BLH_lengthLimit('contentLengthPostLimit');">變更</button>
                </div>
                <div style="display: flex; align-items: center; border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px;">
                    留言字數下限:
                    <span id="contentLengthCommentLimit"></span>
                    <button style="margin: 5px;" onclick="BLH_lengthLimit('contentLengthCommentLimit');">變更</button>
                </div>
            </div>
        </div>
    </div>
    <div style="width: 100%; margin: 10px; padding: 10px; border: 1px solid; border-radius: 10px; background: antiquewhite;">
        部分符合關鍵詞封鎖(不分大小寫)<br>
        當設定在此處,文章或留言只要部分符合這個內容就會被過濾。<br>
        當設定在全域變數中,下方就不需再設定同樣的字詞(系統會體醒已在全域過濾清單中有)。<br>
        當文章與留言都有相同的變數,會自動改到全域變數中。
        <div style="display: flex; align-items: center; justify-content: center; flex-flow: row wrap;">
            <div style="border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px; width: 90%; background: beige;">
                <div style="display: flex; align-items: center; justify-content: center; flex-flow: row wrap;">
                    全域過濾關鍵詞(文章與留言皆會過濾)
                    <button style="margin: 5px;" onclick="BLH_addFilter('blockKeywordsPC');">新增</button>
                    <button style="margin: 5px;" onclick="BLH_removeFilter('blockKeywordsPC');">移除選取</button>
                </div>
                <div id="blockKeywordsPC" style="display: flex; flex-flow: row wrap; width: 90%; padding: 10px;">
                </div>
            </div>
            <div style="border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px; width: 45%; background: beige;">
                <div style="display: flex; align-items: center; justify-content: center; flex-flow: row wrap;">
                    文章過濾關鍵詞
                    <button style="margin: 5px;" onclick="BLH_addFilter('postBlockKeywordsPC');">新增</button>
                    <button style="margin: 5px;" onclick="BLH_removeFilter('postBlockKeywordsPC');">移除選取</button>
                </div>
                <div id="postBlockKeywordsPC" style="display: flex; flex-flow: row wrap; width: 90%; padding: 10px;">
                </div>
            </div>
            <div style="border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px; width: 45%; background: beige;">
                <div style="display: flex; align-items: center; justify-content: center; flex-flow: row wrap;">
                    留言過濾關鍵詞
                    <button style="margin: 5px;" onclick="BLH_addFilter('commentBlockKeywordsPC');">新增</button>
                    <button style="margin: 5px;" onclick="BLH_removeFilter('commentBlockKeywordsPC');">移除選取</button>
                </div>
                <div id="commentBlockKeywordsPC" style="display: flex; flex-flow: row wrap; width: 90%; padding: 10px;">
                </div>
            </div>
        </div>
    </div>
    <div style="width: 100%; margin: 10px; padding: 10px; border: 1px solid; border-radius: 10px; background: antiquewhite;">
        完全符合關鍵詞封鎖(不分大小寫)<br>
        當設定在此處,文章或留言必須完全符合這個內容才會被過濾。<br>
        當設定在全域變數中,下方就不需再設定同樣的字詞(系統會體醒已在全域過濾清單中有)。<br>
        當文章與留言都有相同的變數,會自動改到全域變數中。
        <div style="display: flex; align-items: center; justify-content: center; flex-flow: row wrap;">
            <div style="border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px; width: 90%; background: beige;">
                <div style="display: flex; align-items: center; justify-content: center; flex-flow: row wrap;">
                    全域過濾關鍵詞(文章與留言皆會過濾)
                    <button style="margin: 5px;" onclick="BLH_addFilter('blockKeywordsFC');">新增</button>
                    <button style="margin: 5px;" onclick="BLH_removeFilter('blockKeywordsFC');">移除選取</button>
                </div>
                <div id="blockKeywordsFC" style="display: flex; flex-flow: row wrap; width: 90%; padding: 10px;">
                </div>
            </div>
            <div style="border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px; width: 45%; background: beige;">
                <div style="display: flex; align-items: center; justify-content: center; flex-flow: row wrap;">
                    文章過濾關鍵詞
                    <button style="margin: 5px;" onclick="BLH_addFilter('postBlockKeywordsFC');">新增</button>
                    <button style="margin: 5px;" onclick="BLH_removeFilter('postBlockKeywordsFC');">移除選取</button>
                </div>
                <div id="postBlockKeywordsFC" style="display: flex; flex-flow: row wrap; width: 90%; padding: 10px;">
                </div>
            </div>
            <div style="border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px; width: 45%; background: beige;">
                <div style="display: flex; align-items: center; justify-content: center; flex-flow: row wrap;">
                    留言過濾關鍵詞
                    <button style="margin: 5px;" onclick="BLH_addFilter('commentBlockKeywordsFC');">新增</button>
                    <button style="margin: 5px;" onclick="BLH_removeFilter('commentBlockKeywordsFC');">移除選取</button>
                </div>
                <div id="commentBlockKeywordsFC" style="display: flex; flex-flow: row wrap; width: 90%; padding: 10px;">
                </div>
            </div>
        </div>
    </div>
    <div style="width: 100%; margin: 10px; padding: 10px; border: 1px solid; border-radius: 10px; background: antiquewhite;">
        ID封鎖(不分大小寫)<button onclick="localStorage.removeItem('BHBlackList'); location.reload();">強制重獲取黑名單</button><br>
        黑名單會自動獲取已經被黑名單的列表,但若要手動添加,可於上方參數進行添加。<br>
        可以添加強制顯示、強制隱藏,這兩個參數會優先於黑名單列表。<br>
        ID只能在強制顯示與強制隱藏其中一個清單內,不會有重複(輸入時會被提醒)。<br>
        輸入的ID不分大小寫。
        <div style="display: flex; align-items: center; justify-content: center; flex-flow: row wrap;">
            <div style="border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px; width: 45%; background: beige;">
                <div style="display: flex; align-items: center; justify-content: center; flex-flow: row wrap;">
                    顯示名單
                    <button style="margin: 5px;" onclick="BLH_addFilter('forceShowList');">新增</button>
                    <button style="margin: 5px;" onclick="BLH_removeFilter('forceShowList');">移除選取</button>
                </div>
                <div id="forceShowList" style="display: flex; flex-flow: row wrap; width: 90%; padding: 10px;">
                </div>
            </div>
            <div style="border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px; width: 45%; background: beige;">
                <div style="display: flex; align-items: center; justify-content: center; flex-flow: row wrap;">
                    隱藏名單
                    <button style="margin: 5px;" onclick="BLH_addFilter('forceHideList');">新增</button>
                    <button style="margin: 5px;" onclick="BLH_removeFilter('forceHideList');">移除選取</button>
                </div>
                <div id="forceHideList" style="display: flex; flex-flow: row wrap; width: 90%; padding: 10px;">
                </div>
            </div>
        </div>
    </div>
    <div style="margin: 10px; padding: 10px; border: 1px solid; border-radius: 10px; background: antiquewhite;">
        其他設定
        <div style="display: flex; align-items: center; justify-content: center; flex-flow: row wrap;">
            <div style="border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px; background: beige;">
                備份與還原
                <div style="display: flex; align-items: center; border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px;">
                    備份:
                    <input type="text" readonly="" name="BLH_Setting_backupText">
                    <button style="margin: 5px;" onclick="document.getElementsByName('BLH_Setting_backupText')[0].value = localStorage.getItem('BLH_Setting');">手動獲取</button>
                </div>
                <div style="display: flex; align-items: center; justify-content: center; border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px;">
                    還原:
                    <button style="margin: 5px;" onclick="localStorage.setItem('BLH_Setting', window.prompt('請貼上設定檔:(將重新整理)')); location.reload();">還原</button>
                </div>
                <div style="display: flex; align-items: center; justify-content: center; border: 1px solid; padding: 5px; margin: 10px; border-radius: 10px;">
                    還原初始設定:
                    <button style="margin: 5px;" onclick="if(window.confirm('確定要還原初始設定?還原後將重新整理。')) (localStorage.removeItem('BLH_Setting'), location.reload()); ">還原</button>
                </div>
            </div>
        </div>
    </div>
    `);
        let setting = JSON.parse(localStorage.getItem("BLH_Setting"));
        for (let key in setting.switch) {
            document.getElementById(key).innerHTML = (setting.switch[key] ? '開' : '關');
            document.getElementById(key).parentNode.style.color = setting.switch[key] ? 'olive' : 'crimson';
        }
        for (let key in setting.lengthLimit) document.getElementById(key).innerHTML = setting.lengthLimit[key];
        for (let key in setting.data) {
            for (let i = 0; i < setting.data[key].length; i++) {
                jQuery('#' + key).append(`<div style="margin: 10px;" data-value="` + encodeURIComponent(setting.data[key][i]) + `"><input type="checkbox"><span onclick="jQuery(this).parent().find('input').prop('checked', !jQuery(this).parent().find('input').prop('checked'));">` + setting.data[key][i] + `</span></div>`);
            }
        }
    }

    jQuery('body').append('<script type="text/javascript">' + function BLH_switch(key) {
        let setting = JSON.parse(localStorage.getItem("BLH_Setting"));
        setting.switch[key] = !setting.switch[key];
        localStorage.setItem('BLH_Setting', JSON.stringify(setting));
        document.getElementById(key).innerHTML = (setting.switch[key] ? '開' : '關');
        document.getElementById(key).parentNode.style.color = setting.switch[key] ? 'olive' : 'crimson';
        console.log('Switch ' + key + ' to ' + setting.switch[key]);
    } + function BLH_lengthLimit(key) {
        let lengthLimit = window.prompt('請輸入不小於2之過濾字數,若輸入5將過濾4字,5字不過濾。');
        console.log('Input: ' + lengthLimit);
        if (lengthLimit == null) return false;
        else lengthLimit = parseInt(lengthLimit);
        if (isNaN(lengthLimit)) (window.alert('請輸入半形數字。'), console.log('Input value is not a number'));
        else if (lengthLimit < 2) (window.alert('數值不得小於2'), console.log('Input value less then 2!'));
        else {
            let setting = JSON.parse(localStorage.getItem("BLH_Setting"));
            setting.lengthLimit[key] = lengthLimit;
            localStorage.setItem('BLH_Setting', JSON.stringify(setting));
            document.getElementById(key).innerHTML = lengthLimit;
            console.log('Set ' + key + ' to ' + lengthLimit);
        }
    } + function BLH_addFilter(key) {
        let setting = JSON.parse(localStorage.getItem("BLH_Setting"));
        let response = window.prompt('請輸入欲過濾' + (key.includes('force') ? '使用者ID(不分大小寫,非英數字會被過濾)' : '詞(不分大小寫,空白會被過濾。)'));
        console.log('Input: ' + response);
        if (response != null) response = key.includes('force') ? response.replace(/[^a-zA-Z0-9]/g, '').toLowerCase() : response.replace(/\s/g, '').toLowerCase();
        else return false;
        if (response != '' && response != null) {
            let listInfo = {
                forceShowList: {
                    id: 'forceShowList',
                    type: 'ID',
                    name: 'ID封鎖白名單',
                    equal: ['forceHideList']
                }, forceHideList: {
                    id: 'forceHideList',
                    type: 'ID',
                    name: 'ID封鎖黑名單',
                    equal: ['forceShowList']
                }, blockKeywordsPC: {
                    id: 'blockKeywordsPC',
                    type: '關鍵詞',
                    name: '全域關鍵詞部分符合封鎖清單',
                    equal: ['postBlockKeywordsPC', 'commentBlockKeywordsPC']
                }, postBlockKeywordsPC: {
                    id: 'postBlockKeywordsPC',
                    type: '關鍵詞',
                    name: '文章關鍵詞部分符合封鎖清單',
                    partner: 'commentBlockKeywordsPC',
                    parent: 'blockKeywordsPC'
                }, commentBlockKeywordsPC: {
                    id: 'commentBlockKeywordsPC',
                    type: '關鍵詞',
                    name: '留言關鍵詞部分符合封鎖清單',
                    partner: 'postBlockKeywordsPC',
                    parent: 'blockKeywordsPC'
                }, blockKeywordsFC: {
                    id: 'blockKeywordsFC',
                    type: '關鍵詞',
                    name: '全域關鍵詞完全符合封鎖清單',
                    equal: ['postBlockKeywordsFC', 'commentBlockKeywordsFC']
                }, postBlockKeywordsFC: {
                    id: 'postBlockKeywordsFC',
                    type: '關鍵詞',
                    name: '文章關鍵詞完全符合封鎖清單',
                    partner: 'commentBlockKeywordsFC',
                    parent: 'blockKeywordsFC'
                }, commentBlockKeywordsFC: {
                    id: 'commentBlockKeywordsFC',
                    type: '關鍵詞',
                    name: '留言關鍵詞完全符合封鎖清單',
                    partner: 'postBlockKeywordsFC',
                    parent: 'blockKeywordsFC'
                }
            };
            // 當清單中已經有的時候就提示並中止
            if (setting.data[key].includes(response)) {
                window.alert('清單中已經有此' + listInfo[key].type);
                console.log('"' + response + '" already exist in ' + key + '.');
                return false;
            }
            // 檢查其他清單中是不是已經有重複的
            for (let listnum in listInfo[key].equal) {
                if (setting.data[listInfo[key].equal[listnum]].includes(response)) {
                    if (window.confirm(listInfo[listInfo[key].equal[listnum]].name + ' 中已經有此' + listInfo[listInfo[key].equal[listnum]].type + ',是否從清單中移除並加到這個清單?')) {
                        setting.data[listInfo[key].equal[listnum]].splice(setting.data[listInfo[key].equal[listnum]].findIndex(element => element == response), 1);
                        jQuery('#' + listInfo[key].equal[listnum] + ' [data-value="' + encodeURIComponent(response) + '"]').remove();
                        console.log('Removed "' + response + '" from ' + listInfo[key].equal[listnum] + ' because it is ready to add to ' + key + '.');
                    } else {
                        console.log('"' + response + '" already exist in ' + listInfo[key].equal[listnum] + '.');
                        return false;
                    }
                }
            }
            // 如果同位的另一個清單中有了就上放到父清單中
            if (listInfo[key].partner && setting.data[listInfo[key].partner].includes(response)) {
                setting.data[listInfo[key].partner].splice(setting.data[listInfo[key].partner].findIndex(element => element == response), 1);
                jQuery('#' + listInfo[key].partner + ' [data-value="' + encodeURIComponent(response) + '"]').remove();
                key = listInfo[key].parent;
                console.log('Removed "' + response + '" from ' + listInfo[key].partner + ' because it is ready to add to ' + key + '.');
            }
            // 開始加入過濾詞
            setting.data[key][setting.data[key].length] = response;
            localStorage.setItem('BLH_Setting', JSON.stringify(setting));
            jQuery('#' + key).append(`<div style="margin: 10px;" data-value="` + encodeURIComponent(response) + `"><input type="checkbox"><span onclick="jQuery(this).parent().find('input').prop('checked', !jQuery(this).parent().find('input').prop('checked'));">` + response + `</span></div>`);
            console.log('Added ' + response + ' to ' + key + '.');
        } else window.alert('輸入的內容錯誤。');
    } + function BLH_removeFilter(key) {
        let setting = JSON.parse(localStorage.getItem("BLH_Setting"));
        let removeList = [];
        jQuery('#' + key + ' input:checked').each((index, element) => {
            console.log('Removed ' + jQuery(element).parent().attr('data-value') + ' from ' + key);
            setting.data[key].splice(setting.data[key].findIndex(element => element == jQuery(element).parent().attr('data-value')), 1);
            jQuery(element).parent().remove();
        });
        localStorage.setItem('BLH_Setting', JSON.stringify(setting));
    } + '</script>');
})();