南加北加论坛强化脚本(凛+)

加强东南北+功能: 太多了请看功能简介

// ==UserScript==
// @name         南加北加论坛强化脚本(凛+)
// @version      91.1
// @description  加强东南北+功能: 太多了请看功能简介
// @author       遠坂凛
// @namespace    tousakarin
// @license      MIT
// @icon         
// @grant        unsafeWindow
// @grant        GM.getValue
// @grant        GM.setValue
// @grant        GM.deleteValue
// @grant        GM.listValues
// @grant        GM.notification
// @grant        GM.openInTab
// @grant        GM.registerMenuCommand
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_deleteValue
// @grant        GM_listValues
// @grant        GM_notification
// @grant        GM_openInTab
// @grant        GM_registerMenuCommand
// @match        *://*.east-plus.net/*
// @match        *://east-plus.net/*
// @match        *://*.south-plus.net/*
// @match        *://south-plus.net/*
// @match        *://*.south-plus.org/*
// @match        *://south-plus.org/*
// @match        *://*.white-plus.net/*
// @match        *://white-plus.net/*
// @match        *://*.north-plus.net/*
// @match        *://north-plus.net/*
// @match        *://*.level-plus.net/*
// @match        *://level-plus.net/*
// @match        *://*.soul-plus.net/*
// @match        *://soul-plus.net/*
// @match        *://*.snow-plus.net/*
// @match        *://snow-plus.net/*
// @match        *://*.spring-plus.net/*
// @match        *://spring-plus.net/*
// @match        *://*.summer-plus.net/*
// @match        *://summer-plus.net/*
// @match        *://*.blue-plus.net/*
// @match        *://blue-plus.net/*
// @match        *://*.imoutolove.me/*
// @match        *://imoutolove.me/*
// @run-at       document-start
// ==/UserScript==

/* jshint undef: true, unused: true */
/* jshint -W107, -W098, -W069 */
/* globals unsafeWindow, window, document, navigator, atob, btoa, location, localStorage, fetch, console, alert, confirm, prompt, setInterval, setTimeout, clearTimeout, FileReader, FormData, Blob, URL, URLSearchParams, TextEncoder, CompressionStream, DecompressionStream, Response, MutationObserver, DOMParser, GM, GM_getValue, GM_setValue, GM_deleteValue, GM_listValues, GM_notification, GM_openInTab, GM_registerMenuCommand */

const VERSION_MAJOR = 91;
const VERSION_FULL = 9101;
const VERSION_TEXT = '91.1';

function initGM() {
    let gmExists = false;
    try {
        if (typeof GM.getValue == 'function') {
            gmExists = true;
        }
    } catch (ignore) {}

    if (gmExists) {
        return {
            getValue: GM.getValue,
            setValue: GM.setValue,
            async deleteValue(key) {
                return await GM.deleteValue(key);
            },
            listValues: GM.listValues,
            notification: GM.notification,
            openInTab: GM.openInTab,
            registerMenuCommand: GM.registerMenuCommand
        };
    } else {
        return {
            getValue: GM_getValue,
            setValue: GM_setValue,
            async deleteValue(key) {
                return GM_deleteValue(key);
            },
            listValues: GM_listValues,
            notification: GM_notification,
            openInTab: GM_openInTab,
            registerMenuCommand: GM_registerMenuCommand
        };
    }
    
}
(function() {
"use strict";

const GM = initGM();

const INTRO_POST = document.location.origin + '/read.php?tid-2086932.html';

GM.registerMenuCommand('打开功能简介帖 (茶館)', () => GM.openInTab(INTRO_POST, false));
GM.registerMenuCommand('打开设置 (个人首页)', () => GM.openInTab(document.location.origin + '/u.php', false));

const MY_NAME_DISPLAY = 'Me';

const MAIN_CONFIG_KEY = 'my_config';
const DEFAULT_MAIN_CONFIG = {
    myUserHashId: null,
    myNickName: null,
    autoCheckReplyOption: true,
    enhanceSellFrame: true,
    buyRefreshFree: true,
    replyRefreshFree: true,
    enhancePageTitle: true,
    highlightMyself: true,
    heightReductionMode: false,
    heightReductionMode$LV2: false,
    showFloatingMessageIndicator: true,
    showFloatingStoreThreadButton: true,
    showFloatingWatchIndicator: true,
    hideWatchButtonIfEmpty: false,
    subCategoryInheritImageWallMode: true,
    dontFilterRequestReplyByUser: true,
    showInputLimit: true,
    hideDefaultUserpic: false,
    hideOtherUserpic: false,
    selfBypassHideUserpic: true,
    customUserBypassHideUserpic: true,
    stickyUserInfo: true,
    showExtendedUserInfo: true,
    showExtendedUserInfo$HP: false,
    hideIgnoreContentPost: false,
    disablePostCollapse$Self: true,
    treatAllEmojiTheSameWay: true,
    ignoreContentUseTextOnly: true,
    customUserBypassIgnoreList: true,
    customUserpicBypassIgnoreList: false,
    watchSkipIgnoreContent: true,
    customUserBypassWatchSkip: true,
    hideFilteredThread: false,
    requestThreadHighlightEnded: true,
    requestThreadShowExtraBounty: true,
    requestThreadUseHistoryData: true,
    siteAnnouncementSectionDefaultFolded: false,
    siteNoticeSectionDefaultFolded: false,
    showBackToTopButton: true,
    useCustomUserInfoPopup: true,
    hideInactivePinnedUsers: true,
    floatingShortcut: 'tr',
    textSize: 0,
    siteThemeDarkerSubjectLine: false,
    hideRedundantReSubjectLine: true,
    addQuickJumpReSubjectLine: true,
    threadListDefaultOpenNewPage: false,
    infiniteScrollReplaceURL: false,
    infiniteScrollScrollToNewPage: false,
    infiniteScroll$usertopics: false,
    infiniteScroll$userposts: false,
    infiniteScroll$threads: false,
    infiniteScroll$thread_posts: false,
    infiniteScroll$search: false,
    infiniteScroll$msg_inbox: false,
    showResourceSpotsFloating: true,
    showResourceSpots$sells: true,
    showResourceSpots$attachments: true,
    showResourceSpots$shares: true,
    showResourceSpots$images: true,
    showResourceSpots$links: false,
    showResourceSpots$likes: false,
    showFavThreadFloatingList: true,
    showFavThreadFloatingList$withTitle: true,
    showActiveRepliers: false,
    showActiveRepliers$min: 2,
    showShareTypeFilter: true,
    hideSettlementPost: true,
    hideSettlementPost$GreyoutOnly: true,
    hideSettlementPost$UseDefaultKeywords: true,
    hideSettlementPost$HighlightMyself: true,
    customUserBypassThreadFilter: true,
    showRequestThreadFilters: true,
    customUserHashIdMappings: {
        '#4': [4, '1eb7ddbc', '可可萝'],
        '@1eb7ddbc': [4, '1eb7ddbc', '可可萝']
    },
    customUserOrderBy: 'uid',
    userReplyListFolded: false,
    replyShownAsByOp: false,
    hideZeroReply: false,
    keepVisitPostHistory: true,
    showInitialRememberedTitle: false,
    hideFrontPageRecentList: false,
    v15migrationApplied: false,
    v42migrationApplied: false,
    openIntroAfterUpdate: true,
    autoSpTasks: true,
    showDefaultingPicWallOption: true,
    hideMobileVerSwitch: true,
    showSearchBar: false,
    showSearchBar$align: null,
    hasSeenAdminRole: false,
    adminNoScoreNotifByDefault: false,
    adminHideMarkUnscoreButton: false,
    adminHideMarkBadFormatButton: false
};

const MIN_OVER_HEIGHT_STICKY_MODE_TRIGGER = 75;
const THREAD_FILTER_EXEMPTED_USERS = new Set([4, 168153]); // 可可萝(站长), 平安魂加(总版主)

const MIN_REQUEST_INTERVAL = 1100;

const SYSTEM_MESSAGE_TITLES = new Set([
    '您的文章被评分',
    '您的文章被取消评分',
    '您的文章标题被加亮显示',
    '您的文章被置顶.',
    '您的回复被设为最佳答案!',
    '您的回复获得热心助人奖 ..'
]);

const DEFAULT_SCORING_PRESETS = {
    'fid=201': { // COS区: 自购30天 / 优秀30天
        headtopDaysBought: 30,
        headtopDaysCompilation: 30,
        extraScoreAdjustments: [
            { label: '+200', amount: 200, reason: '第一类盘奖励' }
        ]
    },
    'fid=*': { // 各区默认: 自购7天 / 优秀30天
        headtopDaysBought: 7,
        headtopDaysCompilation: 30,
        extraScoreAdjustments: []
    }
};

const SCORE_DIFF_ALLOWANCE = {
    'fid=201': [220, 220],  // COS区
    'fid=*': [10, 100]     // 各区默认
};

const SEARCH_CONFIG_KEY = 'my_search_pref';
const DEFAULT_SEARCH_CONFIG = {
    defaultSearchAll: false,
    defaultTimeRange: '31536000',
    pinnedTopics: {}
};

const THREAD_CUSTOM_CATEGORY_CONFIG_KEY = 'my_thread_categories';
const DEFAULT_THREAD_CUSTOM_CATEGORY_CONFIG = {
    keywords: []
};

const THREAD_FILTER_CONFIG_KEY = 'my_thread_filter';
const DEFAULT_THREAD_FILTER_CONFIG = {
    dislikes: [],
    likes: [],
    settlementKeywords: []
};

const CONTENT_IGNORE_LIST_CONFIG_KEY = 'my_ignore_list';
const DEFAULT_CONTENT_IGNORE_LIST_CONFIG = {
    terms: [],
    exceptions: []
};

const USER_FILTER_CONFIG_KEY = 'my_user_filter';
const DEFAULT_USER_FILTER_CONFIG = {
    users: {}
};

const PINNED_USERS_CONFIG_KEY = 'my_pinned_users';
const DEFAULT_PINNED_USERS_CONFIG = {
    users: {}
};

const SHARETYPE_FILTER_CONFIG_KEY = 'my_sharetype_filter';
const DEFAULT_SHARETYPE_FILTER_CONFIG = {
    hides: []
};

const FAVOR_THREADS_CACHE_CONFIG_KEY = 'my_favor_threads';
const DEFAULT_FAVOR_THREADS_CACHE_CONFIG = {
    time: 0,
    tids: []
};

const QUESTION_AND_REQUEST_AREA_ID = 48;
const DEFAULT_IGNORABLE_MARKER_TAG = '<IGNORABLE>';

const DEFAULT_SETTLEMENT_STOPWORD_PATTERN = /[谢謝][谢謝]|多[谢謝]|大佬|老哥|兄弟|[請请][进進]?|[给給]|收|用户/g;
const DEFAULT_SETTLEMENT_KEYWORD_PATTERN = /^[奖獎][励勵][贴帖]|感[谢謝]|[结結][账賬]|[转轉][账賬]|[约約]定[贴帖]?|[請请][进進]|[结結][贴帖]|[结結]算[贴帖]?|^另?[结結]|[补補][悬懸][赏賞]|[补補]SP|[热熱]心助人|[热熱]心/g;
const DEFAULT_SETTLEMENT_BLACKLIST_PATTERN = /^求(?!物)|资源|出处|视频|合集|完整版/;
const DEFAULT_SETTLEMENT_TITLE_MAX_OTHER_TEXT_AMOUNT = 10;

const ADMIN_TEMPLATE_CONFIG_KEY = 'my_admin_template';
const DEFAULT_ADMIN_TEMPLATEUSER_FILTER_CONFIG = {
    pingReasons: [
        '请注意标题格式',
        '标题请注明大小',
        '标题禁用【】《》符号 敬请注意',
        '转帖不评分',
        '评分前已失效',
        '-----',
        '禁止盗链转存',
        '禁止净放BT',
        '-----',
        '广告帖',
        '政治贴',
        '恶意灌水',
        '无意义话题',
        '与版规不符',
        '重复话题',
        '重复发帖',
        '询问求物',
        '分类违规',
        '-----',
        '优秀文章',
        '原创内容',
    ]
};

const PUNISH_TYPE_BAN = '禁言';
const PUNISH_TYPE_HP = '扣血';
const PUNISH_TYPE_DELETE_THREAD = '删帖';
const PUNISH_TYPE_DELETE_REPLY = '删除回复';
const PUNISH_TYPE_SP = '扣分';
const PUNISH_TYPE_SHIELD = '屏蔽';
const PUNISH_TYPE_UNSHIELD = '取消屏蔽';
const PUNISH_TYPE_CUSTOM = '备注';
const PUNISH_TYPES = [
    PUNISH_TYPE_BAN,
    PUNISH_TYPE_HP,
    PUNISH_TYPE_DELETE_THREAD,
    PUNISH_TYPE_DELETE_REPLY,
    PUNISH_TYPE_SP,
    PUNISH_TYPE_SHIELD,
    PUNISH_TYPE_UNSHIELD,
    PUNISH_TYPE_CUSTOM
];

const REQUEST_ZONE_NAME = '询问&求物';
const NEVER_SCORE_USER_PREFIX = '[不评分]';
const TRUSTED_SCORE_USER_PREFIX = '[OK]';
const SYSTEM_SENDER_DISPLAY_NAME = 'SYSTEM';

const customCss = `
:root {
    --rinsp-active-toggler-bg-color: #EEEE;
    --rinsp-active-toggler-fg-color: #333;
    --rinsp-visited-link-color: #AAA;
    --rinsp-visited-link-hover-color: #333;
    --rinsp-visited-update-color: #7628A2;
    --rinsp-visited-update-border-color: #CAB2D7;
    --rinsp-blocked-row-bg: #F9F9F9;
    --rinsp-blocked-row-label-color: #AAA;
    --rinsp-avatar-replace-default-text-color: #AAA;
    --rinsp-avatar-replace-default-border-color: #EEE;
    --rinsp-avatar-replace-default-bg: #F8F8F8;
    --rinsp-cell-background-grey-out: #EEE;
    --rinsp-cell-background-hgt-new: #FFFFC5;
    --rinsp-cell-background-hgt-err: #FFC5C5;
    --rinsp-cell-background-active: #e4f2f7;
    --rinsp-lightest-bg-color: #FCFCFC;
    --rinsp-darkest-text-color: #000;
    --rinsp-text-color-orange: ##ffa500;
    --rinsp-text-color-green: #06a426;
    --rinsp-text-color-red: #ea2a2a;
    --rinsp-text-color-blue: #161884;
    --rinsp-text-color-violet: #8c19ac;
    --rinsp-text-color-grey: #999;
    --rinsp-infscroll-height: 55px;
    --rinsp-infscroll-divider-text-color: #3a6776;
    --rinsp-infscroll-divider-border-color: #a3b7b8;
    --rinsp-infscroll-divider-background: #ebf6f7;
    --rinsp-infscroll-loader-border-color: #CCC;
    --rinsp-infscroll-loader-background: #F5F5F5;
    --rinsp-infscroll-loader-progress: #EEE;
    --rinsp-image-masking-filter: contrast(0%) brightness(200%) drop-shadow(0px 0px 1px #000);
    --rinsp-pin-icon: url();
}
#mainNav #user-login {
    white-space: nowrap;
}
#mainNav #user-login a[href="u.php"] {
    max-width: 8em;
    overflow: hidden;
    text-overflow: ellipsis;
    display: inline-block;
    white-space: nowrap;
    vertical-align: middle;
}
#guide.guide {
    padding: 0;
}

.rinsp-intro-link {
    margin-left: 7px;
    color: #288407;
    font-weight: bold;
}
.rinsp-dev-toggle {
    opacity: 0;
    border: none;
    outline: none !important;
}
.rinsp-dev-toggle:focus {
    opacity: 0.1;
}

/* === message box add alternating background === */
form[action="message.php"] .set-table2 > tbody > tr:NOT(.gray3):nth-child(odd) {
    background-color: #b4d8e524;
}
form[action="message.php"]:not([onsubmit="return checkCnt();"]) .set-table2 > tbody > tr:NOT(.gray3):hover {
    background-color: #ead67924;
}

/* ====== truncate long post title in crumbs ====== */
.crumbs-item.current {
    max-width: 55%;
}
.crumbs-item.current strong a {
    max-width: 100%;
    overflow: hidden;
    text-overflow: ellipsis;
    display: inline-block;
    white-space: nowrap;
    vertical-align: top;
}

/* ====== watcher function ====== */
#rinsp-watcher-menu {
    margin-left: 40px;
}
.rinsp-common-popup-menu > .bor {
    max-height: calc(90vh - 100px);
    overflow: auto;
    overflow-x: hidden;
}
.rinsp-notification-item.rinsp-notification-item-type-watch > .rinsp-notification-item-count {
    color: #ee9e05;
}
.rinsp-excontrol-item-watch {
    color: #999;
    --rinsp-excontrol-base-hover-color: #f06629;
    --rinsp-excontrol-active-color: #aa0920;
    --rinsp-excontrol-active-hover-color: #000;
}

.new-msg-tips,
.new-msg-tips + span {
    transform: translateX(-55px);
}
.readbot li.rinsp-watch {
    width: 7em;
}
.readbot li.rinsp-watch::after {
    content: "未开启";
    color: #BBB;
}
.readbot li.rinsp-watch:hover::after {
    color: var(--rinsp-darkest-text-color);
}
.readbot li.rinsp-running {
    pointer-events: none;
    cursor: wait;
}
.readbot li.rinsp-running::after {
    content: "工作中";
    color: var(--rinsp-text-color-orange);
    font-weight: bold;
}

.readbot li.rinsp-watch.rinsp-active::after {
    content: "已开启";
    color: var(--rinsp-text-color-green);
    font-weight: bold;
}
tr.rinsp-expired > td {
    background: var(--rinsp-cell-background-grey-out);
}
tr.rinsp-expired:hover > td {
    opacity: 0.5;
}
tr.rinsp-expired > .rinsp-bounty-cell {
    color: var(--rinsp-text-color-red);
    font-weight: bold;
}
tr.rinsp-new > td {
    background: var(--rinsp-cell-background-hgt-new);
}
tr.rinsp-error > td {
    background: var(--rinsp-cell-background-hgt-err);
}
tr.rinsp-new > td.rinsp-status-cell {
    font-weight: bold;
    font-size: 110%;
    color: var(--rinsp-text-color-green);
}
tr.rinsp-ignorable > td.rinsp-status-cell {
    color: var(--rinsp-text-color-blue);
}
tr:not(.rinsp-new):not(.rinsp-error):not(.rinsp-ignorable) > td.rinsp-status-cell {
    font-size: 90%;
    color: #666;
}

td.rinsp-action-bar {
    text-align: left;
}
.rinsp-check-now {
    cursor: pointer;
    padding: 1px 5px;
    display: inline-block;
}
.rinsp-check-now:hover {
    background: #DDD;
}
html.rinsp-dark-mode .rinsp-check-now:hover {
    background: var(--rinsp-dm-color-darkbg-hover);
}
.rinsp-limit-warning {
    color: red;
    font-weight: bold;
    font-size: 1.1em;
    background: #fffa6a;
    padding: 1px 5px;
    display: inline-block;
    float: right;
}
.rinsp-notification-container {
    position: fixed;
    left: calc(50vw - 515px);
    top: 100px;
    display: flex;
    flex-direction: column;
    flex-wrap: wrap;
    z-index: 2;
}
.rinsp-tpcontrol-container,
.rinsp-bmcontrol-container,
.rinsp-excontrol-container {
    position: fixed;
    left: calc(50vw + 475px);
    display: flex;
    flex-direction: column;
    flex-wrap: wrap;
}
.rinsp-excontrol-container {
    top: 50vh;
    transform: translateY(-50%);
}
.rinsp-tpcontrol-container {
    top: 100px;
}
.rinsp-tpcontrol-container .rinsp-excontrol-item {
    width: 1.3em;
    text-align: center;
    padding: 1px 2px;
}
.rinsp-bmcontrol-container {
    bottom: 12px;
}
@media only screen and (max-width: 1024px) {
    .rinsp-notification-container {
        left: 9px;
    }
    .rinsp-tpcontrol-container,
    .rinsp-bmcontrol-container,
    .rinsp-excontrol-container {
        left: unset;
        right: 9px;
    }
}
.rinsp-excontrol-item {
    color: #999;
    padding: 1px 4px;
    background: #333C;
    box-shadow: 2px 2px 1px 0 #0009;
    white-space: pre;
    text-align: center;
    font-size: 14px;
    line-height: 18px;
    flex: 0;
    margin: 8px 0;
    cursor: pointer;
}

.rinsp-excontrol-item:hover {
    text-decoration: none;
}
.rinsp-excontrol-item-hidden {
    display: none;
}
.rinsp-excontrol-item:not(:hover) {
    opacity: 0.75;
}
.rinsp-excontrol-item.rinsp-excontrol-item-ticker:not(hover) {
    opacity: 0.9;
}
.rinsp-excontrol-item.rinsp-excontrol-item-ticker:hover {
    outline: 3px solid var(--rinsp-excontrol-base-hover-color);
}
.rinsp-excontrol-item.rinsp-excontrol-item-ticker:not(.rinsp-active) {
    margin-bottom: 24px;
}
.rinsp-excontrol-item.rinsp-excontrol-item-ticker.rinsp-active {
    outline: 3px solid var(--rinsp-excontrol-active-color);
    color: var(--rinsp-active-toggler-fg-color);
    font-weight: bold;
    background: var(--rinsp-active-toggler-bg-color);
}
.rinsp-excontrol-item.rinsp-excontrol-item-ticker.rinsp-active:hover {
    outline: 3px solid var(--rinsp-excontrol-active-hover-color);
}
.rinsp-excontrol-item.rinsp-excontrol-item-ticker.rinsp-active:after {
    content: '✔';
    display: block;
    height: 16px;
    overflow: hidden;
    font-size: 12px;
    color: var(--rinsp-excontrol-active-color);
}
.rinsp-excontrol-item.rinsp-running {
    pointer-events: none;
    opacity: 0.3;
}

.rinsp-notification-item > .rinsp-notification-item-count {
    font-family: monospace;
    font-weight: bold;
    font-size: 17px;
}
.rinsp-excontrol-item > .rinsp-excontrol-item-count {
    font-family: monospace;
    font-weight: bold;
    font-size: 15px;
    margin: 0 -0.5em;
}
.rinsp-excontrol-item > .rinsp-excontrol-item-count-3d {
    font-size: 13px;
    transform: scaleX(0.85);
    display: inline-block;
}
.rinsp-excontrol-item:after {
    font-family: Arial, Helvetica, sans-serif;
    font-size: 12px;
    color: #999;
}

/* == scoring filter toggler == */
.rinsp-highlight-unscored-thread-toggler:hover,
.rinsp-highlight-unscored-thread-mode .rinsp-highlight-unscored-thread-toggler:hover {
    outline: 5px solid #ce0940;
}
.rinsp-highlight-unscored-thread-toggler:after {
    content: '\\a评\\a分\\a模\\a式';
}
.rinsp-highlight-unscored-thread-mode .rinsp-highlight-unscored-thread-toggler {
    background: var(--rinsp-active-toggler-bg-color);
    outline: 3px solid #ce0940;
    opacity: 1.0;
}

.rinsp-highlight-unscored-thread-mode .rinsp-highlight-unscored-thread-toggler:after {
    content: '\\a评\\a分\\a模\\a式';
    color: var(--rinsp-active-toggler-fg-color);
}
.rinsp-highlight-unscored-thread-toggler > .rinsp-excontrol-item-count {
    color: #ce0940;
}
.rinsp-highlight-unscored-thread-mode .rinsp-thread-filter-scored {
    display: none;
}
.rinsp-highlight-unscored-thread-mode .rinsp-thread-filter-scored + .tr3:not(.t_one) {
    /* this is a management action button row */
    display: none;
}
.rinsp-highlight-unscored-thread-mode .rinsp-thread-filter-miscored h3 + .gray.tpage {
    color: #ce2323;
    font-weight: bold;
}

.rinsp-highlight-unscored-thread-mode .rinsp-thread-filter-unscored > td:first-child {
    border-left: 4px solid #d90927;
}
.rinsp-highlight-unscored-thread-mode .rinsp-post-gf.rinsp-post-unscored {
    border-left: 4px solid #ce0940 !important;
}

.rinsp-excontrol-totop:hover {
    outline: 3px solid #CCC;
}

/* favor button */
.rinsp-excontrol-item-favor {
    color: #999;
    --rinsp-excontrol-base-hover-color: #f9c342;
    --rinsp-excontrol-active-color: #ce972d;
    --rinsp-excontrol-active-hover-color: #000;
}

/* == settlement ignore toggler == */
.rinsp-request-settlement-greyout-mode .rinsp-settlement-ignore-toggler {
    display: none;
}
.rinsp-settlement-ignore-toggler:hover,
.rinsp-settlement-peek-mode .rinsp-settlement-ignore-toggler:hover {
    outline: 5px solid #2d9126;
}

.rinsp-settlement-ignore-toggler > .rinsp-excontrol-item-count {
    color: #09b93d;
}
.rinsp-settlement-ignore-toggler:after {
    content: '\\a隐\\a藏\\a结\\a算';
}
.rinsp-settlement-peek-mode .rinsp-settlement-ignore-toggler {
    background: var(--rinsp-active-toggler-bg-color);
    outline: 3px solid #2d9126;
    opacity: 1.0;
}

.rinsp-settlement-peek-mode .rinsp-settlement-ignore-toggler:after {
    content: '\\a显\\a示\\a结\\a算';
    color: var(--rinsp-active-toggler-fg-color);
}

/* == content ignore toggler == */
.rinsp-content-ignore-toggler:hover,
.rinsp-filter-peek-mode .rinsp-content-ignore-toggler:hover {
    outline: 5px solid #8873ec;
}
.rinsp-content-ignore-toggler > .rinsp-excontrol-item-count {
    color: #8873ec;
}
.rinsp-content-ignore-toggler:after {
    content: '\\a屏\\a蔽\\a内\\a容';
}
.rinsp-filter-peek-mode .rinsp-content-ignore-toggler {
    background: var(--rinsp-active-toggler-bg-color);
    outline: 3px solid #8873ec;
    opacity: 1.0;
}

.rinsp-filter-peek-mode .rinsp-content-ignore-toggler:after {
    content: '\\a显\\a示\\a屏\\a蔽';
    color: var(--rinsp-active-toggler-fg-color);
}

/* == paywall ignore toggler == */
.rinsp-paywall-ignore-toggler:hover,
.rinsp-paywall-peek-mode .rinsp-paywall-ignore-toggler:hover {
    outline: 5px solid #FF40AC;
}
.rinsp-paywall-ignore-toggler > .rinsp-excontrol-item-count {
    color: #FF40AC;
}
.rinsp-paywall-ignore-toggler:after {
    content: '\\a网\\a赚\\a区';
}
.rinsp-paywall-peek-mode .rinsp-paywall-ignore-toggler {
    background: var(--rinsp-active-toggler-bg-color);
    outline: 3px solid #FF40AC;
    opacity: 1.0;
}

.rinsp-paywall-peek-mode .rinsp-paywall-ignore-toggler:after {
    color: var(--rinsp-active-toggler-fg-color);
}

/* == visited thread toggler == */
.rinsp-visited-thread-toggler:hover,
.rinsp-visited-thread-mask-mode .rinsp-visited-thread-toggler:hover {
    outline: 5px solid #AE54AE;
}
.rinsp-visited-thread-toggler > .rinsp-excontrol-item-count {
    color: #AE54AE;
}
.rinsp-visited-thread-toggler:after {
    content: '\\a已\\a读';
}
.rinsp-visited-thread-mask-mode .rinsp-visited-thread-toggler {
    background: var(--rinsp-active-toggler-bg-color);
    outline: 3px solid #AE54AE;
    opacity: 0.7;
}

.rinsp-visited-thread-mask-mode .rinsp-visited-thread-toggler:after {
    color: var(--rinsp-active-toggler-fg-color);
}

/* == closed thread toggler == */
.rinsp-closed-thread-toggler:hover,
.rinsp-closed-thread-view-mode .rinsp-closed-thread-toggler:hover {
    outline: 5px solid #AAA;
}
.rinsp-closed-thread-toggler > .rinsp-excontrol-item-count {
    color: #AAA;
}
.rinsp-closed-thread-toggler:after {
    content: '\\a已\\a关\\a闭';
}
.rinsp-closed-thread-view-mode .rinsp-closed-thread-toggler {
    background: var(--rinsp-active-toggler-bg-color);
    outline: 3px solid #999;
    color: #999;
    opacity: 0.8;
}

.rinsp-closed-thread-view-mode .rinsp-closed-thread-toggler:after {
    color: var(--rinsp-active-toggler-fg-color);
}

/* == floating notitication == */

.rinsp-notification-item {
    background: #FFFE;
    padding: 1px 3px;
    box-shadow: 2px 2px 1px 0 #9d9d9d;
    white-space: pre;
    text-align: center;
    font-weight: bold;
    animation-name: rinsp-watch-new-anim;
    animation-duration: 1s;
    animation-direction: alternate;
    animation-iteration-count: infinite;
    font-size: 14px;
    line-height: 20px;
    flex: 0;
    margin: 0.5em 0;
    cursor: pointer;
    display: none;
}
.rinsp-notification-item.rinsp-notification-item-type-pm {
    background-color: #e1feff;
    color: #354e6f;
    outline-color: #06c3cc !important;
}
.rinsp-notification-item:hover {
    animation: none;
    outline: 5px solid #f9c23c;
    text-decoration: none;
}
.rinsp-notification-item.rinsp-status-new,
.rinsp-notification-item.rinsp-status-error {
    display: block;
}
.rinsp-notification-item.rinsp-status-error {
    color: #b91a1a;
    outline-color: #b91a1a !important;
}
#user-login .rinsp-watch-menuitem {
    color: white;
    padding: 1px 3px;
    display: inline-block;
}
.rinsp-watchemnu-autohide .rinsp-watch-menuitem:not(.rinsp-status-enabled) {
    display: none !important;
}
.rinsp-watch-menuitem.rinsp-status-new {
    color: black;
    font-weight: bold;
    animation-name: rinsp-watch-new-anim;
    animation-duration: 1s;
    animation-direction: alternate;
    animation-iteration-count: infinite;
}
.rinsp-watch-menuitem.rinsp-status-new:not(.rinsp-checking)::before {
    content: attr(data-new-count) " ";
    color: #000;
    display: inline-block;
    font-family: monospace;
    font-size: 12px;
    font-weight: bold;
    border-radius: 16px;
    background: #f9c23c;
    padding: 0 0.3em;
    text-align: center;
    min-width: 8px;
    margin-right: 0.2em;
    text-shadow: 0px 0px 1px white, 0px 0px 2px white, 0px 0px 3px white, 0px 0px 4px white;
}
.rinsp-watch-menuitem.rinsp-checking {
    font-weight: normal;
}
.rinsp-watch-menuitem.rinsp-checking.rinsp-status-new {
    outline: 3px solid #f9c23c;
}

#rinsp-watcher-menu .rinsp-checking .rinsp-status-cell::before,
.rinsp-watch-menuitem.rinsp-checking::before {
    content: "⌛";
    color: white;
    display: inline-block;
    animation-name: rinsp-watch-checking-rotate-anim;
    animation-duration: 2s;
    animation-iteration-count: infinite;
}
.rinsp-reply-self > .rinsp-lastreply-floor {
    color: var(--rinsp-text-color-violet);
}
.readbot li.rinsp-running::after {
    content: "工作中";
    color: var(--rinsp-text-color-orange);
    font-weight: bold;
}

.rinsp-watch-menuitem.rinsp-status-error::after {
    content: "⚠️";
}
.rinsp-bounty-ended {
    color: var(--rinsp-text-color-red) !important;
    font-weight: bold;
}
.rinsp-bounty-answered {
    color: var(--rinsp-text-color-green) !important;
    font-weight: bold;
}
@keyframes rinsp-watch-new-anim {
    50% {
        outline: none;
    }
    51% {
        outline: 3px solid #f9c23c;
    }
    100% {
        outline: 3px solid #f9c23c;
    }
}
@keyframes rinsp-watch-checking-rotate-anim {
    0% {
        transform: rotate(0deg);
    }
    5% {
        transform: rotate(0deg);
    }
    45% {
        transform: rotate(180deg);
    }
    55% {
        transform: rotate(180deg);
    }
    95% {
        transform: rotate(360deg);
    }
    100% {
        transform: rotate(360deg);
    }
}


/* ====== sell-frame ====== */
h6.jumbotron .btn-danger {
    margin-left: 0.5em;
}
.jumbotron.rinsp-sell-free .s3 {
    color: #999;
}
.jumbotron .btn-danger {
    line-height: 16px !important;
}

.jumbotron.rinsp-sell-free .btn-danger {
    background: #516ba0;
}

.jumbotron.rinsp-sell-5 .btn-danger {
    background: #937210;
}
.jumbotron.rinsp-sell-100 .s3 {
    color: #DD0000;
}
.jumbotron.rinsp-sell-high .s3 {
    color: #DD0000;
    font-size: 1.2em;
    background: #fff9c7;
    outline: 2px solid #fff9c7;
}
.jumbotron.rinsp-sell-high .btn-danger {
    background: #DD0000;
}
.jumbotron.rinsp-sell-99999 .btn-danger {
    opacity: 0.2;
    pointer-events: none;
    background: #999;
}
.rinsp-sell-relay-button + .spp-buy-refresh-free,
.rinsp-sell-enhanced {
    max-width: 0;
    max-height: 0;
    padding: 0 !important;
    overflow: hidden;
    z-index: -1;
    visibility: hidden;
    position: absolute;
    pointer-events: none;
}
.rinsp-sell-buying {
    opacity: 0.5;
    pointer-events: none;
    cursor: wait;
}

.rinsp-att-enhanced.rinsp-att-sell > a:before {
    content: "「售价 " attr(data-price) "SP」";
    color: #000;
    font-weight: bold;
    font-size: 13px;
}
.rinsp-att-enhanced.rinsp-att-sell-high > a:before {
    content: "「售价 " attr(data-price) "SP - 高额出售 欢迎举报」";
    color: #DD0000;
    font-weight: bold;
    font-size: 14px;
}
.rinsp-att-enhanced.rinsp-att-sell > a {
    cursor: default;
}

.rinsp-buy-failed {
    background-color: #fffdcf;
    border-color: black;
    color: red;
    font-weight: bold;
}

/* ====== user display ====== */
th.r_two > .user-pic > table {
    position: relative;
    z-index: 2;
}
.rinsp-userframe-userinfo {
    margin-top: -50px;
    padding: 50px 9px 3px 9px;
    background: linear-gradient(to bottom, transparent 0%, #F7F7F7 100%);
    display: flex;
    flex-wrap: wrap;
    position: relative;
    z-index: 1;
    margin-left: -5px;
    max-width: 185px;
    box-sizing: border-box;
}
.rinsp-userframe-userinfo > dl {
    display: inline-flex;
    align-items: center;
    margin: 0;
}
.rinsp-userframe-userinfo > .rinsp-userframe-udata-spacer {
    margin: 0;
    flex-basis: 100%;
}
.rinsp-userframe-userinfo > dl > dt {
    color: var(--rinsp-text-color-blue);
    word-break: keep-all;
    margin: 0;
}
.rinsp-userframe-userinfo > dl > dd {
    padding-left: 0.5em;
    flex: 1;
    margin: 0;
}
.rinsp-userframe-udata-sp {
    flex-grow: 1;
}
.rinsp-userframe-udata-login {
    flex-grow: 1;
    text-align: right;
    font-size: 0.9em;
    color: #888;
}

.rinsp-userframe-udata-login-today {
    color: var(--rinsp-text-color-green);
}
.rinsp-userframe-udata-login-yesterday {
    color: var(--rinsp-text-color-violet);
}
.rinsp-userframe-udata-hp {
    color: #DD0000;
    font-weight: bold;
    margin-left: 6px;
    display: inline-block;
}
.rinsp-userframe-udata-online > dt {
    display: none;
}
.rinsp-userframe-udata-online > dd {
    color: #666;
}
.r_two.rinsp-userframe-unnamed > .rinsp-userframe-userinfo + div > a {
    color: #555;
}

/* ===== user popup action items ===== */
.rinsp-user-popup-action-mailto > a > div {
    width: 16px;
    height: 16px;
    background: url(/images/colorImagination/mail-icon.gif);
    background-repeat: no-repeat;
    background-position: 1px 3px;
}
.rinsp-user-popup-action-topics > a > div {
    width: 16px;
    height: 16px;
    background: url();
}
.rinsp-user-action-pinuser-icon,
.rinsp-user-popup-action-pinuser > a > div {
    width: 16px;
    height: 16px;
    background: var(--rinsp-pin-icon);
}
.rinsp-user-action-pinuser-icon {
    display: inline-block;
}
.rinsp-user-popup-action-list > li:hover > a > div {
    outline: 1px solid #DDD;
    outline-offset: 1px;
    box-shadow: 2px 1px 3px #DDD;
}


/* ===== user names ===== */
span.rinsp-nickname-byother {
    color: #4a4a4a;
    font-weight: bold;
}
span.rinsp-nickname-byowner {
    color: #160;
    font-weight: bold;
}
span.rinsp-nickname-byme {
    color: #c60e87;
    font-weight: bold;
}
.rinsp-nickname-bypinned {
    color: var(--rinsp-darkest-text-color);
    font-weight: bold;
}
.rinsp-nickname-bypinned:before {
    content: "";
    display: inline-block;
    width: 16px;
    height: 16px;
    background: var(--rinsp-pin-icon);
}
a.rinsp-owner-isme {
    color: #c60e87;
    font-weight: bold;
}
a.rinsp-owner-isknown {
    font-weight: bold;
}
.rinsp-byop-noreply-hide-mode td:not(:hover) > .rinsp-nickname-byop-only {
    visibility: hidden;
}

/* ====== config ====== */
.rinsp-config-panel.rinsp-config-saving {
    cursor: wait;
}
.rinsp-config-panel.rinsp-config-saving .rinsp-config-list {
    opacity: 0.5;
    pointer-events: none;
}
.rinsp-config-list {
    margin: 0 5px;
}
.rinsp-config-list > dt {
    margin-top: 5px;
}
.rinsp-config-list > dt > a {
    margin-left: 3px;
    color: #0060df;
    cursor: pointer;
}
.rinsp-config-list > dt > a:hover {
    text-decoration: underline;
}
.rinsp-config-item-sep {
    margin-top: 0.5em;
    padding-top: 0.5em;
    border-top: 1px dotted #AAA;
}
.rinsp-config-item-tip {
    color: #105884;
    padding-left: 4px;
}
.rinsp-config-item-lv1 > span {
    margin-left: 3px;
}
.rinsp-config-item-lv2 {
    margin-left: 20px;
}
.rinsp-config-item-lv3 {
    margin-left: 40px;
}
.rinsp-config-item-lv1:not(.rinsp-config-item-checked) + .rinsp-config-item-lv2:not(.rinsp-config-item-tip),
.rinsp-config-item-lv1:not(.rinsp-config-item-checked) + .rinsp-config-item-lv2 + .rinsp-config-item-lv2:not(.rinsp-config-item-tip),
.rinsp-config-item-lv1:not(.rinsp-config-item-checked) + .rinsp-config-item-lv2 + .rinsp-config-item-lv2 + .rinsp-config-item-lv2:not(.rinsp-config-item-tip) {
    opacity: 0.2;
    pointer-events: none;
}
.rinsp-config-item-lv2:not(.rinsp-config-item-checked) + .rinsp-config-item-lv3:not(.rinsp-config-item-tip),
.rinsp-config-item-lv2:not(.rinsp-config-item-checked) + .rinsp-config-item-lv3 + .rinsp-config-item-lv3:not(.rinsp-config-item-tip),
.rinsp-config-item-lv2:not(.rinsp-config-item-checked) + .rinsp-config-item-lv3 + .rinsp-config-item-lv3 + .rinsp-config-item-lv3:not(.rinsp-config-item-tip) {
    opacity: 0.2;
    pointer-events: none;
}
.rinsp-user-map-icon {
    margin-left: 5px;
    cursor: pointer;
}
.rinsp-user-map-icon.rinsp-config-saving {
    opacity: 0.5;
    pointer-events: none;
}

.rinsp-user-map-icon::after {
    content: "➕收藏";
    font-weight: normal;
    font-size: 12px;
}
.rinsp-user-map-icon.rinsp-user-mapped::after {
    content: "🔖 " attr(rinsp-nickname);
    font-weight: normal;
    font-size: 12px;
    color: #c30a0a;
}
.rinsp-user-tag {
    margin-right: 6px;
    white-space: nowrap;
    min-width: 9em;
    display: inline-block;
    max-width: 9em;
    overflow: hidden;
    text-overflow: ellipsis;
}
.rinsp-user-tag-icon::before {
    content: "🔖 ";
    font-size: 14px;
    display: inline-block;
    cursor: pointer;
}
.rinsp-user-blacklist .rinsp-user-tag-icon::before {
    content: "🚫 ";
}

/* ====== search ====== */
.rinsp-fav-search-area-all {
    font-weight: bold;
    color: blue;
    background: aliceblue;
}
.rinsp-fav-search-area-group {
    color: darkred;
}
.rinsp-temp-disabled {
    pointer-events: none;
}
.rinsp-fav-search-area-pin {
    cursor: pointer;
}
.rinsp-fav-search-setdefault-range {
    cursor: pointer;
}
.rinsp-search-keyword-input {
    width: 100%;
}
.rinsp-fav-search-setdefault-mode {
    cursor: pointer;
    color: #635994;
    float: right;
}


/* ====== fix broken image display ====== */
th.r_two > .user-pic > table > tbody > tr > td {
    max-width: 175px;
    overflow: hidden;
    text-overflow: ellipsis;
}
th.r_two > .user-pic img {
    font-size: 9px;
    line-height: 16px;
    white-space: nowrap;
}


/* ====== compact mode ====== */
.rinsp-compact-mode th.r_two > .user-pic {
    margin-top: -6px;
}
.rinsp-compact-mode th.r_two > .user-pic img {
    max-width: 150px;
    max-height: 150px;
    object-fit: cover;
    object-position: top;
}
.rinsp-compact-mode th.r_two > .user-pic > table:hover img {
    transition: max-height 0.3s ease-out;
    transition-delay: 0.7s;
    max-height: 270px;
}
th.r_two > .user-pic {
    position: relative;
}
.rinsp-compact-mode .rinsp-post-userpic-tall .rinsp-userframe-pulldown:after {
    content: "";
    background-image: url();
    height: 13px;
    background-repeat: no-repeat;
    background-position: center center;
    display: block;
    width: 100%;
    opacity: 0.5;
    max-height: 13px;
    overflow: hidden;
}
.rinsp-compact-mode .rinsp-post-userpic-tall .user-pic > table:hover + .rinsp-userframe-pulldown:after {
    transition: all 0.2s ease-out;
    transition-delay: 0.7s;
    opacity: 0;
    max-height: 0;
}
.rinsp-compact-mode.rinsp-compact-mode-smaller th.r_two > .user-pic img {
    max-width: 120px;
    max-height: 120px;
}
.rinsp-compact-mode.rinsp-compact-mode-smaller th.r_two > .user-pic > table:hover img {
    transition: max-height 0.3s ease-out;
    transition-delay: 0.7s;
    max-width: 150px;
    max-height: 270px;
}
.rinsp-compact-mode th[id^="td_"] > .tiptop {
    margin-bottom: 1em;
}
.rinsp-compact-mode th[id] > .tpc_content {
    padding-bottom: 0.5em;
}
.rinsp-compact-mode .tr1.r_one > th > .tpc_content {
    padding-bottom: 0;
}
.rinsp-compact-mode .tr1.r_one > th > .tipad {
    margin-top: 1em;
}
.rinsp-filter-default-ignorable .user-pic ~ div[align="center"],
.rinsp-compact-mode th.r_two > .user-pic ~ div[align="center"] {
    position: relative;
    max-height: 3em;
    margin-top: -1em;
}
.rinsp-compact-mode .rinsp-userframe-userinfo {
    padding-top: 0.2em;
    margin-top: 0.1em;
    background: #FFFD;
    padding-bottom: 0em;
}

/* ====== ignore marks ====== */
.rinsp-textlist-description {
    margin: 9px 9px 0 9px;
    text-align: left;
    white-space: pre-wrap;
}
.rinsp-textlist-editor {
    width: calc(100% - 28px);
    margin: 9px;
    height: 50vh;
    max-height: 80vh;
}
.rinsp-textlist-popup-table.rinsp-config-saving {
    cursor: wait;
    opacity: 0.5;
    pointer-events: none;
}
.rinsp-post:not(:hover) .tiptop .rinsp-ignore-switch {
    visibility: hidden;
}
.tiptop .rinsp-ignore-switch:after {
    content: " | ";
}
.readbot li.rinsp-ignore-switch {
    width: 6em;
}
.rinsp-filter-ignored-bykeyword .rinsp-ignore-switch > a {
    opacity: 1;
    color: red;
}
.rinsp-filter-ignored .rinsp-ignore-switch > a:before {
    content: "🚫";
    opacity: 1;
}

.rinsp-ignore-switch.rinsp-config-saving {
    opacity: 0.5;
    pointer-events: none;AAAA
}

.rinsp-filter-ignored th.r_two,
.rinsp-filter-ignored .tpc_content {
    opacity: 0.3;
}
.rinsp-filter-ignored:not(.rinsp-filter-bypass) .tiptop > .fr,
.rinsp-filter-ignored:not(.rinsp-filter-bypass) .tr1.r_one {
    visibility: hidden;
}
.rinsp-filter-ignored:hover th.r_two,
.rinsp-filter-ignored:hover .tpc_content {
    opacity: 0.9;
}
.rinsp-filter-ignored:not(.rinsp-filter-bypass):hover .tiptop > .fr,
.rinsp-filter-ignored:not(.rinsp-filter-bypass):hover .tr1.r_one {
    visibility: visible;
}
.rinsp-filter-ignored .tipad .c + .fr.gray {
    visibility: visible;
}

body:not(.rinsp-filter-peek-mode) .rinsp-filter-ignored:not(.rinsp-filter-bypass) .r_one > .tiptop {
    height: 20px;
    line-height: 20px;
    margin-top: 0 !important;
    margin-bottom: 0 !important;
    border-bottom: none !important;
}
.rinsp-filter-ignored-range-end-summary {
    display: none;
}
body:not(.rinsp-filter-peek-mode) .rinsp-filter-ignored:not(.rinsp-filter-bypass) .r_one > .tiptop .rinsp-filter-ignored-range-end-summary {
    display: inline;
}
body:not(.rinsp-filter-peek-mode) .rinsp-filter-ignored:not(.rinsp-filter-bypass) .r_two:after {
    content: "屏蔽内容";
    display: block;
    font-size: 0.9em;
    text-align: center;
    padding-top: 3px;
}
body:not(.rinsp-filter-peek-mode) .rinsp-filter-ignored:not(.rinsp-filter-bypass) .r_two[rinsp-filter-group-size]:after {
    content: "屏蔽内容 ✕ " attr(rinsp-filter-group-size);
    color: var(--rinsp-darkest-text-color);
    font-size: 1.1em;
}

body:not(.rinsp-filter-peek-mode) .t2.t5[rinsp-filter-ignored-cont="1"],
body:not(.rinsp-filter-peek-mode).rinsp-filter-hide-mode .rinsp-filter-ignored:not(.rinsp-filter-bypass) {
    display: none;
}

body:not(.rinsp-filter-peek-mode) .rinsp-filter-ignored:not(.rinsp-filter-bypass) .r_two > .user-pic,
body:not(.rinsp-filter-peek-mode) .rinsp-filter-ignored:not(.rinsp-filter-bypass) .r_two > .user-pic ~ *,
body:not(.rinsp-filter-peek-mode) .rinsp-filter-ignored:not(.rinsp-filter-bypass) .r_one > .tiptop ~ *,
body:not(.rinsp-filter-peek-mode) .rinsp-filter-ignored:not(.rinsp-filter-bypass) .r_one > .tiptop > .fr,
body:not(.rinsp-filter-peek-mode) .rinsp-filter-ignored:not(.rinsp-filter-bypass) .r_one > .tiptop > .fl.bianji,
body:not(.rinsp-filter-peek-mode) .rinsp-filter-ignored:not(.rinsp-filter-bypass) tr.tr1.r_one {
    display: none !important;
}

.rinsp-dialog-modal-mask {
    position: fixed;
    top: 0;
    bottom: 0;
    left: 0;
    right: 0;
    background: #FFF8;
    z-index: 100;
}

input.rinsp-input-max-exceeded,
textarea.rinsp-input-max-exceeded {
    outline: 2px solid red !important;
}
.rinsp-input-max-hint {
    font-size: 10px;
    margin-left: 0.5em;
    background: #f5fafc;
    color: #255d99;
    padding: 1px 3px;
    display: inline-block;
}
textarea + .rinsp-input-max-hint {
    transform: translateY(-100%);
    margin-bottom: -20px;
    position: relative;
    top: -5px;
}
input[name="atc_title"] + .rinsp-input-max-hint {
    position: absolute;
}
.rinsp-input-max-hint.rinsp-input-max-exceeded {
    font-size: 11px;
    background: #fff;
    color: #c41212;
    font-weight: bold;
}

/* ====== thread list ====== */

.rinsp-thread-paywall {
    background: #f4f4f4;
}
.rinsp-thread-paywall > td > a[href^="thread.php?fid-"]:after {
    content: " (网赚)";
    color: #ff219e;
}
body:not(.rinsp-paywall-peek-mode) .rinsp-thread-paywall {
    display: none;
}
.rinsp-thread-filter-like-words,
.rinsp-thread-filter-dislike-words {
    display: inline-block;
    white-space: nowrap;
    cursor: pointer;
}
#subject_tpc .rinsp-thread-filter-like-words,
#subject_tpc .rinsp-thread-filter-dislike-words,
#subject_tpc .rinsp-thread-filter-menu-button {
    font-size: 12px;
    font-weight: normal;
}

.rinsp-thread-filter-like-words {
    color: var(--rinsp-text-color-green);
}
.rinsp-thread-filter-like-bytid .rinsp-thread-filter-like-words {
    font-weight: bold;
    font-size: 13px;
}
.rinsp-thread-filter-dislike-words {
    color: red;
}
.rinsp-thread-filter-like-words:hover,
.rinsp-thread-filter-dislike-words:hover {
    color: #666;
    text-decoration: underline;
}
.rinsp-dislike-thread-ignore-toggler:hover,
.rinsp-dislike-thread-peek-mode .rinsp-dislike-thread-ignore-toggler:hover {
    outline: 5px solid #ff9e36;
}
.rinsp-dislike-thread-ignore-toggler > .rinsp-excontrol-item-count {
    color: #ff9e36;
}
.rinsp-dislike-thread-ignore-toggler:after {
    content: '\\a屏\\a蔽\\a帖\\a子';
}
.rinsp-dislike-thread-peek-mode .rinsp-dislike-thread-ignore-toggler {
    background: var(--rinsp-active-toggler-bg-color);
    outline: 3px solid #ff9e36;
    opacity: 1.0;
}

.rinsp-dislike-thread-peek-mode .rinsp-dislike-thread-ignore-toggler:after {
    content: '\\a屏\\a蔽\\a帖\\a子';
    color: var(--rinsp-active-toggler-fg-color);
}
body.rinsp-dislike-thread-peek-mode .rinsp-thread-filter-masked-bysharetype,
body.rinsp-dislike-thread-peek-mode .rinsp-thread-filter-dislike {
    opacity: 0.3;
    background: #eee;
    transition: none;
}
body.rinsp-threadfilter-hide-mode:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .rinsp-thread-filter-dislike {
    display: none;
}
body.rinsp-dislike-thread-peek-mode .rinsp-thread-filter-masked-bysharetype:hover,
.rinsp-thread-filter-dislike.rinsp-thread-filter-bypass:hover,
body.rinsp-dislike-thread-peek-mode .rinsp-thread-filter-dislike:hover {
    opacity: 1.0;
    transition: 0.1s opacity ease-in;
}
.rinsp-thread-filter-menu-button {
    opacity: 0.5;
    display: inline-block;
    margin-left: 0.5em;
    white-space: nowrap;
    transition-delay: 0s;
    transition: none;
    cursor: default;
}
.rinsp-thread-filter-unhideop-button {
    margin-left: 0.25em;
    display: inline-block;
    cursor: default;
}
#wall .section-intro .rinsp-thread-filter-hideop-button {
    position: absolute;
    left: 0;
    background: white;
    padding: 10px 7px 0 7px;
    height: 100%;
    top: 0;
    box-sizing: border-box;
}
.rinsp-thread-filter-dislike-byuid .rinsp-uid-inspected {
    color: red;
}
.rinsp-thread-filter-dislike-byuid .rinsp-thread-filter-hideop-button {
    display: none;
}
.rinsp-alert-menu-button,
.rinsp-thread-filter-hideop-button,
.rinsp-thread-filter-dislike-button {
    color: var(--rinsp-text-color-red) !important;
}
.rinsp-thread-filter-like-button {
    color: var(--rinsp-text-color-green) !important;
}
.rinsp-thread-filter-liketid-button {
    color: var(--rinsp-text-color-blue) !important;
}
.rinsp-alert-menu-message {
    background: #FFD164;
    font-size: 16px;
    font-weight: bold;
}
.rinsp-thread-filter-menu-button {
    transform: translateY(1px);
    float: right;
}
h1#subject_tpc {
    margin-right: 12px;
}
#subject_tpc > .rinsp-thread-filter-menu-button {
    transform: translateY(4px);
}
.rinsp-thread-filter-menu-button:hover {
    opacity: 1;
}
.rinsp-thread-filter-menu-button:after {
    content: "";
    display: inline-block;
    width: 14px;
    height: 14px;
    background-repeat: no-repeat;
    /* svg is used to avoid the button text to be copied as part of the content */
    background-image: url("data:image/svg+xml,%3Csvg viewBox='0 0 14 14' xmlns='http://www.w3.org/2000/svg'%3E%3Cstyle%3Etext %7B font: 14px sans-serif; fill: %23666;%0A%7D%3C/style%3E%3Ctext x='3' y='11'%3E≡%3C/text%3E%3C/svg%3E");
}
.rinsp-thread-filter-menu-button:hover:after {
    outline: 1px solid #4e7597;
}
.rinsp-thread-selected > td[id] {
    outline: 3px solid #4e7597;
}
.rinsp-thread-filter-like .rinsp-thread-filter-like-button {
    display: none;
}

/* dislike wins over like, so hide like button if filtered out by dislike */

.rinsp-thread-filter-like:not(.rinsp-thread-filter-dislike) {
    background: linear-gradient(to right, var(--rinsp-text-color-green) 2px, #faffe9 3px, transparent 100%);
}
.rinsp-thread-filter-like:not(.rinsp-thread-filter-dislike):hover {
    background: linear-gradient(to right, var(--rinsp-text-color-green) 2px, #f2ffd6 3px, transparent 100%);
}

body:not(.rinsp-dislike-thread-peek-mode) .rinsp-thread-filter-masked-bysharetype,
body:not(.rinsp-dislike-thread-peek-mode) .rinsp-thread-filter-dislike {
    opacity: 0.5;
}
.t5.t2 > .rinsp-thread-filter-masked-bysharetype:hover,
.t5.t2 > .rinsp-thread-filter-dislike:hover {
    opacity: 1.0;
}

body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .rinsp-thread-filter-masked-bysharetype > th,
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .rinsp-thread-filter-masked-bysharetype > td,
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .rinsp-thread-filter-dislike:not(.rinsp-thread-filter-bypass) > th,
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .rinsp-thread-filter-dislike:not(.rinsp-thread-filter-bypass) > td {
    padding: 0;
    height: 7px;
    background: var(--rinsp-blocked-row-bg);
    line-height: 1px !important;
    overflow: hidden;
    color: transparent;
}
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .rinsp-thread-filter-masked-bysharetype > th,
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .rinsp-thread-filter-masked-bysharetype > td,
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .rinsp-thread-filter-dislike:not(.rinsp-thread-filter-bypass) > th,
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .rinsp-thread-filter-dislike:not(.rinsp-thread-filter-bypass) > td {
    font-size: 0;
}
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .rinsp-thread-filter-masked-bysharetype-collapse,
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .rinsp-thread-filter-masked-bysharetype > th > *,
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .rinsp-thread-filter-masked-bysharetype > td > *,
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .rinsp-thread-filter-dislike:not(.rinsp-thread-filter-bypass) > th > *,
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .rinsp-thread-filter-dislike:not(.rinsp-thread-filter-bypass) > td > * {
    display: none;
}
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .tr3.tac.rinsp-thread-filter-dislike:not(.rinsp-thread-filter-bypass) > th.y-style:after,
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .rinsp-thread-filter-dislike:not(.rinsp-thread-filter-bypass) > td[id^="td_"]:after {
    font-size: 11px;
    color: var(--rinsp-blocked-row-label-color);
    display: inline-block;
    line-height: normal;
    padding: 1px 0 1px 7px;
}
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .tr3.tac.rinsp-thread-filter-dislike-bytitle:not(.rinsp-thread-filter-bypass) > th.y-style:after,
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .rinsp-thread-filter-dislike-bytitle:not(.rinsp-thread-filter-bypass) > td[id^="td_"]:after {
    content: "主题已屏蔽";
}

body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .tr3.tac.rinsp-thread-filter-dislike-byuid:not(.rinsp-thread-filter-bypass) > th.y-style:after,
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .rinsp-thread-filter-dislike-byuid:not(.rinsp-thread-filter-bypass) > td[id^="td_"]:after {
    content: "楼主已屏蔽";
}
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .tr3.tac.rinsp-thread-filter-masked-bysharetype:not(.rinsp-dummy) > th.y-style:after,
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .rinsp-thread-filter-masked-bysharetype:not(.rinsp-dummy) > td[id^="td_"]:after {
    content: "不合适网盘";
    font-size: 11px;
    color: var(--rinsp-blocked-row-label-color);
    display: inline-block;
    line-height: normal;
    padding: 1px 0 1px 7px;
}
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .tr3.tac.rinsp-thread-filter-masked-bysharetype:not(.rinsp-dummy) > th.y-style[rinsp-sharetype-chain]:after,
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) .rinsp-thread-filter-masked-bysharetype:not(.rinsp-dummy) > td[id^="td_"][rinsp-sharetype-chain]:after {
    content: "不合适网盘 ✕ " attr(rinsp-sharetype-chain);
}

/* image wall mode */
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) #wall .rinsp-thread-filter-masked-bysharetype,
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) #wall .rinsp-thread-filter-dislike:not(.rinsp-thread-filter-bypass) {
    box-shadow: none;
}
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) #wall .rinsp-thread-filter-masked-bysharetype > .inner,
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) #wall .rinsp-thread-filter-dislike:not(.rinsp-thread-filter-bypass) > .inner {
    display: none;
}
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) #wall .rinsp-thread-filter-masked-bysharetype:before,
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) #wall .rinsp-thread-filter-dislike:not(.rinsp-thread-filter-bypass):before {
    font-size: 11px;
    color: var(--rinsp-blocked-row-label-color);
    display: block;
    text-align: center;
    line-height: normal;
    width: 1em;
}
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) #wall .rinsp-thread-filter-dislike-bytitle:not(.rinsp-thread-filter-bypass):before {
    content: "主题已屏蔽";
}
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) #wall .rinsp-thread-filter-dislike-byuid:not(.rinsp-thread-filter-bypass):before {
    content: "楼主已屏蔽";
}
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) #wall .rinsp-thread-filter-masked-bysharetype:not(.rinsp-dummy):before {
    content: "不合适网盘";
}
body:not(.rinsp-dislike-thread-peek-mode):not(.rinsp-highlight-unscored-thread-mode) #wall .rinsp-thread-filter-masked-bysharetype:not(.rinsp-dummy)[rinsp-sharetype-chain]:before {
    content: "不合适网盘 ✕ " attr(rinsp-sharetype-chain);
}

#wall .rinsp-thread-filter-like:not(.rinsp-thread-filter-dislike) {
    background: none;
    box-shadow: 0 0 2px 1px #5bbd37;
}
#wall .rinsp-thread-filter-like:not(.rinsp-thread-filter-dislike) .section-intro {
    background-color: #1d8232 !important;
}



.rinsp-nickname-list:empty:after,
.rinsp-user-blacklist:empty:after {
    content: "列表为空";
    color: #999;
    margin-left: 1em;
}

.dropdown-content.menu .rinsp-filter-block-menu-sep {
    border-top: 1px dotted #CCC;
    display: block;
}
.dropdown-content.menu a.rinsp-filter-block-menu-item {
    white-space: nowrap;
}
.dropdown > a.rinsp-filter-block-menu-item-active {
    color: var(--rinsp-text-color-red);
}
.dropdown-content.menu a.rinsp-filter-block-menu-item-active {
    padding-left: 0;
    color: var(--rinsp-text-color-red);
}
.dropdown-content.menu > .apeacemaker[data-active="true"] {
    color: var(--rinsp-text-color-red);
}

.rinsp-userpic-replace .user-pic:not(:hover) > table a > img {
    display: none;
}

.rinsp-userpic-replace .user-pic:not(:hover) a[href^="u.php?action-show-uid-"]::before {
    color: var(--rinsp-avatar-replace-default-text-color);
    text-align: center;
    display: block;
    white-space: nowrap;
    padding-right: 0.9em;
    box-sizing: border-box;
    line-height: 150px;
    width: 158px;
    height: 158px;
    border: 5px solid var(--rinsp-avatar-replace-default-border-color);
    background: var(--rinsp-avatar-replace-default-bg);
}

.rinsp-compact-mode.rinsp-compact-mode-smaller  .rinsp-userpic-replace .user-pic:not(:hover) a[href^="u.php?action-show-uid-"]::before {
    line-height: 120px;
    width: 128px;
    height: 128px;
    font-size: 0.9em;
}
.rinsp-userpic-replace-default .user-pic:not(:hover) td > a[href^="u.php?action-show-uid-"]::before {
    content: "👤\\FE0E默认头像";
}
.rinsp-userpic-replace-custom .user-pic:not(:hover) td > a[href^="u.php?action-show-uid-"]::before {
    content: "🎭\\FE0E屏蔽头像";
}

.rinsp-filter-default-ignorable .tiptop {
    margin: 0.25em 1em !important;
}

.rinsp-filter-default-ignorable:not(.rinsp-filter-ignored).rinsp-userpic-replace th.r_two {
    padding-top: 0.3em !important;
}

.rinsp-userframe-username {
    margin-top: -0.75em;
    margin-bottom: 1em;
}
.rinsp-userbookmark-takeover > a,
.rinsp-userframe-username > a > strong {
    display: inline-block;
    max-width: calc(100% - 48px);
}
.rinsp-userframe-userknown .rinsp-userbookmark > a {
    font-weight: bold;
}
.rinsp-userframe-userknown.rinsp-userframe-userrenamed .rinsp-userbookmark > a {
    color: #e81210;
}
.rinsp-userframe-userknown .rinsp-userbookmark-takeover + a {
    display: block;
}
.rinsp-userframe-userknown .rinsp-userframe-unnamed .rinsp-userbookmark + a {
    display: none;
}
.rinsp-userframe-username-sticky .rinsp-userbookmark > span:first-child:before,
.rinsp-userframe-username .rinsp-userbookmark > span:first-child:before {
    content: "➕";
    font-weight: normal;
    margin-left: -15px;
    margin-right: 2px;
    opactity: 0.8;
    visibility: hidden;
}
.rinsp-userframe-username-sticky .rinsp-userbookmark > span:first-child:hover:before,
.rinsp-userframe-username .rinsp-userbookmark > span:first-child:hover:before {
    opactity: 1;
}
.user-pic:hover + .rinsp-userframe-username .rinsp-userbookmark > span:first-child:before,
.rinsp-userpic-sticky:hover > .rinsp-userframe-username-sticky .rinsp-userbookmark > span:first-child:before,
.rinsp-userframe-username-sticky:hover .rinsp-userbookmark > span:first-child:before,
.rinsp-userframe-username:hover .rinsp-userbookmark > span:first-child:before {
    visibility: visible;
}
.rinsp-userframe-userknown .rinsp-userframe-username-sticky .rinsp-userbookmark > span:first-child:before,
.rinsp-userframe-userknown .rinsp-userframe-username .rinsp-userbookmark > span:first-child:before {
    content: "🔖";
    opactity: 1;
    visibility: visible;
}

/* ==== sticky user pic ==== */
th.r_two > .user-pic.rinsp-userpic-sticky {
    position: fixed;
    top: 0;
    width: 180px;
    z-index: 2;
    background-color: #EEEEEE;
}
th.r_two > .user-pic.rinsp-userpic-sticky .rinsp-userframe-username-sticky {
    display: block;
}

.rinsp-userframe-username-sticky {
    text-align: center;
    margin-left: -5px;
    padding-top: 2px;
    display: none;
}
.rinsp-user-pic-dummy {
    display: none;
}
.rinsp-userpic-sticky + .rinsp-user-pic-dummy {
    display: block;
}
.rinsp-userpic-sticky ~ .rinsp-userframe-username {
    opacity: 0;
}

/* ==== message selection ==== */
.rinsp-message-selection-panel {
    text-align: right;
    background: #F8F8F8;
    padding: 0.2em 1.4em;
    margin-top: -0.5em;
}
.rinsp-selection-sep,
.rinsp-selection-button {
    display: inline-block;
    margin-left: 1em;
    white-space: nowrap;
    cursor: default;
}
.rinsp-selection-button:hover {
    text-decoration: underline;
}

/* ==== announcement folding ==== */
.rinsp-top-announcement-row.rinsp-announcement-folded {
    background: linear-gradient(to bottom, white 0, #ddd 100%)
}
.rinsp-top-announcement-row.rinsp-announcement-folded h3 {
    display: none;
}
.rinsp-announcement-toggler {
    cursor: pointer;
    position: relative;
    top: 4px;
    margin-left: 2px;
}
.rinsp-announcement-row-folded {
    display: none;
}

/* ==== mentioning ==== */
.rinsp-thread-mention-me:not(.rinsp-request-thread-ended) {
    background: #ffeaf8 !important;
}
.rinsp-thread-mention-me:not(.rinsp-request-thread-ended) > td[id^="td_"] > h3 > a[id^="a_ajax_"] {
    color: #C60E87;
}
.rinsp-thread-mention-me:not(.rinsp-request-thread-ended) > td[id^="td_"] > h3 > a[id^="a_ajax_"]:before {
    content: "❤️";
    font-size: 12px;
    font-weight: bold;
    color: #C60E87;
    margin-right: 0.25em;
}

/* ==== request visibility toggler ==== */
.rinsp-answered-request-ignore-toggler:hover,
.rinsp-answered-request-hide-mode .rinsp-answered-request-ignore-toggler:hover {
    outline: 5px solid #3f9b15;
}
.rinsp-answered-request-ignore-toggler {
    background: #d2ddcdee;
    outline: 3px solid #407727;
    opacity: 1.0;
}
.rinsp-answered-request-ignore-toggler > .rinsp-excontrol-item-count {
    color: #3f9b15;
}
.rinsp-answered-request-ignore-toggler:after {
    content: '\\a有\\a答\\a案';
    color: var(--rinsp-active-toggler-fg-color);
}
.rinsp-answered-request-hide-mode .rinsp-answered-request-ignore-toggler {
    background: #333C;
    opacity: 0.75;
    outline: none;
    box-shadow: 2px 2px 1px 0 #0009;
}
.rinsp-answered-request-hide-mode .rinsp-answered-request-ignore-toggler:after {
    color: #999;
}
.rinsp-answered-request-hide-mode .rinsp-request-thread.rinsp-request-thread-ended:not(.rinsp-thread-byme):not(.rinsp-thread-mention-me) {
    display: none;
}


.rinsp-unanswered-request-ignore-toggler:hover,
.rinsp-unanswered-request-hide-mode .rinsp-unanswered-request-ignore-toggler:hover {
    outline: 5px solid #ce324f;
}
.rinsp-unanswered-request-ignore-toggler {
    background: #f0e8ebde;
    outline: 3px solid #ae3a50;
    opacity: 1.0;
}
.rinsp-unanswered-request-ignore-toggler > .rinsp-excontrol-item-count {
    color: #bd1d3b;
}
.rinsp-unanswered-request-ignore-toggler:after {
    content: '\\a悬\\a赏\\a中';
    color: var(--rinsp-active-toggler-fg-color);
}
.rinsp-unanswered-request-hide-mode .rinsp-unanswered-request-ignore-toggler {
    background: #333C;
    opacity: 0.75;
    outline: none;
    box-shadow: 2px 2px 1px 0 #0009;
}
.rinsp-unanswered-request-hide-mode .rinsp-unanswered-request-ignore-toggler:after {
    color: #999;
}
.rinsp-unanswered-request-hide-mode .rinsp-request-thread.rinsp-request-thread-ongoing:not(.rinsp-thread-byme):not(.rinsp-thread-mention-me) {
    display: none;
}


.rinsp-expired-request-ignore-toggler:hover,
.rinsp-expired-request-hide-mode .rinsp-expired-request-ignore-toggler:hover {
    outline: 5px solid #eee;
}
.rinsp-expired-request-ignore-toggler {
    background: #ccccccee;
    outline: 3px solid #828282;
    opacity: 1.0;
}
.rinsp-expired-request-ignore-toggler > .rinsp-excontrol-item-count {
    color: #5c5c5c;
}
.rinsp-expired-request-ignore-toggler:after {
    content: '\\a已\\a超\\a时';
    color: var(--rinsp-active-toggler-fg-color);
}
.rinsp-expired-request-hide-mode .rinsp-expired-request-ignore-toggler > .rinsp-excontrol-item-count {
    color: #aaa;
}
.rinsp-expired-request-hide-mode .rinsp-expired-request-ignore-toggler {
    background: #333C;
    opacity: 0.75;
    outline: none;
    box-shadow: 2px 2px 1px 0 #0009;
}
.rinsp-expired-request-hide-mode .rinsp-expired-request-ignore-toggler:after {
    color: #999;
}
.rinsp-expired-request-hide-mode .rinsp-request-thread.rinsp-request-thread-expired:not(.rinsp-thread-byme):not(.rinsp-thread-mention-me) {
    display: none;
}

/* ==== to top button ==== */
.rinsp-excontrol-totop {
    opacity: 100;
    transition: opacity 0.5s ease-in;
}
.rinsp-excontrol-totop::after {
    content: '▲\\a顶\\a部';
    font-size: 16px;
    line-height: 1.2em;
}
.rinsp-opacity-0 {
    opacity: 0 !important;
    transition: opacity 0.1s ease-in;
}

/* ==== closed threads ==== */
.rinsp-thread-filter-closed {
    background-color: #FEE;
}
.tr3.rinsp-thread-filter-closed:hover {
    background-color: #FEE;
}
.rinsp-thread-filter-closed > td[id^="td_"] {
    text-decoration: line-through;
    color: red;
}
.rinsp-closed-thread-mask-mode .rinsp-thread-filter-closed {
    display: none;
}

/* ==== visited threads ==== */
.rinsp-visited-thread-view-mode .rinsp-thread-visited:not(:hover) {
    opacity: 0.9;
}
.rinsp-visited-thread-view-mode.rinsp-uphp-post #u-contentmain .rinsp-thread-visited:not(:hover) th .gray,
.rinsp-visited-thread-view-mode.rinsp-uphp-topic #u-contentmain .rinsp-thread-visited:not(:hover) th .gray {
    opacity: 0.5;
}

.rinsp-visited-thread-view-mode .rinsp-thread-visited td[id^="td_"] h3 > a:not(:hover),
.rinsp-visited-thread-view-mode .rinsp-thread-visited th a[href^="read.php?tid-"]:not(:hover),
.rinsp-visited-thread-view-mode .rinsp-thread-visited th a[href^="job.php?action-topost-tid-"]:not(:hover),
.rinsp-visited-thread-view-mode .rinsp-thread-visited .inner > .section-title > a[href^="./read.php?tid-"]:not(:hover) {
    color: var(--rinsp-visited-link-color);
}
.rinsp-visited-thread-view-mode .rinsp-thread-visited td[id^="td_"] h3 > a:hover,
.rinsp-visited-thread-view-mode .rinsp-thread-visited th a[href^="read.php?tid-"]:hover,
.rinsp-visited-thread-view-mode .rinsp-thread-visited th a[href^="job.php?action-topost-tid-"]:hover,
.rinsp-visited-thread-view-mode .rinsp-thread-visited .inner > .section-title > a[href^="./read.php?tid-"]:hover {
    color: var(--rinsp-visited-link-hover-color);
}
.rinsp-visited-thread-view-mode .rinsp-thread-visited td[id^="td_"] h3 > a:not(:hover) > font,
.rinsp-visited-thread-view-mode .rinsp-thread-visited td[id^="td_"] h3 > a:not(:hover) > b > font,
.rinsp-visited-thread-view-mode .rinsp-thread-visited th a[href^="read.php?tid-"]:not(:hover) > font,
.rinsp-visited-thread-view-mode .rinsp-thread-visited th a[href^="read.php?tid-"]:not(:hover) > b > font,
.rinsp-visited-thread-view-mode .rinsp-thread-visited .inner > .section-title > a[href^="./read.php?tid-"]:not(:hover) > b > font,
.rinsp-visited-thread-view-mode .rinsp-thread-visited .inner > .section-title > a[href^="./read.php?tid-"]:not(:hover) > b > font {
    filter: saturate(0.7);
    opacity: 0.3;
}

/* on search results */
.rinsp-thread-visited-update td.smalltxt.y-style + .y-style {
    position: relative;
    color: var(--rinsp-visited-update-color);
}
.rinsp-thread-visited-update td.smalltxt.y-style + .y-style:before {
    content: "";
    display: inline-block;
    width: 3em;
    position: absolute;
    border: 1px solid var(--rinsp-visited-update-border-color);
    border-radius: 12px;
    left: 50%;
    margin-left: -1.5em;
    height: 1.2em;
    margin-top: -0.1em;
}

/* on thread listing */
.rinsp-thread-visited-update td.f10 > .s8 {
    border: 1px solid var(--rinsp-visited-update-border-color);
    color: var(--rinsp-visited-update-color);
    padding: 1px 3px;
    border-radius: 12px;
    margin-left: -4px;
}
/* thread wall mode */
.rinsp-thread-visited-update .inner > .section-text > span + span {
    color: var(--rinsp-visited-update-color) !important;
    font-weight: bold;
}

/* ==== request threads ==== */
.rinsp-request-highlight-ended-mode .rinsp-request-thread-ended:not(.rinsp-request-settlement-thread) > td[id^="td_"] > h3:before {
    content: "— 已有答案 —";
    font-weight: bold;
    color: #407727;
}
.rinsp-request-highlight-ended-mode .rinsp-request-thread-won:not(.rinsp-request-settlement-thread) > td[id^="td_"] > h3:before {
    content: "— ✓ 最佳 —";
    font-weight: bold;
    color: #407727;
}
.rinsp-request-highlight-ended-mode .rinsp-request-thread-won.rinsp-request-settlement-thread > td[id^="td_"] > h3:before {
    content: "— ✓ 已领取 —";
    font-weight: bold;
    color: #999;
}

.rinsp-request-bounty {
    color: #c60e87;
}
.rinsp-request-bounty + .s1 {
    display: none;
}
.rinsp-request-bounty .rinsp-extra-bounty {
    display: none;
}
.rinsp-request-show-extra-bounty .rinsp-request-bounty .rinsp-extra-bounty {
    display: inline;
}
.rinsp-request-thread-expired .rinsp-request-bounty {
    color: #999;
}
.rinsp-request-settlement-thread:not(:hover) {
    background: #EEE;
    opacity: 0.7;
}
.rinsp-request-settlement-thread .rinsp-request-bounty {
    color: #666;
    font-weight: normal;
}


body.rinsp-request-settlement-hide-mode:not(.rinsp-settlement-peek-mode) .rinsp-request-settlement-thread:not(.rinsp-request-settlement-bypass) > th,
body.rinsp-request-settlement-hide-mode:not(.rinsp-settlement-peek-mode) .rinsp-request-settlement-thread:not(.rinsp-request-settlement-bypass) > td {
    padding: 0;
    height: 7px;
    background: var(--rinsp-blocked-row-bg);
    line-height: 1px;
    overflow: hidden;
    color: transparent;
}
body.rinsp-request-settlement-hide-mode:not(.rinsp-settlement-peek-mode) .rinsp-request-settlement-thread:not(.rinsp-request-settlement-bypass) > th > *,
body.rinsp-request-settlement-hide-mode:not(.rinsp-settlement-peek-mode) .rinsp-request-settlement-thread:not(.rinsp-request-settlement-bypass) > td > * {
    display: none;
}
body.rinsp-request-settlement-hide-mode:not(.rinsp-settlement-peek-mode) .tr3.tac.rinsp-request-settlement-thread:not(.rinsp-request-settlement-bypass) > th.y-style:after,
body.rinsp-request-settlement-hide-mode:not(.rinsp-settlement-peek-mode) .rinsp-request-settlement-thread:not(.rinsp-request-settlement-bypass) > td[id^="td_"]:after {
    content: "结算帖";
    font-size: 12px;
    color: #AAA;
    display: inline-block;
    line-height: normal;
    padding: 1px 0 1px 7px;
}

/* ==== bookmarked user indications on message list ==== */
.rinsp-message-user-mapped > a::before {
    content: "🔖";
    position: absolute;
    transform: translateX(-100%);
    margin-left: -1px;
    font-weight: normal;
    font-size: 12px;
    color: #c30a0a;
}
.rinsp-message-user-pinned > a {
    font-weight: bold;
    
}
.rinsp-message-user-pinned > a:before {
    content: "";
    width: 16px;
    height: 16px;
    background: var(--rinsp-pin-icon);
    padding-right: 1px;
    position: absolute;
    transform: translateX(-100%);
}

/* ==== improve column width ratio in message list ==== */

form[name="del"][action="message.php"] .set-table2 > tbody > tr > td:nth-child(1) {
    width: 8%;
}
form[name="del"][action="message.php"] .set-table2 > tbody > tr > td:nth-child(2) {
    width: 15%;
}
form[name="del"][action="message.php"] .set-table2 > tbody > tr > td:nth-child(4) {
    width: 22%;
}

/* ==== topic and reply list filter shortcuts ==== */
.rinsp-uphp-topic #u-contentmain > table > tbody > tr + tr:hover,
.rinsp-uphp-post #u-contentmain > table > tbody > tr + tr:hover {
    outline: 1px solid #DDD;
}

.rinsp-uphp-trade #u-contentmain .rinsp-gf-link,
.rinsp-uphp-topic #u-contentmain .rinsp-gf-link,
.rinsp-uphp-post #u-contentmain .rinsp-gf-link {
    float: right;
    color: #BBB;
    white-space: nowrap;
}
.rinsp-uphp-trade #u-contentmain .rinsp-gf-link,
.rinsp-uphp-post #u-contentmain .rinsp-gf-link {
    margin-right: 5px;
    margin-left: 5px;
}
.rinsp-uphp-trade #u-contentmain .rinsp-gf-link:hover,
.rinsp-uphp-topic #u-contentmain .rinsp-gf-link:hover,
.rinsp-uphp-post #u-contentmain .rinsp-gf-link:hover {
    color: #000;
}

/* ==== reply list ==== */
.rinsp-reply-list-controls {
    color: #333;
    font-size: 11px;
    display: block;
    text-align: right;
    margin-right: -18%;
}

.rinsp-reply-fold-table {
    width: 100%;
    border-collapse: collapse;
    table-layout: fixed;
}
.rinsp-reply-fold-table > tbody > tr > td + td {
    width: 20%;
}
/* need to keep the old table for soul++ infinite scroll */
.rinsp-reply-fold-table + .u-table {
    display: none;
}

.rinsp-reply-fold-item {
    display: flex;
    align-items: flex-start;
}
.rinsp-reply-fold-item > a {
    flex-shrink: 1;
}
.rinsp-reply-fold-pages {
    max-width: 14em;
    min-width: 3em;
    display: flex;
    align-items: flex-start;
    padding-right: 7px;
    white-space: nowrap;
    font-size: 13px;
    overflow: hidden;
}
.rinsp-reply-fold-count {
    display: inline-block;
    color: #FFF;
    font-family: monospace;
    font-weight: bold;
    background-color: #AAA;
    padding: 0px 4px;
    margin-left: 0.5em;
    border-radius: 20px;
    line-height: normal;
    cursor: default;
    white-space: nowrap;
    position: relative;
    top: 1px;
}
.rinsp-reply-fold-remains {
    white-space: nowrap;
    overflow: hidden;
    text-overflow: ellipsis;
    font-family: monospace;
}
.rinsp-reply-fold-remains-start {
    flex-shrink: 1;
}
.rinsp-reply-fold-remains-end {
    flex-grow: 0;
    flex-shrink: 0;
}
.rinsp-reply-fold-remains > a {
    margin-left: 4px;
    color: #999;
    white-space: nowrap;
}
.rinsp-reply-fold-remains > a:hover {
    color: #333;
}

/* ==== thread history list ==== */
.rinsp-quick-filter-box {
    display: flex;
    align-items: center;
}
.rinsp-quick-filter-box > :first-child {
    min-width: 3.5em;
}
.rinsp-quick-filter-box + .rinsp-quick-filter-box {
    margin-top: 2px;
}
.rinsp-quick-filter-box > .rinsp-quick-filter-checkbox {
    margin-right: 0.5em;
    padding: 0px 6px 0px 4px;
    border-radius: 16px;
    color: #333;
}
.rinsp-quick-filter-box > .rinsp-quick-filter-bought-checkbox {
    background: #FBEFD9;
}
.rinsp-quick-filter-box > .rinsp-quick-filter-replied-checkbox {
    background: #CFE8FF;
}
.rinsp-quick-filter-box > .rinsp-quick-filter-notmine-checkbox {
    background: #FBEAF5;
}
html.rinsp-dark-mode .rinsp-quick-filter-box > .rinsp-quick-filter-bought-checkbox {
    background: #523706;
    color: var(--rinsp-text-color-orange);
}
html.rinsp-dark-mode .rinsp-quick-filter-box > .rinsp-quick-filter-replied-checkbox {
    background: #092947;
    color: var(--rinsp-text-color-blue);
}
html.rinsp-dark-mode .rinsp-quick-filter-box > .rinsp-quick-filter-notmine-checkbox {
    background: #3c1332;
    color: var(--rinsp-text-color-violet);
}

.rinsp-quick-filter-box > input {
    flex: 1;
}
.rinsp-thread-history-table {
    width: 100%;
}
.rinsp-quick-filter-box > input:focus,
.rinsp-table-filtered .rinsp-quick-filter-box > input {
    outline: 3px solid #c6ae30;
    outline-offset: -1px;
}
.rinsp-thread-status-icons {
    float: right;
    margin-left: 1em;
}
.rinsp-thread-replied-icon {
    color: #0779AA;
    cursor: default;
}
.rinsp-thread-bought-icon {
    color: #9D674C;
    cursor: default;
}
.rinsp-thread-byme > th > a[href^="read.php?tid"] {
    color: #C60E87;
}
.rinsp-thread-deleted > th > a[href^="read.php?tid"] {
    color: #C22;
    text-decoration: line-through;
}
.rinsp-thread-initial-title::before {
    content: "原始标题: ";
    color: #739;
}
.rinsp-thread-initial-title {
    color: var(--rinsp-text-color-grey);
    padding: 1px 0px 1px 5px;
    margin-left: -5px;
    background: var(--rinsp-lightest-bg-color);
}
.rinsp-thread-title-replaced + br {
    display: block;
    height: 0;
}

.rinsp-thread-history-table > tbody > tr:hover {
    outline: 1px solid #DDD;
}
.rinsp-thread-record-uid > a::before {
    content: "by ";
    font-size: 0.9em;
}
.rinsp-thread-record-uid {
    float: right;
}

.rinsp-thread-populate-button {
    color: #BB4411;
    margin-left: 0.5em;
}

/* ==== friends ==== */
.rinsp-uphp-friend #u-content .u-table td:nth-child(2) > a[href^="u.php?action-show-uid-"] {
    color: #AAA;
}
.rinsp-uphp-friend #u-content .u-table .rinsp-user-state-online > td:nth-child(2) > b {
    color: #11AA44;
}

/* ==== user bookmark listing action menu ==== */
#rinsp-userbookmark-action-menu .bor td {
    padding-left: 12px;
}
.rinsp-mailto-user-bookmark-button::before {
    content: "";
    width: 14px;
    height: 12px;
    display: inline-block;
    background: url(/images/colorImagination/mail-icon.gif);
    position: absolute;
    margin-left: -18px;
    margin-top: 4px;
}

/* ==== user profile page actions ==== */
#u-portrait ~ .bdbA > table > tbody > tr > td {
    padding-bottom: 0.5em;
}

/* ==== paywell area rename ==== */
.rinsp-area-scoped-item-171 a[href="thread.php?fid-171.html"]::after,
.rinsp-area-scoped-item-172 a[href="thread.php?fid-172.html"]::after,
.rinsp-area-scoped-item-173 a[href="thread.php?fid-173.html"]::after,
.rinsp-area-scoped-item-174 a[href="thread.php?fid-174.html"]::after {
    content: " (网赚)";
}

/* ==== area filter ==== */
.rinsp-area-scoped-item-hidden {
    display: none;
}

/* ==== share type filter ==== */
#ajaxtable .hthread {
    height: auto;
    display: flex;
}
#ajaxtable .hthread > .threadlist {
    position: relative;
    float: none;
    padding-right: 1em;
}
#ajaxtable .threadlist li.current a, .threadlist li.current a:hover {
    height: 18px;
}

.rinsp-sharetype-filter {
    flex: 1;
    padding-top: 0.2em;
    text-align: right;
}
.rinsp-sharetype-item {
    display: inline-block;
    font-size: 11px;
    font-weight: normal;
    line-height: 16px;
    border: 1px solid #CAB2D7;
    border-radius: 9px;
    padding: 1px 5px;
    background: #FFFE;
    margin-right: 0.5em;
    cursor: pointer;
    margin-bottom: 1px;
}
.rinsp-sharetype-item[disabled="1"] {
    opacity: 0.5;
    border-color: #CCC;
    color: #999;
}
.rinsp-sharetype-item[disabled="1"]::after {
    color: #999;
}
.rinsp-sharetype-item[count="0"] {
    display: none;
}
.rinsp-sharetype-item::after {
    content: " " attr(count);
    color: #7628A2;
    font-size: 10px;
}
.rinsp-sharetype-filter-clear {
    cursor: pointer;
}
.rinsp-sharetype-filter-clear[disabled="1"] {
    cursor: default;
    opacity: 0.1;
    pointer-events: none;
}

/* ==== category filter ==== */

.rinsp-category-filter {
    padding-top: 0.2em;
}
.rinsp-category-toggle-item {
    display: inline-block;
    line-height: 16px;
    color: #666;
    border: 1px solid #CCC;
    border-radius: 9px;
    padding: 1px 5px;
    background: #FFFE;
    margin-right: 0.5em;
    margin-bottom: 0.5em;
    cursor: pointer;
}
.rinsp-category-toggle-item.rinsp-category-active {
    border: 1px solid #613578;
    background: #CAB2D7;
    color: #000;
}
.rinsp-category-toggle-item::after {
    content: " " attr(count);
    color: #7628A2;
    font-size: 10px;
}
.rinsp-category-hide {
    display: none;
}

/* ==== history-based bounty status ==== */
.rinsp-thread-bounty-status {
    font-size: 0.9em;
    margin-right: 0.5em;
    white-space: nowrap;
}
.rinsp-thread-bounty-status-won > span,
.rinsp-thread-bounty-status-ended > span {
    display: none;
}
.rinsp-thread-bounty-status > span::after {
    content: "SP";
}
.rinsp-thread-bounty-status-expired.rinsp-thread-bounty-status-own {
    color: #b10b0b;
}
.rinsp-thread-bounty-status-ongoing {
    color: #c11986;
}
.rinsp-thread-bounty-status-ongoing.rinsp-thread-bounty-status-own::before {
    content: "悬赏中 ";
    font-weight: bold;
    font-size: 1.2em;
}
.rinsp-thread-bounty-status-expired.rinsp-thread-bounty-status-own::before {
    content: "⚠️已超时 ";
    font-size: 1.2em;
    font-weight: bold;
}
.rinsp-thread-bounty-status-ended::before,
.rinsp-thread-bounty-status-won::before {
    content: "✓";
    font-size: 1.2em;
    font-weight: bold;
}
.rinsp-thread-bounty-status-won,
.rinsp-thread-bounty-status-ended.rinsp-thread-bounty-status-own {
    color: #2d8e0d;
}
.rinsp-thread-bounty-status-ended {
    color: #136cbb;
}
.rinsp-thread-bounty-status-won::after {
    content: " 最佳";
}
.rinsp-thread-bounty-status-ended::after {
    content: " 有答案";
}
.rinsp-thread-bounty-status-ended.rinsp-thread-bounty-status-own::after {
    content: "";
}
.rinsp-thread-bounty-status-expired::after {
    content: " 已结束";
    color: #ae700a;
}
.rinsp-thread-bounty-status-expired.rinsp-thread-bounty-status-own::after {
    content: "";
}
.rinsp-thread-bounty-status-unknown {
    display: none;
}

/* ==== other ==== */
.rinsp-filter-multiip-focus-mode .t5.t2:has(> .rinsp-post:not(.rinsp-post-multi-uid-selected)) {
    display: none;
}
.rinsp-hide {
    display: none;
}
.tiptop > .fr.js-hidepuremark {
    position: relative;
    z-index: 1;
}
.js-puremark-content + .bianji > a:not(:hover) {
    color: #aaa;
}
/* override and take over forum default blocking mode */
form > .t5.t2[hidden] {
    display: block;
}
.rinsp-fid-search-button {
    margin-right: 3px;
}
.rinsp-fid-search-button:not(:hover) {
    opacity: 0.8;
}
.rinsp-fid-search-button:before {
    content: "";
    display: inline-block;
    background: url() no-repeat;
    background-size: 12px 12px;
    width: 12px;
    height: 12px;
    vertical-align: middle;
}
.rinsp-fid-search-button:hover:before {
    color: #007788;
}

/* ==== menu quick search ==== */
#guide.rinsp-quicksearch-added {
    display: flex;
}
#guide.rinsp-quicksearch-added > li {
    float: none;
    flex: 0 0 auto;
}
#guide.rinsp-quicksearch-align-center {
    width: 100%;
}
#guide.rinsp-quicksearch-align-center .rinsp-spacer {
    flex: 1 1;
}
.rinsp-quicksearch {
    display: flex;
    align-items: center;
    padding: 2px 4px 2px 2px;
    position: relative;
}
.rinsp-quicksearch-field {
    flex: 1;
    font-size: 12px;
    padding: 1px 22px 1px 5px;
    margin: 0px;
    border-width: 1px;
    border-radius: 0.5em;
    border-color: transparent;
    overflow: hidden;
    width: 14em;
    position: relative;
}
.rinsp-quicksearch-button {
    flex: 0;
    padding: 0 4px 0 6px;
    position: absolute;
    right: 0;
}
.rinsp-quicksearch-button:after {
    content: "";
    display: inline-block;
    cursor: pointer;
    background: url() no-repeat;
    background-size: 14px 14px;
    width: 18px;
    height: 14px;
}
.rinsp-quicksearch-field:invalid {
    opacity: 0.9;
}
.rinsp-quicksearch:has(.rinsp-quicksearch-field:invalid):after {
    content: "搜索";
    font-size: 11px;
    position: absolute;
    right: 24px;
}

#guide.rinsp-quicksearch-added a[href="plugin.php?H_name-tasks.html"] {
    font-size: 0;
}
#guide.rinsp-quicksearch-added a[href="plugin.php?H_name-tasks.html"]:after {
    content: "任务";
    font-size: 12px;
}

/* ==== pin users ==== */
.rinsp-pinned-user-list {
    display: flex;
    flex-wrap: wrap;
    width: 530px;
}

.rinsp-pinuser-main {
    position: fixed;
    left: calc(50vw + 510px);
    transform: translateY(-50%);
    top: 50vh;
    max-height: 90vh;
    overflow: auto;
}

.rinsp-pinuser-list > div {
    padding-right: 20px;
}
.rinsp-pinuser-list > b {
    display: inline-block;
    color: #FFF8;
    border-bottom: 3px solid #FFF8;
    box-sizing: border-box;
    font-size: 1.3em;
    font-weight: normal;
    min-width: 106px;
}

.rinsp-pinuser-item {
    margin: 5px 0;
    display: flex;
    
}
.rinsp-pinuser-item img {
    width: 100px;
    height: 100px;
    object-fit: cover;
    object-position: top;
    border: 3px solid #FFF;
}
.rinsp-pinuser-item.rinsp-pinuser-item-absent img {
    width: 50px;
    height: 50px;
}
.rinsp-pinuser-item.rinsp-pinuser-item-absent .rinsp-pinuser-item-face {
    display: flex;
}
.rinsp-pinuser-item.rinsp-pinuser-item-absent .rinsp-pinuser-item-face > div {
    min-width: 85px;
    max-width: 85px;
    flex-direction: column-reverse;
    background-color: transparent;
    background-image: linear-gradient(to right, #FFF8 0%, #FFF1 100%);
}
.rinsp-pinuser-item.rinsp-pinuser-item-absent:hover .rinsp-pinuser-item-face > div {
    background: #FFF8;
}
.rinsp-pinuser-item.rinsp-pinuser-item-absent .rinsp-pinuser-unpin-icon {
    text-align: right;
}
.rinsp-pinuser-item.rinsp-pinuser-item-absent .rinsp-pinuser-item-face > div > a {
    max-height: 50px;
    overflow: hidden;
}
.rinsp-pinuser-item-locs:empty {
    display: none;
}
.rinsp-pinuser-item-locs {
    margin: 0 0 0 5px;
    display: flex;
    flex-direction: column;
    flex-wrap: wrap;
    font-size: 0.8em;
    max-height: 130px;
    width: 5.5em;
}
.rinsp-pinuser-item-locs > dt {
    margin-right: 6px;
}
.rinsp-pinuser-item-locs > dt > a {
    color: #FFF8;
    font-weight: bold;
    margin-bottom: 1px;
    white-space: nowrap;
}

.rinsp-pinuser-item-locs > dt.rinsp-pinuser-loc-ignoredmark > a {
    color: #FF09;
}
.rinsp-pinuser-item-locs > dt.rinsp-pinuser-loc-gf > a {
    color: #0F09;
}
.rinsp-pinuser-item-locs > dt.rinsp-pinuser-loc-banned > a {
    color: #F44E;
}
.rinsp-pinuser-item-face > div {
    max-width: 106px;
    padding: 2px 4px;
    margin-right: -20px;
    display: flex;
    background: #FFF8;
    box-sizing: border-box;
}
.rinsp-pinuser-item:hover > .rinsp-pinuser-item-face > div {
    background: #FFFC;
}
.rinsp-pinuser-item-face > div > a {
    flex: 1;
    word-break: break-all;
}
.rinsp-pinuser-unpin-icon {
    cursor: pointer;
    height: 0;
}
.rinsp-pinuser-unpin-icon::before {
    content: "❌";
    font-size: 10px;
}
a:not([name="tpc"]) + .rinsp-thread-filter-active-reply {
    outline: 2px solid #36C;
    outline-offset: -1px;
}
.rinsp-thread-filter-pinned {
    outline: 3px solid #F08;
    outline-offset: -1px;
}
.rinsp-thread-filter-pinned .tr1 .r_two,
.rinsp-thread-filter-pinned th.r_two > .rinsp-userpic-sticky {
    background-color: #F2CFE0;
}
.rinsp-pinuser-item-absent {
    opacity: 0.3;
}

.rinsp-thread-user-pinned {
    outline: 2px solid #F08;
    outline-offset: -3px;
}
.rinsp-thread-user-pinned .rinsp-uid-inspected {
    font-weight: bold;
    color: #F08;
}
.rinsp-user-action-pinuser-button {
    display: inline-block;
    margin-left: 0.5em;
    vertical-align: top;
}
.rinsp-user-action-pinuser-button:not([data-pinned]):not(:hover) {
    filter: grayscale(1);
    opacity: 0.5;
}

.rinsp-quick-action-overlay {
    z-index: 4;
}
.rinsp-quick-action-overlay:focus > label::after {
    content: "凛+ (版本" attr(version) ")";
    font-size: 13px;
    color: #BBB;
}
.rinsp-quick-action-overlay:focus > label {
    border-bottom: 1px solid #999;
}

.rinsp-quick-action-overlay {
    position: fixed;
    font-size: 14px;
}
.rinsp-quick-action-overlay-pos-tr {
    right: 15px;
    top: 16px;
    padding-bottom: 32px;
    padding-left: 32px;
    text-align: right;
}
.rinsp-quick-action-overlay-pos-tr:focus {
    padding: 5px;
    margin-top: -5px;
    margin-right: -5px;
    border-radius: 5px;
    background: #FFF7;
}
.rinsp-quick-action-overlay-pos-tl {
    left: 25px;
    top: 16px;
    padding-bottom: 32px;
    padding-right: 32px;
    text-align: left;
}
.rinsp-quick-action-overlay-pos-tl:focus {
    padding: 5px;
    margin-top: -5px;
    margin-left: -5px;
    border-radius: 5px;
    background: #FFF7;
}

.rinsp-quick-action-overlay:focus {
    border-radius: 5px;
    background: #FFFA;
}
.rinsp-quick-action-overlay > label {
    font-size: 16px;
    filter: grayscale(1);
    opacity: 0.2;
    margin-bottom: 5px;
    display: block;
}
.rinsp-quick-action-overlay:focus > label {
    filter: grayscale(0.5) invert(1);
    opacity: 1;
}
.rinsp-quick-action-overlay:not(:focus) > label ~ * {
    display: none;
}
.rinsp-quick-action-overlay > label ~ * {
    margin-bottom: 0.2em;
}

.rinsp-quick-action-divider {
    border-top: 1px solid #999;
    margin-top: 0.3em;
    margin-bottom: 0.3em;
}
.rinsp-quick-action {
    margin-left: 1px;
}
.rinsp-quick-action::before {
    cursor: pointer;
}
.rinsp-quick-action::before {
    content: attr(label);
    color: var(--rinsp-darkest-text-color);
    display: inline-block;
}
.rinsp-quick-action:not(:hover)::before {
    opacity: 0.7;
}


.new-msg-tips,
.new-msg-tips + span {
    z-index: 5;
}
@media (max-width: 1278px) {
    .rinsp-pinuser-main {
        position: absolute;
    }
    .rinsp-pinuser-main:not(:empty) {
        top: 0;
        left: 0;
        padding: 5px 0 5px 0;
        width: 100%;
        box-sizing: border-box;
        transform: none;
        background-image: url();
        background-repeat: repeat;
        z-index: 2;
    }
    .rinsp-pinuser-list {
        text-align: center;
        white-space: nowrap;
    }
    .rinsp-pinuser-list > b {
        writing-mode: vertical-rl;
        min-width: auto;
        border-bottom: none;
        border-right: 3px solid #FFF8;
        min-height: 76px;
    }
    .rinsp-pinuser-list > div {
        display: inline-block;
        text-align: left;
        padding: 0;
        white-space: nowrap;
        max-width: 80vw;
        overflow-x: auto;
        overflow-y: hidden;
    }
    .rinsp-pinuser-item {
        margin: 0 5px;
    }
    .rinsp-pinuser-item-face {
        display: inline-flex;
        flex-direction: column;
        background: #FFF3;
        vertical-align: top;
    }
    .rinsp-pinuser-item.rinsp-pinuser-item-absent .rinsp-pinuser-item-face {
        display: inline-flex;
    }
    .rinsp-pinuser-item-face > div,
    .rinsp-pinuser-item.rinsp-pinuser-item-absent .rinsp-pinuser-item-face > div {
        max-width: 106px;
        min-width: 106px;
        margin: 0;
        flex-direction: row;
    }
    .rinsp-pinuser-item.rinsp-pinuser-item-absent .rinsp-pinuser-item-face > div {
        max-width: 78px;
        min-width: 78px;
    }
    .rinsp-pinuser-item .rinsp-pinuser-item-face > div > a,
    .rinsp-pinuser-item.rinsp-pinuser-item-absent .rinsp-pinuser-item-face > div > a {
        max-height: 1.4em;
        overflow: hidden;
    }
    .rinsp-pinuser-list .rinsp-pinuser-item {
        margin-right: 5px;
        display: inline-block;
    }
    .rinsp-pinuser-item-locs {
        width: auto;
        vertical-align: top;
        display: inline-flex;
        margin-left: -36px;
        max-height: 66px;
        overflow: auto;
        flex-wrap: nowrap;
    }
    .rinsp-pinuser-item img {
        width: 60px;
        height: 60px;
    }
}

/* ==== subject line functions ==== */
.rinsp-theme-darksubject h1#subject_tpc {
    color: #333;
}
.rinsp-theme-darksubject h1[id^="subject_"] {
    color: #999;
}
.rinsp-theme-darksubject h1#subject_tpc .rinsp-subject-class {
    color: #BBB;
    margin-right: 0.2em;
    font-size: 15px;
}
.rinsp-subject-redundant {
    display: none;
}
.rinsp-subject-floor-link {
    color: #3399CC;
}
.blockquote3 .rinsp-subject-floor-link {
    font-weight: bold;
    color: #3399CC;
    margin: 0 0.2em;
}

/* ==== resource spots ==== */
.rinsp-spots-main {
    position: fixed;
    left: calc(50vw - 490px);
    transform: translateX(-100%);
    top: calc(49vh - 30px);
}
.rinsp-highlight-spots > dl {
    max-height: calc(50vh - 40px);
    overflow: auto;
}
.rinsp-highlight-spots > b {
    display: inline-block;
    color: #FFF8;
    box-sizing: border-box;
    font-size: 1.3em;
    font-weight: normal;
}
.rinsp-highlight-spots > dl {
    background: #FFF1;
    border-top: 3px solid #FFF8;
    font-size: 1.1em;
    margin-top: 0;
    padding: 0.2em 0.5em;
    scrollbar-gutter: stable;
    scrollbar-width: thin;
}

.rinsp-highlight-spots > dl > dt > a {
    display: flex;
    color: #CCC;
}
.rinsp-spot-label {
    flex: 1;
    max-width: 10em;
    white-space: nowrap;
    overflow: hidden;
    text-overflow: ellipsis;
}
.rinsp-highlight-spots > dl > dt > a .rinsp-spot-floor {
    color: #EEE;
    font-weight: bold;
    font-family: monospace;
    margin-right: 0.3em;
    min-width: 2.5em;
    display: inline-block;
}

.rinsp-thread-header {
    display: flex;
}
.rinsp-spots-bar {
    display: flex;
    flex: 1;
    white-space: nowrap;
    order: 2;
    overflow: hidden;
    text-overflow: ellipsis;
}
.rinsp-spots-bar .rinsp-spot-floor {
    font-weight: bold;
    font-size: 0.8em;
}
.rinsp-spots-bar > a {
    margin-left: 5px;
}
.rinsp-spots-bar > a + a {
    border-left: 1px solid #AAA;
    padding-left: 5px;
}
.rinsp-thread-header > .fl {
    flex: 0;
    order: 1;
}
.rinsp-thread-header > .fl:not(:empty) {
    padding-right: 1em;
}
.rinsp-thread-header > .fr {
    flex: 0;
    order: 3;
    padding-left: 1.5em;
}


@media (max-width: 1278px) {
    .rinsp-spots-main .rinsp-spot-label {
        display: none;
    }
}
@media (max-width: 1150px) {
    .rinsp-highlight-spots {
        font-size: 0.8em;
    }
}

/* ==== infinite scroll ==== */

.rinsp-infscroll-armed .rinsp-infscroll-page-endtrigger {
    --rinsp-infscroll-height: 70px; /* add 15px */
}
.rinsp-infscroll-switch.rinsp-active {
    background: var(--rinsp-cell-background-active);
}
div.rinsp-infscroll-divider,
tbody.rinsp-infscroll-divider > tr > td {
    color: var(--rinsp-infscroll-divider-text-color);
    border-top: 1px solid var(--rinsp-infscroll-divider-border-color);
    background: var(--rinsp-infscroll-divider-background);
    padding: 0.2em 0 0.2em 1em;
    text-align: center;
    clear: both;
}
div.rinsp-infscroll-divider {
    border-bottom: 1px dotted var(--rinsp-infscroll-divider-border-color);
    margin-bottom: 8px;
}
.rinsp-infscroll-enabled .rinsp-infscroll-page-endtrigger {
    width: 1px;
    height: var(--rinsp-infscroll-height);
}
.rinsp-infscroll-enabled .rinsp-infscroll-page-endtrigger:after {
    content: "";
    position: absolute;
    top: 100vh;
    width: 1px;
    height: var(--rinsp-infscroll-height);
    z-index: 10000;
}
.rinsp-infscroll-armed .rinsp-infscroll-loader-bar {
    position: fixed;
    left: 50%;
    transform: translatex(-50%);
    bottom: 0;
    width: 940px;
    border-top: 1px dotted var(--rinsp-infscroll-loader-border-color);
    height: 15px;
    background: var(--rinsp-infscroll-loader-background);
    overflow: hidden;
}
.rinsp-infscroll-firing .rinsp-infscroll-loader-bar:before {
    content: "";
    display: block;
    position: relative;
    top: 0;
    left: 0;
    height: 100%;
    background: linear-gradient(to right, var(--rinsp-infscroll-loader-progress) 90%, transparent 100%);
    animation: count-down 2s;
}
.rinsp-infscroll-armed:not(.rinsp-infscroll-firing) .rinsp-infscroll-loader-bar:after {
    content: "";
    display: block;
    position: absolute;
    top: 0;
    left: 0;
    right: 0;
    bottom: 0;
    background: url() top center no-repeat;
}

@keyframes count-down {
    0% {
        width: 0%;
    }
    100% {
        width: 120%;
    }
}

/* ==== lazy image loading ==== */
img[src][data-rinsp-defer-src] {
    width: auto;
    animation: fade-in 2s alternate infinite;
}

@keyframes fade-in {
    0% {
        opacity: 0.2;
    }
    100% {
        opacity: 1;
    }
}

/* ==== reply refresh free anim ==== */
.rinsp-reply-refresh-free {
    position: relative;
    display: block;
}
.rinsp-reply-refresh-free.rinsp-refresh-free-submitting::after {
    content: "提交中 ...";
    display: flex;
    align-items: center;
    justify-content: center;
    position: absolute;
    top: 0;
    height: 100%;
    left: 0;
    width: 100%;
    z-index: 1;
    background: rgba(255,255,255,0.7);

}
.rinsp-reply-refresh-free.rinsp-refresh-free-submitting::before {
    content: "⌛";
    font-size: 16px;
    display: inline-block;
    animation-name: rinsp-watch-checking-rotate-anim;
    animation-duration: 2s;
    animation-iteration-count: infinite;
    z-index: 2;
    position: absolute;
    top: calc(50% - 0.8em);
    left: calc(50% - 3em);
}

/* ==== get sp animation ==== */
#guide a[href="plugin.php?H_name-tasks.html"][rinsp-sp-get]::before {
    content: "+ " attr(rinsp-sp-get) "SP";
    position: absolute;
    transform: translate(0, 20%);
    opacity: 0;
    color: #000;
    background: linear-gradient(to bottom, #fff7a0 0%, #ffc033 100%);
    padding: 2px 5px;
    border: 1px solid #ee9905;
    border-radius: 16px;
    box-shadow: inset 0 0 2px #fff, inset 0 0 2px #fff, inset 0 0 2px #fff, inset 0 0 2px #fff;
    line-height: normal;
    z-index: 1;
    font-weight: normal;
    animation-name: sp-get;
    animation-duration: 5s;
    animation-iteration-count: 1;
    animation-timing-function: ease-out;
}


@keyframes sp-get {
    0% {
        transform: translate(0, 20%);
        opacity: 1;
    }
    20% {
        transform: translate(0, -140%);
        opacity: 1;
    }
    80% {
        transform: translate(0, -140%);
        opacity: 1;
    }
    100% {
        transform: translate(0, -140%);
        opacity: 0;
    }
}


/* ==== batch ping/sel functions ==== */
.rinsp-batchsel-form {
    max-height: calc(90vh - 150px);
    overflow-x: hidden;
    overflow-y: auto;
}
.rinsp-batchsel-form-op-cell {
    text-align: left;
    padding-left: 20px;
}
.rinsp-batchsel-form-op-cell.rinsp-batchsel-form-op-known::before {
    content: "🔖 ";
    margin-left: -20px;
}
.rinsp-batchsel-form-item-unchecked {
    opacity: 0.8;
    background: #EEE;
}
.rinsp-batchsel-form-item-unavailable {
    opacity: 0.3;
}
.rinsp-batchsel-form-item-unavailable > td:first-child > * {
    display: none;
}
.rinsp-batchping-form-item-unscore {
    background: linear-gradient(to right, #FEE 0%, transparent 50%);
}
.rinsp-batchping-form-item-unscore .rinsp-batchping-form-ping-cell::before,
.rinsp-batchping-form-item-unchecked .rinsp-batchping-form-ping-cell::before {
    content: "---";
    color: #CCC;
    display: block;
    text-align: center;
}
.rinsp-batchping-form-item-unchecked .rinsp-batchping-form-ping-cell > *,
.rinsp-batchping-form-item-unchecked .rinsp-batchping-form-ping-cell + td > * {
    display: none;
}
.rinsp-batchping-form-item-unscore .rinsp-batchping-form-sp-input,
.rinsp-batchping-form-item-unscore .rinsp-batchping-form-sp-input + span {
    display: none;
}
.rinsp-batchping-progress-list {
    padding: 0 1em 1em 1em !important;
}
.rinsp-batchping-progress-list > li {
    padding: 0.3em 0 !important;
}
.rinsp-batchping-progress-list > li:first-child {
    font-weight: bold;
}

/* ==== ping functions ==== */
.rinsp-quickping-form {
    width: 500px;
    padding: 0.2em;
    display: flex;
    flex-wrap: wrap;
}
.rinsp-quickping-form > dt {
    padding: 0.5em 0.5em 0.5em 2em;
    flex-basis: 4em;
    flex-shrink: 0;
    flex-grow: 0;
    text-align: right;
}
.rinsp-quickping-form > dd {
    padding: 0.5em 0.5em;
    flex-grow: 1;
}
.rinsp-quickping-form > div {
    flex-grow: 1;
    flex-basis: 100%;
    border-top: 1px dotted #CCC;
}
.rinsp-quickping-attr-ownbought {
    color: #0000FF;
}
.rinsp-quickping-attr-owntranslate {
    color: #FF0000;
}
.rinsp-quickping-attr-compilation {
    color: #FF00FF;
}
.rinsp-quickping-attr-ownbought, .rinsp-quickping-attr-owntranslate, .rinsp-quickping-attr-compilation, .rinsp-quickping-attr-extra {
    font-weight: bold;
    display: flex;
    align-items: center;
}
.rinsp-quickping-status-error {
    color: #990000;
}

.h2 a.rinsp-quickping-button {
    font-weight: bold;
    color: #ce0940;
}
.h2 a.rinsp-instantping-button {
    font-weight: bold;
    color: #0928ce;
}
.h2 a.rinsp-quickping-button.rinsp-quickping-grey,
.h2 a.rinsp-instantping-button.rinsp-instantping-grey {
    color: #808080;
    font-weight: normal;
}
.h2 a.rinsp-noping-button,
.h2 a.rinsp-badping-button {
    font-weight: bold;
    color: #cc0000;
}
.h2 a.rinsp-quickping-status {
    font-weight: bold;
    color: #112591;
}

.rinsp-quickping-form > .rinsp-quickping-preview-section {
    padding-top: 0.3em;
    border-top: none;
    text-align: center;
    font-size: 18px;
}
.rinsp-quickping-subject {
    padding: 0.5em 1em;
    word-break: break-all;
    font-size: 14px;
    max-width: 40em;
    background: #EEE;
}
.rinsp-quickping-reason-cell {
    display: flex;
}
.rinsp-quickping-reason-cell > textarea {
    flex: 1;
}

.rinsp-post-illegal-sell .tpc_content h6.quote.jumbotron > .s3.f12.fn {
    color: red;
    font-weight: bold;
}
.readbot .rinsp-markcorrect-switch,
.readbot .rinsp-punishrequest-switch,
.readbot .rinsp-punishclassify-switch {
    width: 5em;
}
.rinsp-ping-preset-switch::before {
    content: "🔘";
}
.rinsp-ping-preset-switch {
    margin-left: 0.5em;
    color: #333;
}

form[action="job.php?action=endreward"] table > tbody > tr > th:first-child {
    text-align: right;
    padding: .3em .6em;
    line-height: 25px;
}

/* ==== admin show user name ==== */
.rinsp-user-name-reveal {
    font-family: monospace;
    padding: 0px 2px;
    display: inline-block;
    outline: 1px solid #CCC;
    margin-left: 0.5em;
}

/* ==== admin punish status ==== */
.rinsp-profile-punish-tag-cell {
    padding-right: 16px;
    text-align: right;
}
.rinsp-profile-punish-tag-cell .rinsp-profile-punish-tags {
    font-size: 13px;
}
.rinsp-profile-punish-tag {
    font-weight: bold;
    font-family: monospace;
    margin-left: 0.4em;
    border: 1px solid #CCC;
    padding: 1px 5px;
}

.rinsp-profile-punish-tags .rinsp-profile-punish-tag::before {
    font-weight: normal;
    margin-right: 1px;
    font-size: 0.9em;
}

.rinsp-profile-punish-tags .rinsp-profile-punish-tag-blood {
    border-color: #C00;
    color: #FFF;
    background-color: #C00;
}
.rinsp-profile-punish-tags .rinsp-profile-punish-tag-blood::before {
    content: "血";
}
.rinsp-profile-punish-tags .rinsp-profile-punish-tag-ban {
    border-color: #C00;
    color: #C00;
}
.rinsp-profile-punish-tags .rinsp-profile-punish-tag-ban::before {
    content: "禁";
}

.rinsp-profile-punish-tags .rinsp-profile-punish-tag-del {
    border-color: #D80;
    color: #D80;
}
.rinsp-profile-punish-tags .rinsp-profile-punish-tag-del::before {
    content: "删";
}

.rinsp-profile-punish-tags .rinsp-profile-punish-tag-misc {
    border-color: #059;
    color: #059;
}
.rinsp-profile-punish-tags .rinsp-profile-punish-tag-misc::before {
    content: "他";
}

.rinsp-userframe-udata-punish-tags .rinsp-profile-punish-tags {
    font-size: 11px;
}
.rinsp-userframe-udata-punish-tags .rinsp-profile-punish-tags::before {
    content: "🚩";
}
.rinsp-userframe-udata-punish-tags .rinsp-profile-punish-tag {
    border-width: 0;
    margin-left: 0;
    padding: 0;
}
.rinsp-userframe-udata-punish-tags .rinsp-profile-punish-tag:first-child::before {
    content: "";
}
.rinsp-userframe-udata-punish-tags .rinsp-profile-punish-tag::before {
    content: "/";
    color: #CCC;
}
.rinsp-userframe-udata-punish-tags .rinsp-profile-punish-tag-blood {
    background: none;
    color: #000;
}
.rinsp-punish-history-table .rinsp-profile-punish-tag-blood > th {
    color: #C00;
    border-left: 3px solid #C00;
}
.rinsp-punish-history-table .rinsp-profile-punish-tag-ban > th {
    border-left: 3px solid #C00;
}
.rinsp-punish-history-table .rinsp-profile-punish-tag-del > th {
    border-left: 3px solid #D80;
}
.rinsp-punish-history-table .rinsp-profile-punish-tag-misc > th {
    border-left: 3px solid #059;
}

.rinsp-punish-history-table .rinsp-profile-punish-tag-none > th {
    border-left: 3px solid #CCC;
}


/* ==== admin punish log ==== */
input[name="ifdel"][value="0"]:checked + .rinsp-record-delatc-msg {
    display: none;
}
input[name="step"][value="2"]:checked + .rinsp-record-showping-option {
    display: none;
}
.rinsp-punish-history-table {
    width: 100%;
}
.rinsp-punish-history-table > tbody > tr > th {
    width: 6em;
}
.rinsp-punish-history-table > tbody > tr > th > div {
    font-weight: bold;
}
.rinsp-punish-history-table .rinsp-punish-history-summary-cell {
    color: var(--rinsp-darkest-text-color);
    cursor: default;
}
.rinsp-punish-history-table .rinsp-punish-history-summary-cell summary::marker {
    color: #999;
}
.rinsp-punish-history-table .rinsp-punish-history-summary-cell summary > div {
    margin-left: 10px;
    white-space: pre-line;
    color: #AAA;
}
.rinsp-punish-history-data {
    margin-left: 2px;
    border-left: 1px solid #ccc;
    padding-left: 9px;
}
.rinsp-punish-history-data label {
    color: #4e83a8;
    margin-right: 0.5em;
}
.rinsp-punish-history-data label + span {
    color: #666;
}

.rinsp-punish-history-table > tbody > tr:hover {
    outline: 1px solid #DDD;
}
.rinsp-punish-history-table .rinsp-punish-history-del-cell {
    width: 5em;
    text-align: right;
}
.rinsp-punish-history-del-cell > a {
    cursor: pointer;
}

/* ==== safe mode ==== */
.rinsp-safe-mode #cate_thread #tab_1 img:not([data-rinsp-defer-src]):not(:hover),
.rinsp-safe-mode .tpc_content img:not([data-rinsp-defer-src]):not([src^="images/post/smile/"]):not(:hover) {
    filter: var(--rinsp-image-masking-filter);
    transform: scale(0.99);
}
.rinsp-safe-mode .rinsp-pinuser-item-face img:not([data-rinsp-defer-src]):not(:hover),
.rinsp-safe-mode:not(.rinsp-safe-mode-allow-myself) #user_info a[href="u.php"] img:not([data-rinsp-defer-src]):not(:hover),
.rinsp-safe-mode.rinsp-safe-mode-allow-myself #u-portrait img.pic:not(.rinsp-myavatar):not(:hover),
.rinsp-safe-mode:not(.rinsp-safe-mode-allow-myself) #u-portrait img.pic:not(:hover),
.rinsp-safe-mode .js-post:not(.rinsp-my-post) a[href^="u.php?action-show-uid-"]:not(:hover) > img:not([data-rinsp-defer-src]),
.rinsp-safe-mode:not(.rinsp-safe-mode-allow-myself) .js-post.rinsp-my-post a[href^="u.php?action-show-uid-"]:not(:hover) > img:not([data-rinsp-defer-src]) {
    content: url(/images/face/none.gif);
}
.rinsp-safe-mode .rinsp-quick-action.rinsp-safe-mode-toggle {
    display: block;
}

/* ==== dark mode toggle ==== */

.rinsp-quick-action.rinsp-dark-mode-toggle::before {
    content: "🌒开启暗黑模式";
}
.rinsp-dark-mode-set .rinsp-quick-action.rinsp-dark-mode-toggle::before {
    content: "☀️关闭暗黑模式";
}

/* ==== safe mode toggle ==== */

.rinsp-quick-action.rinsp-safe-mode-toggle::before {
    content: "😇开启贤者模式";
}
.rinsp-safe-mode .rinsp-quick-action.rinsp-safe-mode-toggle::before {
    content: "😈关闭贤者模式";
}

/* ==== patch forum default styles ==== */

/* prevent category column to become too narrow due to user name being too long */
#u-contentmain > form[action="u.php?action=favor&"] > table > tbody > tr > td:nth-child(3) {
    min-width: 2em;
}
.rinsp-user-link {
    max-width: 7em;
    display: inline-block;
    font-size: 12px
}
.rinsp-uphp-post #u-contentmain > table > tbody > tr > th + td {
    font-size: 10px;
}
.rinsp-uphp-post #u-contentmain > table > tbody > tr > th + td > .rinsp-user-link {
    display: block;
}

.rinsp-hide-mobileswitch #main > center.gray3 > div > a[href^="/simple/"] {
    display: none;
}

/* text size overrides */
html.rinsp-textsize-step-1 {
    font-size: 62.5%;
}
html.rinsp-textsize-step-1 .set-table2 td {
    padding: 0.5em 0.5em;
}
html.rinsp-textsize-step-1 #set-menu,
html.rinsp-textsize-step-1 #info_base td > a[href^="message.php?"],
html.rinsp-textsize-step-1 th.y-style > a {
    font-size: 1.4rem;
}
html.rinsp-textsize-step-1 .t_one > td > h3 {
    font-size: 13px;
}

html.rinsp-textsize-step-1 #u-fav-cate form[action="u.php?action=favor&"] {
    font-size: 1.3rem;
}

html.rinsp-textsize-step-2 #main,
html.rinsp-textsize-step-2 #rinsp-watcher-menu {
    font-size: 1.4rem;
}

html.rinsp-textsize-step-2 h1 {
    font-size: 1.8rem;
}
html.rinsp-textsize-step-2 .small {
    font-size: 1.4rem;
}
html.rinsp-textsize-step-2 .f14,
html.rinsp-textsize-step-2 .middle {
    font-size: 1.7rem;
}
html.rinsp-textsize-step-2 .big {
    font-size: 2rem;
}

html.rinsp-textsize-step-3 #main,
html.rinsp-textsize-step-3 #rinsp-watcher-menu {
    font-size: 1.5rem;
}

html.rinsp-textsize-step-3 .tiptop {
    font-size: 1.3rem;
}
html.rinsp-textsize-step-3 h1 {
    font-size: 2rem;
}
html.rinsp-textsize-step-3 .small {
    font-size: 1.4rem;
}
html.rinsp-textsize-step-3 .f14,
html.rinsp-textsize-step-3 .middle {
    font-size: 1.8rem;
}
html.rinsp-textsize-step-3 .big {
    font-size: 2.1rem;
}
html.rinsp-textsize-step-3 .t_one > td > h3 {
    font-size: 14px;
}


/* fix textarea too wide in firefox */
form[action="message.php"] textarea#atc_content {
    width: calc(100% - 0.5em);
}
form[action="mawhole.php?"] textarea[name="atc_content"] {
    width: calc(100% - 300px);
}
#info_base .td1 {
    min-width: 4em;
}

/* 短消息设置 width bug */
textarea[name="banidinfo"] {
    width: 100%;
}

/* handy highlights */
.rinsp-last-clicked {
    outline: 2px solid #dbbc60;
    outline-offset;
}
`;

const darkThemeCss = `
:root {
    --rinsp-dm-invert-filter: invert(1) hue-rotate(180deg);
    --rinsp-dm-invert-filter-blk: invert(1) hue-rotate(180deg) saturate(1.2) contrast(0.81);
    --rinsp-dm-invert-filter-darkbg1: invert(1) hue-rotate(180deg) saturate(1.2) contrast(0.76) brightness(1.1);
    --rinsp-dm-hgt-text-filter: contrast(0.8) brightness(1.2);
    --rinsp-dm-color-black: #0d0d0d;
    --rinsp-dm-color-darkbg1: #141414;
    --rinsp-dm-color-darkbg2: #2d2d2d;
    --rinsp-dm-color-darkbg-hgt: #2f2f2f;
    --rinsp-dm-color-darkbg-hover: #36311f;
    --rinsp-dm-color-border: #666;
    --rinsp-dm-color-light-border: #AAA;
    --rinsp-dm-color-text: #CCC;
    --rinsp-dm-color-link: #BBB;
    --rinsp-dm-color-content-link: #6388A8;
    --rinsp-dm-color-content-link-hover: #8FB6D7;
    --rinsp-dm-color-heading: #eaeaea;
    --rinsp-dm-color-label: #eaeaea;
    --rinsp-dm-color-ctrl: #eaeaea;
    --rinsp-dm-color-grey: #AAA;
    --rinsp-dm-button-bg: #4b0707;
}

@media (prefers-color-scheme: light) {
    html.rinsp-dark-mode input[type="checkbox"],
    html.rinsp-dark-mode input[type="radio"] {
        filter: invert(1) hue-rotate(180deg) brightness(0.8);
    }
}

html.rinsp-dark-mode {
    opacity: 1;
    transition: opacity 0.08s ease-in;
    --rinsp-active-toggler-bg-color: #2b2b2b;
    --rinsp-active-toggler-fg-color: #CCC;
    --rinsp-visited-link-color: #AAA;
    --rinsp-visited-link-hover-color: #CCC;
    --rinsp-visited-update-color: #dba4f9;
    --rinsp-visited-update-border-color: #843cab;
    --rinsp-blocked-row-bg: #222;
    --rinsp-blocked-row-label-color: #999;
    --rinsp-avatar-replace-default-text-color: #999;
    --rinsp-avatar-replace-default-border-color: #202224;
    --rinsp-avatar-replace-default-bg: #303234;
    --rinsp-cell-background-grey-out: #444;
    --rinsp-cell-background-hgt-new: #44430e;
    --rinsp-cell-background-hgt-err: #491616;
    --rinsp-cell-background-active: #283d44;
    --rinsp-lightest-bg-color: #111;
    --rinsp-darkest-text-color: #CCC;
    --rinsp-text-color-orange: #bf9342;
    --rinsp-text-color-violet: #aa4ec4;
    --rinsp-text-color-green: #2fa847;
    --rinsp-text-color-red: #a63232;
    --rinsp-text-color-blue: #4ca0d3;
    --rinsp-text-color-grey: #999;
    --rinsp-infscroll-divider-text-color: #556a71;
    --rinsp-infscroll-divider-border-color: #535c5c;
    --rinsp-infscroll-divider-background: #1e2d30;
    --rinsp-infscroll-loader-border-color: #555;
    --rinsp-infscroll-loader-background: #272727;
    --rinsp-infscroll-loader-progress: #444;
}

html.rinsp-dark-mode.rinsp-dark-theme-darkbeige {
    --rinsp-dm-color-black: #181818;
    --rinsp-dm-color-darkbg1: #1c1c19;
    --rinsp-dm-color-darkbg2: #26292b;
    --rinsp-dm-color-darkbg-hgt: #242d31;
    --rinsp-dm-color-darkbg-hover: #413d2c;
    --rinsp-dm-color-border: #666;
    --rinsp-dm-color-light-border: #AAA;
    --rinsp-dm-color-text: #BBB;
    --rinsp-dm-color-link: #BBB;
    --rinsp-dm-color-content-link: #6388A8;
    --rinsp-dm-color-content-link-hover: #8FB6D7;
    --rinsp-dm-color-heading: #a08c6c;
    --rinsp-dm-color-label: #BBB;
    --rinsp-dm-color-ctrl: #958568;
    --rinsp-dm-color-grey: #AAA;
    --rinsp-dm-button-bg: #4b0707;
}

html.rinsp-dark-mode.rinsp-dark-theme-darkblue {
    --rinsp-dm-color-black: #161b1f;
    --rinsp-dm-color-darkbg1: #0d0e14;
    --rinsp-dm-color-darkbg2: #0e1622;
    --rinsp-dm-color-darkbg-hgt: #151f25;
    --rinsp-dm-color-darkbg-hover: #363019;
    --rinsp-dm-color-border: #383d3e;
    --rinsp-dm-color-light-border: #666;
    --rinsp-dm-color-text: #BBB;
    --rinsp-dm-color-link: #CCC;
    --rinsp-dm-color-heading: #668184;
    --rinsp-dm-color-label: #999;
    --rinsp-dm-color-ctrl: #8faab1;
    --rinsp-dm-color-grey: #9a9a9a;
    --rinsp-dm-button-bg: #4b0707;
}

html.rinsp-dark-mode.rinsp-dark-theme-hcdarkbeige {
    --rinsp-dm-invert-filter-blk: invert(1) hue-rotate(180deg) saturate(1.2) contrast(0.91);
    --rinsp-dm-invert-filter-darkbg1: invert(1) hue-rotate(180deg) saturate(1.2) contrast(0.88) brightness(1.1);
    --rinsp-dm-hgt-text-filter: none;
    --rinsp-darkest-text-color: #EEE;
    --rinsp-dm-color-black: #000;
    --rinsp-dm-color-darkbg1: #111;
    --rinsp-dm-color-darkbg2: #161616;
    --rinsp-dm-color-darkbg-hgt: #242d31;
    --rinsp-dm-color-darkbg-hover: #413d2c;
    --rinsp-dm-color-border: #666;
    --rinsp-dm-color-light-border: #AAA;
    --rinsp-dm-color-text: #EEE;
    --rinsp-dm-color-link: #DDD;
    --rinsp-dm-color-heading: #e6b466;
    --rinsp-dm-color-label: #CCC;
    --rinsp-dm-color-ctrl: #d7aa5b;
    --rinsp-dm-color-grey: #AAA;
    --rinsp-dm-button-bg: #4b0707;
    --rinsp-dm-color-content-link: #64b7ff;
    --rinsp-dm-color-content-link-hover: #8FB6D7;
}

html.rinsp-dark-mode.rinsp-dark-theme-hcdarkblue {
    --rinsp-dm-invert-filter-blk: invert(1) hue-rotate(180deg) saturate(1.2) contrast(0.91);
    --rinsp-dm-invert-filter-darkbg1: invert(1) hue-rotate(180deg) saturate(1.2) contrast(0.88) brightness(1.1);
    --rinsp-dm-hgt-text-filter: none;
    --rinsp-dm-color-black: #000;
    --rinsp-dm-color-darkbg1: #111;
    --rinsp-dm-color-darkbg2: #161616;
    --rinsp-dm-color-darkbg-hgt: #242d31;
    --rinsp-dm-color-darkbg-hover: #363019;
    --rinsp-dm-color-border: #666;
    --rinsp-dm-color-light-border: #666;
    --rinsp-dm-color-text: #EEE;
    --rinsp-dm-color-link: #DDD;
    --rinsp-dm-color-heading: #9bd8e6;
    --rinsp-dm-color-label: #CCC;
    --rinsp-dm-color-ctrl: #88b5d7;
    --rinsp-dm-color-grey: #AAA;
    --rinsp-dm-button-bg: #4b0707;
    --rinsp-dm-color-content-link: #64b7ff;
    --rinsp-dm-color-content-link-hover: #8FB6D7;
}


html.rinsp-dark-mode.rinsp-dark-theme-darkgrey {
    --rinsp-dm-invert-filter-blk: invert(1) hue-rotate(180deg) saturate(1.2) contrast(0.91);
    --rinsp-dm-invert-filter-darkbg1: invert(1) hue-rotate(180deg) saturate(1.2) contrast(0.88) brightness(1.1);
    --rinsp-dm-color-black: #000;
    --rinsp-dm-color-darkbg1: #111;
    --rinsp-dm-color-darkbg2: #222;
    --rinsp-dm-color-darkbg-hgt: #333;
    --rinsp-dm-color-darkbg-hover: #444;
    --rinsp-dm-color-border: #444;
    --rinsp-dm-color-light-border: #666;
    --rinsp-dm-color-text: #BBB;
    --rinsp-dm-color-link: #999;
    --rinsp-dm-color-heading: #CCC;
    --rinsp-dm-color-label: #AAA;
    --rinsp-dm-color-ctrl: #999;
    --rinsp-dm-color-grey: #999;
    --rinsp-dm-button-bg: #440303;
}


html.rinsp-dark-mode.rinsp-dark-theme-hcdarkbeige body,
html.rinsp-dark-mode.rinsp-dark-theme-hcdarkblue body,
html.rinsp-dark-mode.rinsp-dark-theme-darkgrey body {
    background-image: url();
}

html.rinsp-dark-mode .menu .bor {
    border-color: var(--rinsp-dm-color-border);
}
html.rinsp-dark-mode .rinsp-thread-filter-menu-button:after {
    filter: brightness(2);
}

html.rinsp-dark-mode .rinsp-config-item-tip {
    color: #668ca2;
}

@media (max-width: 1278px) {
    html.rinsp-dark-mode .rinsp-pinuser-main:not(:empty) {
        background-image: url();
    }
}

html.rinsp-dark-mode body {
    background-image: url();
}

html.rinsp-dark-mode {
    --rinsp-image-masking-filter: contrast(0%) brightness(30%) drop-shadow(0px 0px 1px #FFF);
}
html.rinsp-dark-mode #fav-fid li {
    filter: grayscale();
}
html.rinsp-dark-mode .guide li.current a {
    background: #822d2d8f;
}
html.rinsp-dark-mode .rinsp-quick-action-overlay:focus {
    background: #111A;
}

html.rinsp-dark-mode h1,
html.rinsp-dark-mode h2,
html.rinsp-dark-mode h3,
html.rinsp-dark-mode h4,
html.rinsp-dark-mode h5,
html.rinsp-dark-mode h6 {
    color: var(--rinsp-dm-color-heading);
}
html.rinsp-dark-mode body,
html.rinsp-dark-mode .gray2,
html.rinsp-dark-mode .gray3,
html.rinsp-dark-mode #honor {
    color: var(--rinsp-dm-color-ctrl);
}

html.rinsp-dark-mode td,
html.rinsp-dark-mode th,
html.rinsp-dark-mode #set-menu li a {
    color: var(--rinsp-dm-color-text);
}

html.rinsp-dark-mode #u-top-nav li a {
    background: transparent;
    color: var(--rinsp-dm-color-text);
}
html.rinsp-dark-mode #u-top-nav li:not(.current) {
    box-shadow: 0 1px 0 var(--rinsp-dm-color-border);
}

html.rinsp-dark-mode .gray,
html.rinsp-dark-mode .threadlist li a {
    color: var(--rinsp-dm-color-grey);
}

/* code block */
html.rinsp-dark-mode .blockquote2 ol li {
    background-color: unset;
}

/* thread list typical color headings */
html.rinsp-dark-mode .stream li .section-title a font[color="#0000FF"],
html.rinsp-dark-mode .t_one h3 > a > b > font[color="#0000FF"] {
    color: #0af;
}
html.rinsp-dark-mode .stream li .section-title a font[color="#FF0000"],
html.rinsp-dark-mode .t_one h3 > a > b > font[color="#FF0000"] {
    color: #f33;
}
html.rinsp-dark-mode .stream li .section-title a font[color="#FF00FF"],
html.rinsp-dark-mode .t_one h3 > a > b > font[color="#FF00FF"] {
    color: #e5f;
}


html.rinsp-dark-mode .tpc_content span[style="color:#000000 "] {
    color: unset !important;
}
html.rinsp-dark-mode .tpc_content span[style="background-color:#ffffff "] {
    background-color: unset !important;
}

html.rinsp-dark-mode .tpc_content span[style="color:#333333 "] {
    color: color: var(--rinsp-dm-color-grey) !important;
}

/* mobile version switcher / redirect landing page */
html.rinsp-dark-mode div[style^="padding:15px 60px;border:1px solid #eeeeee;background:#ffffff"],
html.rinsp-dark-mode #main > center.gray3 > div > a[href^="/simple/"] > div[style] {
    background-color: var(--rinsp-dm-color-black) !important;
    border-color: var(--rinsp-dm-color-black) !important;
    color: var(--rinsp-dm-color-border) !important;
}

html.rinsp-dark-mode #header .banner,
html.rinsp-dark-mode .rinsp-input-max-hint {
    filter: var(--rinsp-dm-invert-filter-blk);
}
html.rinsp-dark-mode #footer {
    background-color: var(--rinsp-dm-color-black);
    border-color: var(--rinsp-dm-color-border);
}
html.rinsp-dark-mode .gonggao {
    background-color: var(--rinsp-dm-color-black);
    border-color: var(--rinsp-dm-color-border);
}
html.rinsp-dark-mode .gongul {
    border-color: transparent;
}
html.rinsp-dark-mode .gonggao .gongul li {
    background: var(--rinsp-dm-color-darkbg1);
}
html.rinsp-dark-mode .gonggao .gongul .lin {
    border-color: var(--rinsp-dm-color-border);
}

html.rinsp-dark-mode body,
html.rinsp-dark-mode button,
html.rinsp-dark-mode input,
html.rinsp-dark-mode select,
html.rinsp-dark-mode textarea,
html.rinsp-dark-mode .tips,
html.rinsp-dark-mode .btn,
html.rinsp-dark-mode .abtn,
html.rinsp-dark-mode .set-h2 {
    color: var(--rinsp-dm-color-grey);
    background-color: var(--rinsp-dm-color-darkbg1);
}
html.rinsp-dark-mode th[bgcolor="#ffffff"],
html.rinsp-dark-mode .t_one,
html.rinsp-dark-mode .f_one,
html.rinsp-dark-mode .r_one {
    background: var(--rinsp-dm-color-darkbg1);
}
html.rinsp-dark-mode .card,
html.rinsp-dark-mode .f_two {
    background: var(--rinsp-dm-color-darkbg2);
}
html.rinsp-dark-mode .stream li .section-text > span {
    filter: invert(0.6);
}
html.rinsp-dark-mode .stream li .section-intro {
    filter: brightness(0.7);
}

html.rinsp-dark-mode .stream li {
    background: var(--rinsp-dm-color-darkbg1);
}
html.rinsp-dark-mode .stream li .section-title a {
    color: var(--rinsp-dm-color-link);
}

html.rinsp-dark-mode .t table {
    border-color: var(--rinsp-dm-color-darkbg1);
}
html.rinsp-dark-mode .tipad .fr a {
    color: var(--rinsp-dm-color-ctrl);
}

html.rinsp-dark-mode #breadcrumbs,
html.rinsp-dark-mode #set-side-wrap,
html.rinsp-dark-mode #u-wrap,
html.rinsp-dark-mode #u-wrap2,
html.rinsp-dark-mode #u-portrait,
html.rinsp-dark-mode .bgA {
    color: var(--rinsp-dm-color-label);
    background: var(--rinsp-dm-color-darkbg1);
    border-color: var(--rinsp-dm-color-border);
}
html.rinsp-dark-mode .tr2,
html.rinsp-dark-mode .h2,
html.rinsp-dark-mode .menu,
html.rinsp-dark-mode .user-infoWrap {
    color: var(--rinsp-dm-color-label);
    background: var(--rinsp-dm-color-darkbg2);
    border-color: var(--rinsp-dm-color-border);
}
html.rinsp-dark-mode .user-infoWrap {
    color: var(--rinsp-dm-color-text);
}
html.rinsp-dark-mode .user-info .co {
    filter: invert(0.5);
}

html.rinsp-dark-mode .h,
html.rinsp-dark-mode .t {
    color: var(--rinsp-dm-color-heading);
    background: var(--rinsp-dm-color-darkbg1);
    border-color: var(--rinsp-dm-color-border);
}
html.rinsp-dark-mode .u-h5 .r,
html.rinsp-dark-mode .u-h5 span {
    background: var(--rinsp-dm-color-darkbg1);
}

html.rinsp-dark-mode .r_two {
    background-color: var(--rinsp-dm-color-darkbg1);
    border-right: 1px dotted var(--rinsp-dm-color-border);
}
html.rinsp-dark-mode th.r_two > .user-pic.rinsp-userpic-sticky {
    background-color: var(--rinsp-dm-color-darkbg1);
}
html.rinsp-dark-mode .tr4 {
    background-color: var(--rinsp-dm-color-darkbg2);
}
html.rinsp-dark-mode .user-infoWraptwo {
    background-color: var(--rinsp-dm-color-darkbg2) !important;
}
html.rinsp-dark-mode .blockquote {
    background-color: var(--rinsp-dm-color-darkbg2);
}
html.rinsp-dark-mode .blockquote2,
html.rinsp-dark-mode .blockquote3 {
    background-color: var(--rinsp-dm-color-darkbg1);
    border-color: var(--rinsp-dm-color-border);
    color: var(--rinsp-dm-color-text);
}
html.rinsp-dark-mode input,
html.rinsp-dark-mode select,
html.rinsp-dark-mode textarea,
html.rinsp-dark-mode #editor-tool span {
    border-color: var(--rinsp-dm-color-border)
}

html.rinsp-dark-mode h1#subject_tpc {
    color: var(--rinsp-dm-color-heading) !important;
}
html.rinsp-dark-mode .rinsp-userframe-userinfo {
    background-color: var(--rinsp-dm-color-darkbg1);
}

html.rinsp-dark-mode img[src="images/colorImagination/post.png"]:not(:hover),
html.rinsp-dark-mode img[src="images/colorImagination/reply.png"]:not(:hover) {
    filter: brightness(0.7);
}
html.rinsp-dark-mode .pages,
html.rinsp-dark-mode .pages ul li,
html.rinsp-dark-mode .set-h2 {
    border-color: var(--rinsp-dm-color-border) !important;
}
html.rinsp-dark-mode .pages ul li b {
    background: var(--rinsp-dm-color-darkbg-hgt);
    color: var(--rinsp-dm-color-text);
}
html.rinsp-dark-mode .pages ul li a:hover {
    background-color: var(--rinsp-dm-color-darkbg2);
}

html.rinsp-dark-mode .rinsp-message-selection-panel,
html.rinsp-dark-mode #set-content {
    background: var(--rinsp-dm-color-darkbg);
    border-color: var(--rinsp-dm-color-darkbg1);
}
html.rinsp-dark-mode .menu #showface {
    background: var(--rinsp-dm-color-darkbg1) !important;
}
html.rinsp-dark-mode .menu #buttons.face {
    filter: var(--rinsp-dm-invert-filter);
}
html.rinsp-dark-mode #smiliebox {
    background: var(--rinsp-dm-color-darkbg1) !important;
    border-color: var(--rinsp-dm-color-darkbg1) !important;
}
html.rinsp-dark-mode #infobox {
    background: var(--rinsp-dm-color-black);
    border-color: var(--rinsp-dm-color-darkbg1);
}
html.rinsp-dark-mode .btn {
    background: var(--rinsp-dm-button-bg);
}
html.rinsp-dark-mode .t5,
html.rinsp-dark-mode .tips,
html.rinsp-dark-mode th,
html.rinsp-dark-mode td,
html.rinsp-dark-mode .btn,
html.rinsp-dark-mode .abtn,
html.rinsp-dark-mode #set-content-wrap {
    border-color: var(--rinsp-dm-color-border);
}
html.rinsp-dark-mode .blockquote {
    color: var(--rinsp-dm-color-text);
    border-color: var(--rinsp-dm-color-light-border);
    background: var(--rinsp-dm-color-darkbg1);
}
html.rinsp-dark-mode #editor-tab span,
html.rinsp-dark-mode #editor-button {
    color: #000;
    filter: var(--rinsp-dm-invert-filter-blk);
}
html.rinsp-dark-mode a {
    color: var(--rinsp-dm-color-link);
}
html.rinsp-dark-mode .tpc_content a {
    color: var(--rinsp-dm-color-content-link);
}
html.rinsp-dark-mode .tpc_content a:hover {
    color: var(--rinsp-dm-color-content-link-hover);
}

html.rinsp-dark-mode img[src="images/post/c_editor/del.gif"],
html.rinsp-dark-mode img[src="images/colorImagination/old.gif"],
html.rinsp-dark-mode img[src="images/colorImagination/new.gif"],
html.rinsp-dark-mode img[src="images/colorImagination/lock.gif"],
html.rinsp-dark-mode img[src^="images/post/editor/"],
html.rinsp-dark-mode img[src^="/images/post/smile/"],
html.rinsp-dark-mode img[src^="images/post/smile/"] {
    filter: var(--rinsp-dm-invert-filter-darkbg1);
}

html.rinsp-dark-mode .set-tab-table {
    background: transparent;
    border-bottom: 1px solid var(--rinsp-dm-color-border);
}
html.rinsp-dark-mode .set-tab-table td.current {
    background-color: var(--rinsp-dm-color-darkbg1);
    border-color: var(--rinsp-dm-color-light-border);
    color: var(--rinsp-dm-color-text);
    font-weight: normal;
    border-bottom: 1px solid var(--rinsp-dm-color-darkbg1);
    position: relative;
    top: 1px;
}
html.rinsp-dark-mode .set-tab-table td:not(.current) {
    border-color: transparent;
    position: relative;
    top: 1px;
}
html.rinsp-dark-mode .set-tab-table td.current a {
    color: var(--rinsp-dm-color-text);
}
html.rinsp-dark-mode .set-tab-box {
    border-color: transparent;
    background-color: var(--rinsp-dm-color-darkbg1);
}
html.rinsp-dark-mode .tt2 {
    border-color: var(--rinsp-dm-color-border);
}
html.rinsp-dark-mode .tt2 table {
    background: var(--rinsp-dm-color-darkbg1);
}
html.rinsp-dark-mode form[action="message.php"] .set-table2 > tbody > tr:not(.gray3):nth-child(odd) {
    background-color: var(--rinsp-dm-color-darkbg-hgt);
}
html.rinsp-dark-mode form[action="message.php"]:not([onsubmit="return checkCnt();"]) .set-table2 > tbody > tr:NOT(.gray3):hover {
    background-color: var(--rinsp-dm-color-darkbg-hover);
}
html.rinsp-dark-mode .set-table2 td {
    padding: .5em 1em;
    border-bottom: 1px dotted var(--rinsp-dm-color-light-border);
}
html.rinsp-dark-mode form[action="message.php"] a[href^="message.php?"]:not(.b) {
    color: var(--rinsp-dm-color-ctrl);
}

html.rinsp-dark-mode .z .tr3:hover,
html.rinsp-dark-mode .threadlist li a:hover {
    background-color: var(--rinsp-dm-color-black);
}
html.rinsp-dark-mode .menu .ul2 li a:hover,
html.rinsp-dark-mode .threadlist li.current a,
html.rinsp-dark-mode .threadlist li.current a:hover {
    background-color: var(--rinsp-dm-color-darkbg1);
    color: var(--rinsp-dm-color-text);
}

html.rinsp-dark-mode .rinsp-notification-item {
    background-color: var(--rinsp-dm-color-darkbg);
}
html.rinsp-dark-mode .rinsp-notification-item {
    color: var(--rinsp-dm-color-grey);
}
html.rinsp-dark-mode #peacemakerconfig .blackListPlan {
    filter: var(--rinsp-dm-invert-filter);
    border-color: transparent;
}

html.rinsp-dark-mode .rinsp-pinuser-item-face > div {
    background-color: var(--rinsp-dm-color-darkbg1);
}
html.rinsp-dark-mode .rinsp-pinuser-item:hover .rinsp-pinuser-item-face > div {
    background-color: var(--rinsp-dm-color-darkbg-hover);
}
html.rinsp-dark-mode .rinsp-thread-filter-like:not(.rinsp-thread-filter-dislike) {
    background: linear-gradient(to right, var(--rinsp-text-color-green) 2px, transparent 3px);
}
html.rinsp-dark-mode .rinsp-thread-filter-like:not(.rinsp-thread-filter-dislike):hover {
    background: linear-gradient(to right, var(--rinsp-text-color-green) 2px, #142707 3px, transparent 50%);
}

html.rinsp-dark-mode .rinsp-sharetype-filter {
    filter: var(--rinsp-dm-invert-filter-blk);
}
html.rinsp-dark-mode .rinsp-sharetype-filter > .rinsp-sharetype-item {
    color: #333;
}

html.rinsp-dark-mode .rinsp-visited-thread-view-mode .rinsp-thread-visited:not(:hover) {
    opacity: 0.7;
}
html.rinsp-dark-mode img[src="/images/noimageavailble_icon.png"] {
    filter: var(--rinsp-dm-invert-filter-blk);
    opacity: 0.5;
}
html.rinsp-dark-mode .u-table .fav.forum {
    background-image: url();
}
html.rinsp-dark-mode .u-table .fav.current {
    background-image: url();
}

html.rinsp-dark-mode .u-table .fav {
    background-image: url();
}

html.rinsp-dark-mode .rinsp-reply-refresh-free.rinsp-refresh-free-submitting::after {
    background: #00000099;
}
html.rinsp-dark-mode .rinsp-thread-filter-pinned .tr1 .r_two,
html.rinsp-dark-mode .rinsp-thread-filter-pinned th.r_two > .rinsp-userpic-sticky {
    background-color: var(--rinsp-dm-color-darkbg1);
}

html.rinsp-dark-mode .rinsp-highlight-spots > dl {
    opacity: 0.8;
}
html.rinsp-dark-mode .t_one h3 > a > b > font[color] {
    filter: var(--rinsp-dm-hgt-text-filter) !important;
    opacity: 1 !important;
}

html.rinsp-dark-mode .rinsp-userframe-userinfo,
html.rinsp-dark-mode .rinsp-filter-ignored .r_one {
    background: transparent;
}
html.rinsp-dark-mode .rinsp-dialog-modal-mask {
    background: #0008;
}

html.rinsp-dark-mode .t_one td > a[title="打开新窗口"] {
    filter: var(--rinsp-dm-invert-filter-darkbg1);
}

html.rinsp-dark-mode .rinsp-answered-request-ignore-toggler,
html.rinsp-dark-mode .rinsp-unanswered-request-ignore-toggler,
html.rinsp-dark-mode .rinsp-expired-request-ignore-toggler {
    background: var(--rinsp-active-toggler-bg-color);
}

html.rinsp-dark-mode .jumbotron.rinsp-sell-free .btn-danger {
    background: #0f1e2d;
}
html.rinsp-dark-mode .jumbotron.rinsp-sell-5 .btn-danger {
    background: #312705;
}
html.rinsp-dark-mode .jumbotron.rinsp-sell-100 .btn-danger {
    background: #490909;
}
html.rinsp-dark-mode .jumbotron.rinsp-sell-high .s3 {
    background: transparent;
    outline-color: transparent;
}
html.rinsp-dark-mode .jumbotron.rinsp-sell-high .btn-danger {
    background: #950000;
}
html.rinsp-dark-mode .jumbotron.rinsp-sell-99999 .btn-danger {
    background: transparent;
    color: #fff;
    font-weight: normal;
}

html.rinsp-dark-mode .rinsp-pinuser-item-face img {
    filter: brightness(0.8) contrast(1.1);
}
html.rinsp-dark-mode .user-pic img {
    filter: brightness(0.8) contrast(1.1);
    border-color: var(--rinsp-dm-color-darkbg1);
    background: var(--rinsp-dm-color-darkbg1);
}

html.rinsp-dark-mode .rinsp-quickping-subject {
    background: var(--rinsp-dm-color-darkbg1);
    color: var(--rinsp-dm-color-text);
}
html.rinsp-dark-mode .rinsp-quickping-preview-section,
html.rinsp-dark-mode .rinsp-quickping-form dd {
    color: var(--rinsp-dm-color-text);
}
html.rinsp-dark-mode .h2 a.rinsp-quickping-status {
    filter: var(--rinsp-dm-invert-filter);
}
html.rinsp-dark-mode .h2 a.rinsp-instantping-button {
    color: #0ac679;
}
`;

const devCss = `
body .rinsp-dev-toggle {
    opacity: 1;
    cursor: default;
}
*:focus-visible {
    outline: none;
}
#header a[href][onclick^="ga("] {
    display: none;
}
`;

const RESOURCE_AREA_IDS = new Set([14,128,4,73,5,6,201,135,142]);
const PAYWALL_AREA_IDS = new Set([171,172,173,174]);

const DEV_MODE = checkDevMode();
const DEBUG_MODE = localStorage.getItem('rinsp-debug-mode') === '1';
const TEMP_INCREMENT = 10000;
const MIN_REQUEST_DELAY = 5000;
const MAX_ACC_REQUEST_RATE_COUNT = 20;
const MAX_ACC_REQUEST_RATE_BASE = 60000 * 5;
const MIN_CHECK_INTERVAL = 'MTA';
const MAX_CHECK_INTERVAL = 'ODAw';
const MIN_CHECK_INTERVAL_OLDEST_ITEM = 'NQ';
const MAX_CHECK_INTERVAL_OLDEST_ITEM = 'NjAw';
const MAX_WATCH_ITEM_COUNT = DEV_MODE&&'NTA'||'MjA';
const OLD_ITEM_THRESHOLD_DAYS = 14;
const OLD_ITEM_SCALING = 1.3;
const MAX_IGNORE_CONTENT_HTML_LENGTH = 2048;
const MAX_IGNORE_CONTENT_TEXT_LENGTH = 512;

const MAX_RECENT_ACCESS_LOG = DEV_MODE ? 5000 : 2000;

const WATCHER_POPUP_MENU_ID = 'rinsp-watcher-menu';
const IGNORE_LIST_POPUP_MENU_ID = 'rinsp-ignorelist-menu';
const THREAD_FILTER_POPUP_MENU_ID = 'rinsp-threadfilter-menu';
const THREAD_LIKE_POPUP_MENU_ID = 'rinsp-threadlike-menu';
const THREAD_FILTER_ACTION_POPUP_MENU_ID = 'rinsp-threadfilter-action-menu';
const USER_BOOKMARK_ACTION_POPUP_MENU_ID = 'rinsp-userbookmark-action-menu';
const SCORING_DIALOG_POPUP_MENU_ID = 'rinsp-scoring-dialog';
const LOADING_DIALOG_POPUP_MENU_ID = 'rinsp-loading-dialog';
const SCORING_REASON_TEMPLATE_POPUP_MENU_ID = 'rinsp-scoring-reasonlist-menu';

const darkModeEnabledProperty = createLocalStorageProperty('rinsp-dark-mode');
const darkModeThemeProperty = createLocalStorageProperty('rinsp-dark-theme', 'darkbeige');
const darkModeFollowsSystemProperty = createLocalStorageProperty('rinsp-dark-mode-auto');

const textEncoder = (function() {
    try {
        return new TextEncoder();
    } catch (e) {
        return null;
    }
})();

async function getNumMapValue(key) {
    const strValue = await GM.getValue(key);
    if (strValue) {
        try {
            const items = strValue.split(' ')
                .map(item => {
                    const pair = item.split(':');
                    return [ pair[0] * 1, (pair[1] * 1)||0 ];
                })
                .filter(pair => !Number.isNaN(pair[0]));
            return new Map(items);
        } catch (ex) {}
    }
    return new Map();
}

async function setNumMapValue(key, mapEntries) {
    const strValue = Array.from(mapEntries).map(entry => entry[1] > 0 ? `${entry[0]}:${entry[1]}` : `${entry[0]}`).join(' ');
    if (strValue) {
        await GM.setValue(key, strValue);
    } else {
        await GM.deleteValue(key).catch(ex => null);
    }
}

function createIdStore(namespace, bucketSize) {

    const indexKey = `${namespace}:index`;

    function getBucketKey(bucketId) {
        return `${namespace}:#${bucketId}`;
    }

    async function adjustCount(bucketId, adjustment) {
        const bucketSizeMap = await getNumMapValue(indexKey);
        const current = bucketSizeMap.get(bucketId) || 0;
        const newCount = current + adjustment;
        if (newCount > 0) {
            bucketSizeMap.set(bucketId, newCount);
        } else {
            bucketSizeMap.delete(bucketId);
        }
        await setNumMapValue(indexKey, bucketSizeMap.entries());
    }

    return {
        async set(id, numValue) {
            const bucketId = Math.floor(id / bucketSize);
            const localId = id % bucketSize;
            const localMap = await getNumMapValue(getBucketKey(bucketId));
            const current = localMap.get(localId);
            const newValue = numValue||0;
            if (current == null || current !== newValue) {
                localMap.set(localId, newValue);
                await setNumMapValue(getBucketKey(bucketId), localMap.entries());
            }
            if (current == null) {
                await adjustCount(bucketId, 1);
            }
        },
        async remove(id) {
            const bucketId = Math.floor(id / bucketSize);
            const localId = id % bucketSize;
            const localMap = await getNumMapValue(getBucketKey(bucketId));
            if (localMap.get(localId) != null) {
                localMap.delete(localId);
                await setNumMapValue(getBucketKey(bucketId), localMap.entries());
                await adjustCount(bucketId, -1);
            }
        },
        async get(id) {
            const bucketId = Math.floor(id / bucketSize);
            const localId = id % bucketSize;
            const localMap = await getNumMapValue(getBucketKey(bucketId));
            return localMap.get(localId);
        },
        async getBatch(ids) {
            if (ids.length === 0)
                return [];
            if (ids.length === 1)
                return [this.get(ids[0])];
            const buckets = new Map();
            const answer = ids.map(id => null);
            ids.forEach((id, i) => {
                const bucketId = Math.floor(id / bucketSize);
                const localId = id % bucketSize;
                const entry = [localId, i];
                let localEntries = buckets.get(bucketId);
                if (localEntries == null) {
                    localEntries = [entry];
                    buckets.set(bucketId, localEntries);
                } else {
                    localEntries.push(entry);
                }
            });
            async function loadBucket(bucketId, localEntries) {
                const localMap = await getNumMapValue(getBucketKey(bucketId));
                localEntries.forEach(localEntry => {
                    const value = localMap.get(localEntry[0]);
                    answer[localEntry[1]] = value >= 0 ? value : null;
                });
            }
            // console.info('buckets', buckets);
            const promises = [];
            Array.from(buckets.entries()).forEach(entry => {
                promises.push(loadBucket(entry[0], entry[1]));
            });
            await Promise.allSettled(promises);
            return answer;
        },
        async size() {
            const bucketSizeMap = await getNumMapValue(indexKey);
            let sum = 0;
            for (let count of bucketSizeMap.values()) {
                sum += count;
            }
            return sum;
        },
        async clear() {
            const keys = await GM.listValues();
            const results = [];
            keys.filter(key => key.startsWith(namespace + ':')).forEach(key => {
                results.push(GM.deleteValue(key));
            });
            await Promise.allSettled(results).catch(ex => null);
        }
    };
}

function createRecordStore(namespace, maxSize) {

    const indexKey = maxSize >= 0 ? `${namespace}:recent` : null;

    function getItemKey(id) {
        return `${namespace}:#${id}`;
    }

    return {
        async put(id, data) {
            if (indexKey) {
                const now = Date.now();
                const recordMap = await getNumMapValue(indexKey);
                const baseValueOffset = recordMap.get(0)||0; // special value
                recordMap.delete(0);
                const records = [];
                records.push([id, now]);
                recordMap.forEach((value, key) => {
                    if (key !== id) {
                        records.push([key, value + baseValueOffset]);
                    }
                });
                records.sort(comparator(1, true)); // bigger number first
                const removedRecords = [];
                while (records.length > maxSize) {
                    removedRecords.push(records.pop());
                }
                let minValue = records[records.length - 1][1];
                records.forEach(record => {
                    record[1] = record[1] - minValue;
                });
                records.push([0, minValue]);
                await setNumMapValue(indexKey, records);
                for (let removedRecord of removedRecords) {
                    await GM.deleteValue(getItemKey(removedRecord[0])).catch(ex => null);
                }
            }
            await GM.setValue(getItemKey(id), JSON.stringify(data));
        },
        async get(id) {
            const value = await GM.getValue(getItemKey(id));
            return value == null ? null : JSON.parse(value);
        },
        async remove(id) {
            await GM.deleteValue(getItemKey(id)).catch(ex => null);
            if (indexKey) {
                const recordMap = await getNumMapValue(indexKey);
                recordMap.remove(id);
                await setNumMapValue(indexKey, recordMap.entries());
            }
        },
        async list() {
            const keys = await GM.listValues();
            const queue = [];
            keys.filter(key => key.startsWith(namespace + ':#')).forEach(key => {
                queue.push(GM.getValue(key));
            });
            const values = await Promise.all(queue);
            return values.map(value => JSON.parse(value));
        },
        async clear() {
            const keys = await GM.listValues();
            const results = [];
            keys.filter(key => key.startsWith(namespace + ':')).forEach(key => {
                results.push(GM.deleteValue(key));
            });
            await Promise.allSettled(results).catch(ex => null);
        }
    };
}

function createUserHashLookupStore() {
    // uhashStore <uhash-base10: uid>
    return createIdStore('uhash_uid_mapping', 1000); // bucket size = 1000, DO NOT CHANGE
}

function createThreadHistoryStore(myUserId, userConfig) {
    if (!userConfig.keepVisitPostHistory) {
        return null;
    }

    // historyStore <tid: maxFloor>
    const historyStore = createIdStore(`tid_visit_history#${myUserId}`, 1000); // bucket size = 1000, DO NOT CHANGE
    const recentAccessStore = createRecordStore(`tid_recent#${myUserId}`, MAX_RECENT_ACCESS_LOG);
    return {
        historyStore, recentAccessStore
    };
}

function findMyUserId() {
    const userWrap = document.querySelector('#user_info #showface .user-infoWraptwo');
    if (userWrap != null) {
        const userMatch = userWrap.textContent.match(/\sUID: +(\d+)\s/);
        if (userMatch != null) {
            return userMatch[1] * 1;
        }
    }
    const selfInfo = document.querySelector('#menu_profile .ul2 a[href^="u.php?action-show-uid-"]');
    if (selfInfo != null && selfInfo.textContent === '查看个人资料') {
        return Number.parseInt(selfInfo.getAttribute('href').substring(22))||null;
    }
    return null;
}

function hasMentionedMe(title, myUserId, userConfig) {
    if (title.indexOf('@' + userConfig.myUserHashId) !== -1) // e.g. @a89f389e
        return true;
    if (new RegExp('(^|[^A-Za-z0-9])' + escapeRegExp(userConfig.myUserHashId) + '([^A-Za-z0-9]|$)').exec(title) != null) // e.g. a89f389e
        return true;
    
    if (new RegExp('(^|[^0-9])' + myUserId + '([^0-9]|$)').exec(title) != null) {
        if (title.indexOf('#' + myUserId) !== -1) // e.g. #1234567
            return true;
        if (title.match(/(^|[^a-z])u?id([^a-z]|$)/i) != null) // e.g. uid 1234567
            return true;
    }

    const nickNameNorm = (userConfig.myNickName||'').trim().replace(/\s+/, ' ');
    if (nickNameNorm) {
        const titleNorm = title.replace(/\s+/, ' ');
        if (new RegExp('@' + escapeRegExp(nickNameNorm)).exec(titleNorm) != null)
            return true;
        if (new RegExp('(^|请|给|用户|结算|\s|,|。|,|:)' + escapeRegExp(nickNameNorm) + '(\s|大佬|老哥|兄弟|用户|结算|请|,|。|,|$)').exec(titleNorm) != null)
            return true;
    }
}

function getDefaultRating(postTitle) {
    const ownBought = postTitle.match(/自购|自購/) != null;
    const ownTranslate = postTitle.match(/[个個]人(汉化|漢化|翻译|翻譯)/) != null;
    const sizeMatches = Array.from(postTitle.toUpperCase().replace(/\b(R18G)\b/g, '').matchAll(/(?<![0-9.])((?:\d\d?,?)?\d{1,3}(?:\.\d{1,3})?) *([GMKT](?:I?B)?)(?![A-Z])/g));

    // 10M = 1SP
    let baseTotalScore = 0;
    let resourceSizeTotalMB = null;
    let resourceSizeMBs = [];
    let resourceSizeTexts = [];
    if (sizeMatches.length > 0) {
        resourceSizeTotalMB = 0;
        sizeMatches.forEach(sizeMatch => {
            let matchSizeMB = sizeMatch[1].replace(/,/g, '') * 1;
            if (sizeMatch[2] === 'K' && (sizeMatch[1] === '2' || sizeMatch[1] === '4')) {
                // ignore 2K , 4K
                return;
            }
            switch (sizeMatch[2][0]) {
                case 'K':
                    matchSizeMB /= 1000;
                    break;
                case 'G':
                    matchSizeMB *= 1000;
                    break;
                case 'T':
                    matchSizeMB *= 1000000;
                    break;
                case 'M':
                default:
                    break;
            }
            resourceSizeTexts.push(sizeMatch[0]);
            resourceSizeMBs.push(matchSizeMB);
            resourceSizeTotalMB += matchSizeMB;
        });
        if (resourceSizeMBs.length === 1) {
            baseTotalScore = computeBaseScoreText(resourceSizeTotalMB) * 1;
        } else {
            baseTotalScore = 0;
        }
    }
    return {
        resourceSizeTexts,
        resourceSizeMBs,
        resourceSizeTotalMB,
        baseTotalScore,
        ownBought,
        ownTranslate
    };
}

function computeBaseScoreText(resourceSizeMB) {
    return Math.max(1, resourceSizeMB / 10).toFixed(1).replace(/\.0+$/, '');
}

const PAGE_TITLE_RULES = {
    u_php(queryParams, keyword, myUserId, userConfig, adminRole) {
        const myself = queryParams.uid == null || queryParams.uid * 1 === myUserId;
        let suffix = '';
        if (!myself && queryParams.uid) {
            const userId = queryParams.uid * 1;
            const knownUser = userConfig.customUserHashIdMappings['#' + userId];
            if (knownUser) {
                suffix += '| ' + knownUser[2];
            } else {
                suffix += '| #' + queryParams.uid;
            }
        }
        if (queryParams.action == null) {
            return queryParams.uid == null || myself ? '我的个人首页' : '个人首页' + suffix;
        } else if (queryParams.action === 'feed') {
            return myself ? '我的好友近况' : '好友近况' + suffix;
        } else if (queryParams.action === 'show') {
            return myself ? '我的资料' : '用户资料' + suffix;
        } else if (queryParams.action === 'topic') {
            return myself ? '我的主题' : '主题列表' + suffix;
        } else if (queryParams.action === 'post') {
            return myself ? '我的回复' : '回复列表' + suffix;
        } else if (queryParams.action === 'favor') {
            return myself ? '我的收藏' : '收藏列表' + suffix;
        } else if (queryParams.action === 'friend') {
            return myself ? '我的好友' : '好友列表' + suffix;
        } else if (queryParams.action === 'trade') {
            if (myself) {
                if (adminRole && queryParams.view === 'admin') {
                    return '我的管理记录';
                } else {
                    return '我的浏览记录';
                }
            } else if (adminRole) {
                return '不良记录' + suffix;
            } else {
                return '商品列表' + suffix;
            }
        }
    },
    profile_php(queryParams, keyword, myUserId, userConfig) {
        return '用户中心';
    },
    message_php(queryParams, keyword, myUserId, userConfig) {
        function withSubject(title) {
            const subjectCell = document.querySelector('#info_base .set-table2 > tbody > tr:nth-child(2) > td:nth-child(2)');
            if (subjectCell) {
                return title + '| ' + subjectCell.textContent;
            } else {
                return title;
            }
        }
        if (queryParams.action == null || queryParams.action === 'receivebox') {
            return '📩收件箱';
        } else if (queryParams.action == 'sendbox') {
            return '✉️发件箱';
        } else if (queryParams.action == 'readsnd') {
            return withSubject('✉️发件箱');
        } else if (queryParams.action == 'write') {
            if (queryParams.remid) {
                return '✉️' + keyword;
            } else {
                return '✉️发短信';
            }
        } else if (queryParams.action == 'read') {
            return withSubject('📩读短信');
        } else if (queryParams.action == 'scout') {
            return '✉️消息跟踪';
        } else if (queryParams.action == 'readscout') {
            return withSubject('✉️消息跟踪');
        } else if (queryParams.action == 'chatlog') {
            return withSubject('✉️通信记录');
        }
    },
    post_php(queryParams, keyword, myUserId, userConfig) {
        if (queryParams.action == null) {
            const currentArea = document.querySelector('#breadcrumbs > .crumbs-item.current > strong > a');
            return '✏️发新帖' + (currentArea ? '| ' + currentArea.textContent : '');
        } else if (queryParams.action == 'modify') {
            return '✏️编辑| ' + keyword;
        } else if (queryParams.action == 'reply') {
            return '↩️回帖| ' + keyword;
        }
    },
    search_php(queryParams, keyword, myUserId, userConfig) {
        const message = document.querySelector('#main .t .f_one td center');
        const prefix = keyword ? keyword + '| ' : '';
        if (message && message.textContent.trim() === '没有查找匹配的内容') {
            return prefix + '🚫没搜索结果';
        }
        const totalElem = document.querySelector('#main .bdbA + .t3 ~ .fr');
        if (totalElem) {
            const total = (totalElem.textContent.match(/共搜索到了(\d+)条信息/)||[])[1] * 1;
            if (total > 0) {
                return prefix + `🔍搜索到${total}条`;
            }
        }
        return prefix + '🔍搜索';
    },
    mawhole_php(queryParams, keyword, myUserId, userConfig) {
        switch (queryParams.action||'') {
            case 'headtopic': return '置顶| 帖子管理';
            case 'digest': return '精华| 帖子管理';
            case 'lock': return '锁定| 帖子管理';
            case 'pushtopic': return '提前| 帖子管理';
            case 'downtopic': return '压贴| 帖子管理';
            case 'edit': return '加亮| 帖子管理';
            case 'move': return '移动| 帖子管理';
            case 'copy': return '复制| 帖子管理';
            case 'unite': return '合并| 帖子管理';
            case 'del': return '删除| 帖子管理';
            default: return '帖子管理';
        }
    },
    masingle_php(queryParams, keyword, myUserId, userConfig) {
        switch (queryParams.action||'') {
            case 'banuser': return '禁言| 管理';
            case 'shield': return '屏蔽| 管理';
            case 'remind': return '提醒| 管理';
            case 'delatc': return '删除回复| 管理';
            default: return '管理';
        }
    },
    operate_php(queryParams, keyword, myUserId, userConfig) {
        switch (queryParams.action||new URLSearchParams(document.location.search).get('action')||'') {
            case 'showping': return '评分| 管理';
            default: return '管理';
        }
    },
    job_php(queryParams, keyword, myUserId, userConfig) {
        switch (queryParams.action||new URLSearchParams(document.location.search).get('action')||'') {
            case 'endreward': return '悬赏帖结案 | 管理';
            default: return '管理';
        }
    },
    sendemail_php(queryParams, keyword, myUserId, userConfig) {
        return '发送邮件';
    },
    member_php(queryParams, keyword, myUserId, userConfig) {
        return '会员列表';
    },
    sort_php(queryParams, keyword, myUserId, userConfig) {
        switch (queryParams.action||'') {
            case '': return '基本统计| 统计排行';
            case 'ipstate': return '到访IP统计| 统计排行';
            case 'team': return '管理团队| 统计排行';
            case 'admin': return '管理统计| 统计排行';
            case 'online': return '在线统计| 统计排行';
            case 'member': return '会员排行| 统计排行';
            case 'forum': return '版块排行| 统计排行';
            case 'article': return '帖子排行| 统计排行';
            default: return '统计排行';
        }
    },
    show_php(queryParams, keyword, myUserId, userConfig) {
        return '展区'; // dead area
    },
    push_php(queryParams, keyword, myUserId, userConfig) {
        return '推荐主题'; // dead area
    },
    forumcp_php(queryParams, keyword, myUserId, userConfig) {
        return '版块管理';
    },
    plugin_php(queryParams, keyword, myUserId, userConfig) {
        if (queryParams.H_name == 'tasks') {
            switch (queryParams.actions||'') {
                case '': return '新任务选择| 社区论坛任务';
                case 'newtasks': return '进行中任务| 社区论坛任务';
                case 'endtasks': return '已完成任务| 社区论坛任务';
                case 'errotasks': return '已失败任务| 社区论坛任务';
                default: return '社区论坛任务';
            }
        }
        return '附加功能';
    },
};

function enhancePageTitle(queryParams, myUserId, userConfig, adminRole) {
    const originalTitle = document.title;
    const parts = originalTitle.match(/^(?:(.+) )?([^ -]+\+[^+-]+) - powered by Pu!mdHd$/);
    if (parts == null) {
        return;
    }
    const keyTitle = (parts[1]||'').replace(/ ? - ?$/, '');
    const siteName = parts[2];
    const key = document.location.pathname.replace(/^\//, '').replace(/\./g, '_');
    const handler = PAGE_TITLE_RULES[key];
    let keyword = null;
    if (handler) {
        keyword = handler(queryParams, keyTitle||'', myUserId, userConfig, adminRole);
    }
    if (keyword == null) {
        keyword = keyTitle||'';
    }
    if (keyword) {
        document.title = keyword.trim() + ' - ' + siteName;
    } else {
        document.title = siteName;
    }
}

function initSiteMainMenu(hasManagementRole) {
    const mainGuideBar = document.querySelector('#guide');
    if (hasManagementRole) {
        // add link to 系统设置 if absent
        if (document.querySelector('#guide > li > a[href="admin.php"]') == null) {
            const adminItem = addElem(mainGuideBar, 'li');
            addElem(adminItem, 'a', null, { href: 'admin.php' }).textContent = '系统设置';
        }
    }
}

function addModalMask() {
    return addElem(document.body, 'div', 'rinsp-dialog-modal-mask');
}

function createPopupMenu(popupId, anchor, rightAligned, verticallyInverted) {
    function computeStyle(hShift) {
        if (anchor) {
            const bounds = anchor.getClientRects()[0];
            const left = bounds.x + hShift;
            if (verticallyInverted) {
                return 'opacity: 0.95; left: ' + left.toFixed(0) + 'px; z-index: 3000; visibility: visible; top: ' + (bounds.top + document.documentElement.scrollTop).toFixed(0) + 'px; transform: translateY(-100%);';
            } else {
                return 'opacity: 0.95; left: ' + left.toFixed(0) + 'px; z-index: 3000; visibility: visible; top: ' + (bounds.height + bounds.top + document.documentElement.scrollTop).toFixed(0) + 'px;';
            }
        } else {
            return 'position: fixed; opacity: 0.95; left: calc(50vw - 400px); z-index: 3000; visibility: visible; top: 10vh';
        }

    }
    const menuElem = addElem(document.body, 'div', 'menu rinsp-common-popup-menu', {
        id: popupId,
        style: computeStyle(0)
    });

    // just copy-n-paste from site, lol
    menuElem.innerHTML = `<div class="bor" style="padding:13px 30px"><img src="images/loading.gif" align="absbottom"> 正在加载数据...</div>`;

    return {
        renderContent(renderer) {
            menuElem.innerHTML = '';
            const borElem = addElem(menuElem, 'div', 'bor');
            renderer(borElem);
            if (rightAligned) {
                menuElem.setAttribute('style', computeStyle(-menuElem.getClientRects()[0].width));
            }
        }
    };
}

async function setupPopupMenu(config) {
    const menuConfig = {
        title: '',
        width: 120,
        popupMenuId: '',
        anchor: null,
        rightAligned: false,
        verticallyInverted: false,
        onClose: () => {},
        items: [] // [ { label, class, action } ]
    };
    Object.assign(menuConfig, config);

    const modelMask = addModalMask();
    const popupMenu = createPopupMenu(menuConfig.popupMenuId, menuConfig.anchor, menuConfig.rightAligned, menuConfig.verticallyInverted);
    function close() {
        modelMask.remove();
        closePopupMenu(menuConfig.popupMenuId);
        menuConfig.onClose();
    }
    modelMask.addEventListener('click', () => close());

    popupMenu.renderContent(async function(borElem) {
        const tableElem = addElem(borElem, 'table', null, {
            width: `${menuConfig.width}`,
            cellspacing: '0',
            cellpadding: '0'
        });

        const tbodyElem = addElem(tableElem, 'tbody');
        const trElem1 = addElem(tbodyElem, 'tr');
        const thElem1_1 = addElem(trElem1, 'th', 'h');
        const frElem1 = addElem(thElem1_1, 'span', 'fr', {
            style: 'margin-top:2px;cursor:pointer'
        });
        frElem1.addEventListener('click', () => close());
        addElem(frElem1, 'img', null, {
            src: 'images/close.gif'
        });
        thElem1_1.appendChild(document.createTextNode(menuConfig.title));

        function addButton(item) {
            const trElem = addElem(tbodyElem, 'tr');
            const tdElem = addElem(trElem, 'td', item.cellClass||null);
            const button = addElem(tdElem, 'a', item.class||null);
            button.textContent = item.label;
            if (item.action == null) {
                // nothing
            } else if (typeof item.action === 'string') {
                button.setAttribute('href', item.action);
                if (item.target) {
                    button.setAttribute('target', item.target);
                }
            } else {
                button.setAttribute('href', 'javascript:');
                button.addEventListener('click', () => {
                    close();
                    setTimeout(() => item.action());
                    return false;
                });
            }
            return button;
        }
        menuConfig.items.forEach(item => {
            addButton(item);
        });
    });
}

function closePopupMenu(popupId) {
    const menu = document.getElementById(popupId);
    if (menu) {
        menu.remove();
    }
}

async function showMessagePopup(message, anchor, autoCloseTimeout) {
    return runWithProgressPopup(async () => message, '', anchor, autoCloseTimeout);
}

async function runWithProgressPopup(action, loadingMessage, anchor, feedbackTimeout) {
    return new Promise((resolve, reject) => {
        const modelMask = addModalMask();
        const popupMenu = createPopupMenu(LOADING_DIALOG_POPUP_MENU_ID, anchor);
        popupMenu.renderContent(async function(borElem) {
            // not happy about copy-n-paste ...
            borElem.innerHTML = '';
            const loadingContent = addElem(borElem, 'div', null, { style: 'padding: 16px 30px' });
            addElem(loadingContent, 'img', null, { src: 'images/loading.gif', align: 'absbottom' });
            loadingContent.appendChild(document.createTextNode(loadingMessage + ' '));
            const customContent = addElem(borElem, 'div');
        
            action(customContent)
                .then(async status => {
                    borElem.innerHTML = '';
                    const loadingContent = addElem(borElem, 'div', null, { style: 'padding: 16px 30px; font-size: 1.5em' });
                    loadingContent.appendChild(document.createTextNode(status||'✔️已完成'));
                    await sleep(feedbackTimeout);
                    closePopupMenu(LOADING_DIALOG_POPUP_MENU_ID);
                    resolve();
                })
                .catch(ex => {
                    reject(ex);
                })
                .finally(() => modelMask.remove());
        });
    });
}

async function openTextListEditor(popupMenuId, config, callback) {
    const modelMask = addModalMask();
    const popupMenu = createPopupMenu(popupMenuId, null);
    
    const initContent = await config.read();
    popupMenu.renderContent(async function(borElem) {
        const tableElem = addElem(borElem, 'table', 'rinsp-textlist-popup-table', {
            width: '700',
            cellspacing: '0',
            cellpadding: '0'
        });

        const tbodyElem = addElem(tableElem, 'tbody');
        const trElem1 = addElem(tbodyElem, 'tr');
        const thElem1_1 = addElem(trElem1, 'th', 'h');
        const frElem1 = addElem(thElem1_1, 'span', 'fr', {
            style: 'margin-top:2px;cursor:pointer'
        });
        frElem1.addEventListener('click', function() {
            modelMask.remove();
            closePopupMenu(popupMenuId);
        });
        addElem(frElem1, 'img', null, {
            src: 'images/close.gif'
        });
        thElem1_1.appendChild(document.createTextNode(config.title));
        
        const trElem2 = addElem(tbodyElem, 'tr', 'tr2 tac');
        const tdElem2 = addElem(trElem2, 'td');
        if (config.description) {
            addElem(tdElem2, 'div', 'rinsp-textlist-description').textContent = config.description;
        }
        const editorTextArea = addElem(tdElem2, 'textarea', 'rinsp-textlist-editor');
        editorTextArea.value = initContent;

        const footer = addElem(borElem, 'ul', null, {
            style: 'text-align:center;padding:4px 0;'
        });
        const submitButton = addElem(footer, 'input', 'btn', { type: 'button', value: '保存' });
        submitButton.addEventListener('click', async function() {
            tableElem.classList.remove('rinsp-config-saving');
            await config.save(editorTextArea.value)
                .then(function() {
                    modelMask.remove();
                    closePopupMenu(popupMenuId);
                    if (callback) callback();
                })
                .finally(function() {
                    tableElem.classList.remove('rinsp-config-saving');
                });
        });
    });
}

function initConfigAccess(myUserId, configKey, defaultValue) {
    return {
        async read() {
            return readUserConfig(myUserId, configKey, defaultValue);
        },
        async write(newValue) {
            await GM.setValue(configKey + '#' + myUserId, JSON.stringify(newValue));
        },
        async update(updater) {
            return updateUserConfig(myUserId, updater, configKey, defaultValue);
        }
    };
}

async function readUserConfig(myUserId, configKey, defaultValue) {
    try {
        let savedData = await GM.getValue(configKey + '#' + myUserId);
        return Object.assign({}, defaultValue, JSON.parse(savedData));
    } catch (e) {}
    return Object.assign({}, defaultValue);
}

async function updateUserConfig(myUserId, updater, configKey, defaultValue) {
    let oldConfig = await readUserConfig(myUserId, configKey, defaultValue);
    let newConfig = await updater(oldConfig);
    if (newConfig != null) {
        await GM.setValue(configKey + '#' + myUserId, JSON.stringify(newConfig));
    }
    return newConfig;
}

async function populateThreadAccess(tid, myUserId, threadHistoryAccess) {
    const doc = await fetchGetPage(`${document.location.origin}/read.php?tid-${tid}.html`);
    const pageError = findErrorMessage(doc);
    if (pageError) {
        if (pageError.indexOf('数据已被删除') !== -1) {
            await recordThreadDeleted(tid, myUserId, threadHistoryAccess);
        } else {
            throw new Error(pageError);
        }
    } else {
        const posts = getPosts(doc);
        await recordThreadAccess(doc, posts, myUserId, threadHistoryAccess, {});
    }
}

async function recordThreadDeleted(tid, myUserId, threadHistoryAccess) {
    const lastAccessRecord = await threadHistoryAccess.recentAccessStore.get(tid);
    if (lastAccessRecord && !lastAccessRecord.deleted) {
        lastAccessRecord.deleted = true;
        await threadHistoryAccess.recentAccessStore.put(tid, lastAccessRecord);
    }
}

async function recordThreadAccess(doc, posts, myUserId, threadHistoryAccess, memoryObj) {
    const currentPostLink = doc.querySelector('#breadcrumbs .crumbs-item.current strong > a[href^="read.php?tid-"]');
    if (currentPostLink == null) {
        return;
    }
    const areaLink = currentPostLink.closest('.crumbs-item').previousElementSibling;
    if (!areaLink.matches('a[href^="thread.php?fid-"]')) {
        return;
    }
    const fid = Number.parseInt(areaLink.getAttribute('href').substring(15));
    const threadTitle = currentPostLink.textContent.trim();
    const tid = Number.parseInt(currentPostLink.getAttribute('href').substring(13));
    const areaName = areaLink.textContent.trim();
    const uid = posts[0].floor === 0 ? posts[0].postUid : 0;

    const lastPost = posts[posts.length - 1];
    const lastFloor = lastPost.floor;
    if (memoryObj.lastFloor !== lastFloor) {
        if (DEBUG_MODE) console.info('recordThreadAccess', tid, lastFloor);
        await threadHistoryAccess.historyStore.set(tid, lastFloor);
        memoryObj.lastFloor = lastFloor;
    }

    const bountyStatus = getBountyStatus(doc);
    if (memoryObj.lastAccessRecord == null) {
        memoryObj.lastAccessRecord = await threadHistoryAccess.recentAccessStore.get(tid);
    }
    function getNewRecord(lastAccessRecord) {
        let hasBoughtAny = false;
        let lastReplied = 0;
        posts.forEach(post => {
            if (post.postUid === myUserId) {
                if (post.floor > lastReplied) {
                    lastReplied = post.floor;
                }
            } else {
                if (post.contentElem.querySelector('h6.quote.jumbotron + blockquote.jumbotron') != null) {
                    hasBoughtAny = true;
                }
            }
        });

        const newRecord = {
            tid, uid, fid, area: areaName, title: threadTitle, time: Date.now(), bought: hasBoughtAny, replied: lastReplied, bounty: bountyStatus
        };
        if (lastAccessRecord == null) {
            return newRecord;
        }
        if ((lastAccessRecord.replied||0) > lastReplied) {
            newRecord.replied = lastAccessRecord.replied;
        }
        if (bountyStatus == null && lastAccessRecord.bounty != null) {
            newRecord.bounty = lastAccessRecord.bounty;
        }
        let savingRecord = null;
        if (lastAccessRecord.initialTitle) {
            newRecord.initialTitle = lastAccessRecord.initialTitle;
        }
        if (lastAccessRecord.title !== threadTitle) {
            newRecord.bought = newRecord.bought || lastAccessRecord.bought;
            if (!newRecord.initialTitle) {
                newRecord.initialTitle = lastAccessRecord.title;
            }
            savingRecord = newRecord;
        }
        if (!lastAccessRecord.bought && newRecord.bought) {
            savingRecord = newRecord;
        }
        if (!lastAccessRecord.uid && newRecord.uid > 0) {
            savingRecord = newRecord;
        }
        if (lastReplied > (lastAccessRecord.replied||0)) {
            savingRecord = newRecord;
        }
        if (bountyStatus != null) {
            if (lastAccessRecord.bounty == null || lastAccessRecord.bounty.ended != bountyStatus.ended) {
                savingRecord = newRecord;
            }
        }
        return savingRecord;
    }
    const newRecord = getNewRecord(memoryObj.lastAccessRecord);
    if (newRecord) {
        if (DEBUG_MODE) console.info('recordThreadAccess', tid, newRecord);
        await threadHistoryAccess.recentAccessStore.put(tid, newRecord);
        memoryObj.lastAccessRecord = newRecord;
    }
}

function checkManagementRole(myUserId, userConfig, mainConfigAccess) {
    if (document.querySelector('#guide > li > a[href="admin.php"]') != null) {
        return true;
    }
    let hasManageRole = false;
    const userNavTopicLink = document.querySelector('#user_info .fl > .link5[href="u.php?action-topic.html"]');
    if (userNavTopicLink) {
        let rank = (userNavTopicLink.parentElement.textContent.match(/等级:([^,]+),/)||[])[1];
        hasManageRole = ['管理员', '总版主', '论坛版主'].includes(rank);
        if (hasManageRole !== userConfig.hasSeenAdminRole) {
            userConfig.hasSeenAdminRole = hasManageRole;
            mainConfigAccess.update(updatingUserConfig => {
                updatingUserConfig.hasSeenAdminRole = hasManageRole;
                return updatingUserConfig;
            });
        }
    } else {
        // some pages do not have rank information visible, use cached state instead
        hasManageRole = !!userConfig.hasSeenAdminRole;
    }
    return hasManageRole;
}

function showNativeWarningPopup(msg) {
    try {
        unsafeWindow.ajax.request = unsafeWindow.Object();
        unsafeWindow.ajax.request.responseText = msg;
        unsafeWindow.ajax.guide();
    } catch (ignore) {
        alert(msg);
    }
}

function showNativeSpinner(anchor) {
    try {
        unsafeWindow.read.obj = anchor;
        unsafeWindow.read.guide();
    } catch (ignore) {}
}
function closeNativeSpinner() {
    try {
        unsafeWindow.closep();
    } catch (ignore) {}
}

function addFloatingFavorButton(topControlContainer, tid, favorThreadsCacheAccess) {
    const storeButton = addElem(topControlContainer, 'a', 'rinsp-excontrol-item rinsp-excontrol-item-ticker');
    storeButton.classList.add('rinsp-excontrol-item-favor');
    storeButton.classList.add('rinsp-running');
    storeButton.textContent = '✨\n收\n藏';
    storeButton.addEventListener('click', async () => {
        showNativeSpinner();
        let favId = storeButton.dataset.favId * 1;
        if (favId) {
            let data;
            if (favId === -1) {
                data = await readFavorRecords(favorThreadsCacheAccess, true, false);
                if (Object_hasOwn(data, tid)) {
                    favId = data[tid].id;
                } else {
                    favId = null;
                }
            } else {
                data = await readFavorRecords(favorThreadsCacheAccess, false, false);
            }
            delete data[tid];
            await favorThreadsCacheAccess.write(data);
            if (favId != null) {
                const postData = new FormData();
                postData.set('verify', verifyhash());
                postData.set('selid[]', favId);
                postData.set('type', 0);
                postData.set('job', 'clear');
                await fetch(`${document.location.origin}/u.php?action=favor`, {
                    method: 'POST',
                    mode: 'same-origin',
                    credentials: 'same-origin',
                    body: postData
                })
                .then(resp => {
                    if (!resp.ok) throw Error();
                })
                .then(() => {
                    storeButton.dataset.favId = 0;
                    storeButton.classList.remove('rinsp-active');
                })
                .catch(ex => {
                    showNativeWarningPopup('操作出错');
                })
                .finally(() => closeNativeSpinner());
            }
        } else {
            fetchGetPage(`${document.location.origin}/pw_ajax.php?action=favor&tid=${tid}&type=0&nowtime=${Date.now()}&verify=${verifyhash()}`, 'text/xml')
                .then(async xml => {
                    const msg = xml.childNodes[0].textContent;
                    if (['您已经收藏了该主题', '帖子收藏成功!'].includes(msg)) {
                        await favorThreadsCacheAccess.update(cache => {
                            if (cache && cache.data) {
                                if (Object_hasOwn(cache.data, tid)) {
                                    return null;
                                }
                                cache.data[tid] = { tid: tid };
                                return cache;
                            } else {
                                return { data: {}, time: 0 };
                            }
                        });
                    } else {
                        throw msg;
                    }
                })
                .then(() => {
                    storeButton.dataset.favId = -1;
                    storeButton.classList.add('rinsp-active');
                })
                .catch(msg => {
                    showNativeWarningPopup('' + msg);
                })
                .finally(() => closeNativeSpinner());
        }
    });

    readFavorRecords(favorThreadsCacheAccess)
        .catch(ex => null)
        .then(data => {
            storeButton.classList.remove('rinsp-running');
            const favId = data && Object_hasOwn(data, tid) ? data[tid].id || -1 : 0;
            storeButton.dataset.favId = favId;
            if (favId) {
                storeButton.classList.add('rinsp-active');
            }
        });
}

function readFavorList(doc) {
    const items = {};
    doc.querySelectorAll('#u-contentmain .u-table tr > th > a[href^="read.php?tid-"]').forEach(a => {
        const id = a.closest('tr').querySelector('input[name="selid[]"]').value * 1;
        const tid = Number.parseInt(a.getAttribute('href').substring(13));
        const opLink = a.closest('tr').querySelector('a[href^="u.php?uid-"]');
        const opId = Number.parseInt(opLink.getAttribute('href').substring(10));
        const category = opLink.closest('td').nextElementSibling.textContent.trim();
        items[tid] = {
            id,
            tid,
            op: opId,
            category
        };
    });
    return items;
}

function readCurrentAndMaxPage(doc) {
    let currentPage = 1;
    let maxPage = 1;
    const pagesElem = doc.querySelector('.pages li.pagesone');
    if (pagesElem != null) {
        let match = pagesElem.textContent.replace(/\u00a0/g, ' ').match(/Pages: *(\d+)[^\/]*\/ *(\d+) */);
        if (match != null) {
            currentPage = Number.parseInt(match[1]);
            maxPage = Number.parseInt(match[2]);
        }
    }
    return [currentPage, maxPage];
}

function isCurrentThreadMatchingFid(fid) {
    if (document.location.pathname === '/thread.php') {
        return document.location.search.startsWith(`?fid-${fid}-`) ||
               document.location.search.startsWith(`?fid-${fid}.`) ||
               document.location.search.startsWith(`?fid=${fid}&`) ||
               document.location.search.endsWith(`?fid=${fid}`) ||
               document.location.search.indexOf(`&fid=${fid}&`);
    }
    return false;
}

function addBackToTopButton(container) {
    const button = addElem(container, 'a', 'rinsp-excontrol-item rinsp-excontrol-totop rinsp-opacity-0');
    button.addEventListener('click', () => {
        document.body.querySelector('#toptool').scrollIntoView({ behavior: 'smooth', block: 'start' });
    });
    function checkScroll() {
        if (document.documentElement.scrollTop > 30) {
            button.classList.remove('rinsp-opacity-0');
        } else {
            button.classList.add('rinsp-opacity-0');
        }
    }
    document.addEventListener('scroll', () => {
        checkScroll();
    });
    setTimeout(checkScroll);
}

async function addSearchBar(queryParams, searchConfigAccess, userConfig) {
    const guide = document.querySelector('#guide');
    if (guide == null) {
        return;
    }
    const searchPref = await searchConfigAccess.read();
    guide.classList.add('rinsp-quicksearch-added');
    const quicksearchForm = newElem('form', 'rinsp-quicksearch', {
        method: 'post',
        action: '/search.php?'
    });
    function setFormTarget(forceNewWindow) {
        quicksearchForm.setAttribute('target', forceNewWindow || document.location.pathname !== '/search.php' ? '_blank' : '_self');
    }
    function addHiddenField(k, v) {
        return addElem(quicksearchForm, 'input', null, { type: 'hidden', name: k, value: v });
    }
    const keywordDataField = addHiddenField('keyword', '');
    addHiddenField('step', '2');
    addHiddenField('method', searchPref.defaultSearchAll ? 'AND' : 'OR');
    addHiddenField('sch_time', searchPref.defaultTimeRange||'all');
    addHiddenField('pwuser', '');
    addHiddenField('sch_area', '0');
    addHiddenField('f_fid', 'all');
    addHiddenField('orderway', 'postdate');
    addHiddenField('asc', 'DESC');

    if (userConfig.showSearchBar$align === 'menu-first') {
        guide.insertBefore(quicksearchForm, guide.firstChild);
    } else if (userConfig.showSearchBar$align === 'bar-center') {
        let gap = guide.getBoundingClientRect().width - 250;
        guide.classList.add('rinsp-quicksearch-align-center');
        guide.insertBefore(newElem('li', 'rinsp-spacer'), guide.firstChild);
        guide.insertBefore(quicksearchForm, guide.firstChild);
        guide.insertBefore(newElem('li', 'rinsp-spacer'), guide.firstChild);
        guide.insertBefore(newElem('li', null, { style: `flex: 0 1 ${gap.toFixed(0)}px` }), guide.firstChild);
    } else {
        guide.appendChild(quicksearchForm);
    }
    const searchField = addElem(quicksearchForm, 'input', 'rinsp-quicksearch-field', { required: '', value: queryParams.keyword||'' });
    const searchButton = addElem(quicksearchForm, 'a', 'rinsp-quicksearch-button');
    searchButton.addEventListener('click', evt => {
        if (beforeSubmit(evt.shiftKey)) {
            quicksearchForm.submit();
        } else {
            searchField.focus();
        }
    });
    quicksearchForm.addEventListener('submit', evt => {
        evt.stopPropagation();
        if (!beforeSubmit()) {
            evt.preventDefault();
            return false;
        }
    });
    function beforeSubmit(forceNewWindow) {
        keywordDataField.value = '';
        let keyword = searchField.value.trim();
        if (keyword.length === 0) {
            return false;
        }
        if (keyword.match(/^[\x00-\x7F]+$/)) {
            switch (keyword.replace(/\s/g, '').length) {
            case 0:
                return false;
            case 1:
                keyword = keyword + ' ' + keyword + ' ' + keyword;
                break;
            case 2:
                keyword = keyword + ' ' + keyword;
                break;
            }
        }
        keyword = keyword.replace(/\s+/g, ' ');
        keywordDataField.value = keyword;
        quicksearchForm.setAttribute('action', `/search.php?keyword-${encodeURIComponent(keyword).replace(/-/g, '%2D')}.html`);
        setFormTarget(forceNewWindow);
        return true;
    }
}

async function init() {
    // improve hover popup closing mechanism to avoid closing still active popups
    const scpt = document.createElement('script');
    scpt.textContent = `
    if (window['PwMenu']) {
        PwMenu.prototype.close = function() {
            read.t = setTimeout(() => {if (!read.menu.matches(':hover')) {closep()}}, 100);
        };
    }
    `;
    document.head.appendChild(scpt);

    // start main script
    const myUserId = findMyUserId();
    if (myUserId == null) {
        if (DEV_MODE) console.info('(southplus-watcher) cannot find user id');
        return;
    }

    let domainRedirectTarget = await getDomainRedirectTarget();
    if (domainRedirectTarget != null && domainRedirectTarget !== document.location.origin) {
        setDomainRedirectEnabled(true); // update target domain
        domainRedirectTarget = document.location.origin;
    }

    const mainConfigAccess = initConfigAccess(myUserId, MAIN_CONFIG_KEY, DEFAULT_MAIN_CONFIG);
    const searchConfigAccess = initConfigAccess(myUserId, SEARCH_CONFIG_KEY, DEFAULT_SEARCH_CONFIG);
    const contentIgnoreListConfigAccess = initConfigAccess(myUserId, CONTENT_IGNORE_LIST_CONFIG_KEY, DEFAULT_CONTENT_IGNORE_LIST_CONFIG);
    const threadFilterConfigAccess = initConfigAccess(myUserId, THREAD_FILTER_CONFIG_KEY, DEFAULT_THREAD_FILTER_CONFIG);
    const threadCategoryConfigAccess = initConfigAccess(myUserId, THREAD_CUSTOM_CATEGORY_CONFIG_KEY, DEFAULT_THREAD_CUSTOM_CATEGORY_CONFIG);
    const userFilterConfigAccess = initConfigAccess(myUserId, USER_FILTER_CONFIG_KEY, DEFAULT_USER_FILTER_CONFIG);
    const pinnedUsersConfigAccess = initConfigAccess(myUserId, PINNED_USERS_CONFIG_KEY, DEFAULT_PINNED_USERS_CONFIG);
    const sharetypeFilterConfigAccess = initConfigAccess(myUserId, SHARETYPE_FILTER_CONFIG_KEY, DEFAULT_SHARETYPE_FILTER_CONFIG);
    const favorThreadsCacheAccess = initConfigAccess(myUserId, FAVOR_THREADS_CACHE_CONFIG_KEY, DEFAULT_FAVOR_THREADS_CACHE_CONFIG);

    let userConfig = await mainConfigAccess.read();
    if (!userConfig.v15migrationApplied) {
        let migratedUserConfig = await mainConfigAccess.update(function(updatingUserConfig) {
            if (updatingUserConfig.v15migrationApplied) {
                return null;
            }
            const spuUserIds = new Set();
            Object.keys(updatingUserConfig.customUserHashIdMappings)
                .forEach(key => {
                    if (key[0] === '#') {
                        spuUserIds.add(key);
                    }
                });
            Object.entries(updatingUserConfig.customUserHashIdMappings)
                .forEach(entry => {
                    if (entry[0][0] === '@') {
                        spuUserIds.delete('#' + entry[1][0]);
                    }
                });
            for (let spuUserId of spuUserIds) {
                delete updatingUserConfig.customUserHashIdMappings[spuUserId];
            }
            updatingUserConfig.v15migrationApplied = true;
            return updatingUserConfig;
        });
        if (migratedUserConfig != null) {
            userConfig = migratedUserConfig;
        }
    }
    if (!userConfig.v42migrationApplied) {
        let migratedUserConfig = await mainConfigAccess.update(function(updatingUserConfig) {
            if (updatingUserConfig.v42migrationApplied) {
                return null;
            }
            const newMappings = {};
            Object.entries(updatingUserConfig.customUserHashIdMappings)
                .forEach(entry => {
                    if (entry[0][0] === '#') {
                        let userHashId = entry[1][1];
                        if (typeof userHashId === 'object' && typeof userHashId.hashId === 'string') {
                            userHashId = userHashId.hashId;
                        }
                        let newRecord = [entry[1][0] * 1, userHashId, entry[1][2]];
                        newMappings['#' + newRecord[0]] = newRecord;
                        newMappings['@' + newRecord[1]] = newRecord;
                    }
                });
            updatingUserConfig.customUserHashIdMappings = newMappings;
            updatingUserConfig.v42migrationApplied = true;
            return updatingUserConfig;
        });
        if (migratedUserConfig != null) {
            userConfig = migratedUserConfig;
        }
    }

    for (let i = userConfig.textSize * 1; i > 0; i--) {
        document.documentElement.classList.add('rinsp-textsize-step-' + i);
    }
    if (userConfig.siteThemeDarkerSubjectLine) {
        document.body.classList.add('rinsp-theme-darksubject');
    }
    if (userConfig.hideMobileVerSwitch) {
        document.body.classList.add('rinsp-hide-mobileswitch');
    }

    let showResourceSpots = null;
    Object.keys(DEFAULT_MAIN_CONFIG).filter(k => k.startsWith('showResourceSpots$')).forEach(k => {
        if (userConfig[k]) {
            showResourceSpots = showResourceSpots || {};
            showResourceSpots[k.substring(18)] = true;
        }
    });

    // 所在地是免空区
    const inResourceArea = document.querySelector('#breadcrumbs .crumbs-item[href="thread.php?fid-13.html"] + .crumbs-item') != null;

    // 所在地是事务受理
    const inOfficeArea = document.querySelector('#breadcrumbs .crumbs-item[href="thread.php?fid-2.html"] + .crumbs-item') != null;

    // 所在地是网赚区
    const inPaywallArea = document.querySelector('#breadcrumbs .crumbs-item[href="thread.php?fid-170.html"] + .crumbs-item') != null;
    const hasManagementRole = checkManagementRole(myUserId, userConfig, mainConfigAccess);

    initSiteMainMenu(hasManagementRole);
    const pendingWatchItemChecks = new Set();

    if (userConfig.floatingShortcut !== 'off') {
        const quickActionOverlay = addElem(document.body, 'div', 'rinsp-quick-action-overlay', { tabindex: "1" });
        quickActionOverlay.addEventListener('click', evt => {
            evt.stopPropagation();
        });
        document.body.addEventListener('click', () => {
            if (document.activeElement === quickActionOverlay) {
                quickActionOverlay.blur();
            }
        });
        addElem(quickActionOverlay, 'label', null, { version: VERSION_TEXT }).textContent = '⚙️';
    
        addElem(quickActionOverlay, 'div', 'rinsp-quick-action', { label: '🚫屏蔽关键词' })
            .addEventListener('mousedown', () => {
                openIgnoreThreadEditor(() => {
                    userPinArea.fireUpdateListeners();
                });
            });
    
        addElem(quickActionOverlay, 'div', 'rinsp-quick-action', { label: '💚喜欢关键词' })
            .addEventListener('mousedown', () => {
                openLikeThreadEditor(() => {
                    userPinArea.fireUpdateListeners();
                });
            });
    
        addElem(quickActionOverlay, 'div', 'rinsp-quick-action-divider');
    
        addElem(quickActionOverlay, 'div', 'rinsp-quick-action rinsp-dark-mode-toggle')
            .addEventListener('mousedown', () => {
                darkModeEnabledProperty.set(darkModeEnabledProperty.get() ? null : '1');
                applyDarkTheme();
            });
    
        addElem(quickActionOverlay, 'div', 'rinsp-quick-action rinsp-safe-mode-toggle')
            .addEventListener('mousedown', () => {
                setSafeMode(!isSafeMode());
            });
    
        addElem(quickActionOverlay, 'div', 'rinsp-quick-action-divider');
    
        addElem(quickActionOverlay, 'div', 'rinsp-quick-action', { label: '🔧设置页' })
            .addEventListener('mousedown', () => {
                document.location.href = '/u.php';
            });
        if (userConfig.floatingShortcut === 'tl') {
            quickActionOverlay.classList.add('rinsp-quick-action-overlay-pos-tl');
        } else {
            quickActionOverlay.classList.add('rinsp-quick-action-overlay-pos-tr');
        }
    }

    const notificationContainer = addElem(document.body, 'div', 'rinsp-notification-container');
    const extraControlContainer = addElem(document.body, 'div', 'rinsp-excontrol-container');
    const topControlContainer = addElem(document.body, 'div', 'rinsp-tpcontrol-container');
    const bottomControlContainer = addElem(document.body, 'div', 'rinsp-bmcontrol-container');
    const pmNotificationElem = addElem(notificationContainer, 'a', 'rinsp-notification-item rinsp-notification-item-type-pm');
    const watchNotificationElem = addElem(notificationContainer, 'div', 'rinsp-notification-item rinsp-notification-item-type-watch');
    if (userConfig.showBackToTopButton) {
        addBackToTopButton(bottomControlContainer);
    }

    function addIgnoreToggler(togglerClass, modeActiveClass, modeInactiveClass) {
        const lastStateKey = 'rinplus.mode.' + togglerClass + '.laststate';
        const togglerElem = addElem(extraControlContainer, 'a', 'rinsp-excontrol-item rinsp-excontrol-item-hidden');
        togglerElem.classList.add(togglerClass);
        togglerElem.addEventListener('click', function() {
            let newValue = !isActive();
            setActive(newValue);
            if (newValue) {
                localStorage.setItem(lastStateKey, '1');
            } else {
                localStorage.setItem(lastStateKey, '0');
            }
        });
        function restoreLastState(defaultActive) {
            const savedValue = localStorage.getItem(lastStateKey);
            if (defaultActive && savedValue !== '0') {
                setActive(true);
            } else {
                setActive(savedValue === '1');
            }
        }
        function isActive() {
            return document.body.classList.contains(modeActiveClass);
        }
        function setActive(active) {
            if (active) {
                document.body.classList.add(modeActiveClass);
                if (modeInactiveClass) {
                    document.body.classList.remove(modeInactiveClass);
                }
            } else {
                document.body.classList.remove(modeActiveClass);
                if (modeInactiveClass) {
                    document.body.classList.add(modeInactiveClass);
                }
            }
        }
        function setCount(count) {
            togglerElem.innerHTML = '';
            if (count !== 0) {
                togglerElem.classList.remove('rinsp-excontrol-item-hidden');
                addElem(togglerElem, 'span', `rinsp-excontrol-item-count rinsp-excontrol-item-count-${(''+count).length}d`).textContent = String(count);
            } else {
                togglerElem.classList.add('rinsp-excontrol-item-hidden');
            }
        }
        return {
            isActive,
            setActive,
            restoreLastState,
            setCount
        };
    }

    document.body.addEventListener('click', evt => {
        if (!evt.target) {
            return;
        }
        if (evt.target.classList.contains('r_two')) {
            const row = evt.target.closest('table');
            if (!row || !row.classList.contains('rinsp-filter-ignored')) {
                return;
            }
            document.body.classList.add('rinsp-filter-peek-mode');
            evt.target.scrollIntoView({
                behavior: 'auto',
                block: 'center'
            });
        }
        if (evt.target.tagName === 'TD' && (evt.target.getAttribute('id')||'').startsWith('td_')) {
            if (evt.target.parentNode.classList.contains('rinsp-thread-filter-dislike') || evt.target.parentNode.classList.contains('rinsp-thread-filter-masked-bysharetype')) {
                document.body.classList.add('rinsp-dislike-thread-peek-mode');
                evt.target.parentNode.scrollIntoView({
                    behavior: 'auto',
                    block: 'center'
                });
            } else if (evt.target.parentNode.classList.contains('rinsp-request-settlement-thread') && !evt.target.parentNode.classList.contains('rinsp-request-settlement-bypass')) {
                document.body.classList.add('rinsp-settlement-peek-mode');
                evt.target.scrollIntoView({
                    behavior: 'auto',
                    block: 'center'
                });
            }
        }
    });

    const userHashLookupStore = createUserHashLookupStore();
    const threadHistoryAccess = createThreadHistoryStore(myUserId, userConfig);
    const adminFunctions = (() => {
        if (!hasManagementRole) {
            return null;
        }
        const userPunishRecordStore = createRecordStore(`user_punish#${myUserId}`);
        return initAdminFunctions(myUserId, userConfig, userPunishRecordStore);
    })();

    const ignoreContentToggler = addIgnoreToggler('rinsp-content-ignore-toggler', 'rinsp-filter-peek-mode');
    const ignoreSettlementToggler = addIgnoreToggler('rinsp-settlement-ignore-toggler', 'rinsp-settlement-peek-mode');
    const ignorePaywellToggler = addIgnoreToggler('rinsp-paywall-ignore-toggler', 'rinsp-paywall-peek-mode');
    ignorePaywellToggler.restoreLastState(true);
    const ignoreDislikeThreadToggler = addIgnoreToggler('rinsp-dislike-thread-ignore-toggler', 'rinsp-dislike-thread-peek-mode');
    let hideAnsweredRequestToggler = null;
    let hideUnansweredRequestToggler = null;
    let hideExpiredRequestToggler = null;
    if (userConfig.showRequestThreadFilters) {
        if (isCurrentThreadMatchingFid(QUESTION_AND_REQUEST_AREA_ID)) {
            hideAnsweredRequestToggler = addIgnoreToggler('rinsp-answered-request-ignore-toggler', 'rinsp-answered-request-hide-mode');
            hideUnansweredRequestToggler = addIgnoreToggler('rinsp-unanswered-request-ignore-toggler', 'rinsp-unanswered-request-hide-mode');
            hideExpiredRequestToggler = addIgnoreToggler('rinsp-expired-request-ignore-toggler', 'rinsp-expired-request-hide-mode');
            hideAnsweredRequestToggler.restoreLastState();
            hideUnansweredRequestToggler.restoreLastState();
            hideExpiredRequestToggler.restoreLastState();
        }
    }
    let hideVisitedThreadToggler = null;
    if (threadHistoryAccess) {
        hideVisitedThreadToggler = addIgnoreToggler('rinsp-visited-thread-toggler', 'rinsp-visited-thread-view-mode', 'rinsp-visited-thread-mask-mode');
        hideVisitedThreadToggler.restoreLastState(true);

        const profilePopupMenu = document.querySelector('#menu_u > ul');
        if (profilePopupMenu) {
            const historyItem = addElem(profilePopupMenu, 'li');
            const historyLink = addElem(historyItem, 'a', null, { href: 'u.php?action-trade.html' });
            historyLink.textContent = '浏览记录';
        }
    }
    let scoringModeToggler = null;
    if (hasManagementRole && inResourceArea) {
        scoringModeToggler = addIgnoreToggler('rinsp-highlight-unscored-thread-toggler', 'rinsp-highlight-unscored-thread-mode');
        scoringModeToggler.setCount('+');
        scoringModeToggler.restoreLastState();
    }
    if (hasManagementRole && inOfficeArea) {
        document.addEventListener('mousedown', e => {
            if (e.target && e.target.tagName === 'A' && e.target.href) {
                if (!e.target.href.startsWith('javascript:')) {
                    document.querySelectorAll('.rinsp-last-clicked').forEach(el => el.classList.remove('rinsp-last-clicked'));
                    e.target.classList.add('rinsp-last-clicked');
                }
            }
        });
    }

    let hideClosedThreadToggler = addIgnoreToggler('rinsp-closed-thread-toggler', 'rinsp-closed-thread-view-mode', 'rinsp-closed-thread-mask-mode');
    hideClosedThreadToggler.restoreLastState(true);

    // auto-discover nick name the first time
    if (userConfig.myNickName == null) {
        const doc = await fetchGetPage(`${document.location.origin}/u.php?action-show-uid-${myUserId}.html`);
        let nickName = '';
        for (let td of doc.querySelectorAll('#u-profile .u-table > tbody > tr > td')) {
            if (td.textContent.trim() === '昵称') {
                nickName = td.parentNode.querySelector('th').textContent.trim();
                break;
            }
        }
        userConfig = await mainConfigAccess.update(function(updatingUserConfig) {
            updatingUserConfig.myNickName = nickName;
            return updatingUserConfig;
        });
    }

    if (userConfig.openIntroAfterUpdate) {
        tryOpenUpdateSummary();
    }

    if (userConfig.showFloatingMessageIndicator) {
        pmNotificationElem.setAttribute('href', 'message.php');
        addPrivateMessageNotifier();
    }
    if (userConfig.hideWatchButtonIfEmpty) {
        document.body.classList.add('rinsp-watchemnu-autohide');
    }

    if (userConfig.heightReductionMode) {
        const main = document.querySelector('#main');
        if (main) {
            main.classList.add('rinsp-compact-mode');
            if (userConfig.heightReductionMode$LV2) {
                main.classList.add('rinsp-compact-mode-smaller');
            }
        }
    }

    if (userConfig.hideFilteredThread) {
        document.body.classList.add('rinsp-threadfilter-hide-mode');
    }
    if (userConfig.requestThreadHighlightEnded) {
        document.body.classList.add('rinsp-request-highlight-ended-mode');
    }
    if (userConfig.requestThreadShowExtraBounty) {
        document.body.classList.add('rinsp-request-show-extra-bounty');
    }
    if (userConfig.requestThreadUseHistoryData) {
        document.body.classList.add('rinsp-request-show-bounty-history');
    }
    if (userConfig.hideSettlementPost) {
        if (userConfig.hideSettlementPost$GreyoutOnly) {
            document.body.classList.add('rinsp-request-settlement-greyout-mode');
        } else {
            document.body.classList.add('rinsp-request-settlement-hide-mode');
        }
    }
    if (userConfig.hideIgnoreContentPost) {
        document.body.classList.add('rinsp-filter-hide-mode');
    }
    if (userConfig.hideZeroReply) {
        document.body.classList.add('rinsp-byop-noreply-hide-mode');
    }
    
    const userPinArea = await addUserPinArea(pinnedUsersConfigAccess, userConfig.hideInactivePinnedUsers, userConfig.showResourceSpotsFloating);
    let shareTypeFilter = null;
    if (userConfig.showShareTypeFilter) {
        if (inResourceArea || inPaywallArea) {
            const headrow = document.querySelector('#ajaxtable .hthread');
            if (headrow) {
                shareTypeFilter = await addShareTypeFilterArea(headrow, sharetypeFilterConfigAccess);
            }
        } else {
            const headrow = document.querySelector('#main .t > table > tbody > tr > .h[colspan]');
            if (headrow && headrow.textContent.trim() === '主题列表') {
                shareTypeFilter = await addShareTypeFilterArea(headrow, sharetypeFilterConfigAccess);
            }
        }
    }

    const queryParams = {};
    const queryParamMatch = document.location.search.match(/^\?([A-Za-z_]+)-([^-]+)(?:-(.*))?\.html.*$/);
    if (queryParamMatch) {
        queryParams[queryParamMatch[1]] = decodeURIComponent(queryParamMatch[2]);
        if (queryParamMatch[3]) {
            const rest = queryParamMatch[3].split('-');
            while (rest.length > 0) {
                const key = rest.shift();
                const value = rest.shift();
                queryParams[key] = decodeURIComponent(value);
            }
        }
    } else if (document.location.search && !document.location.search.endsWith('.html')) {
        (new URLSearchParams(document.location.search)).forEach((value, key) => {
            queryParams[key] = value;
        });
    }
    if (userConfig.showSearchBar) {
        addSearchBar(queryParams, searchConfigAccess, userConfig);
    }

    let action = queryParams.action||'';
    if (document.location.pathname === '/' || document.location.pathname === '/index.php') {
        enhanceFrontPage(userConfig, mainConfigAccess);
    }

    if (document.location.pathname === '/u.php') {
        if (document.location.search === '' || queryParams.action == null && queryParams.uid * 1 === myUserId) {
            document.body.classList.add('rinsp-uphp-self');
            await initConfigPanel();
            rewriteUserRecentPostLinks(userConfig, userPinArea);
            await enhancePostListUserDisplay(queryParams.action||'', myUserId, myUserId, userConfig, userPinArea);
        } else if (['friend', 'favor', 'feed', ''].includes(action)) {
            const uid = queryParams.uid * 1 || myUserId;
            await enhancePostListUserDisplay(queryParams.action||'', uid, myUserId, userConfig, userPinArea);
            if (action === 'feed') {
                rewriteUserRecentPostLinks(userConfig, userPinArea);
            } else if (action === 'favor') {
                if (uid === myUserId) {
                    updateFavorRecords(favorThreadsCacheAccess);
                }
                enhanceFavorPageDisplay(uid, myUserId, userConfig, threadHistoryAccess);
            }
        } else if (action === 'trade') {
            if ((queryParams.uid * 1 || myUserId) === myUserId) {
                if (adminFunctions && queryParams.view === 'admin') {
                    adminFunctions.renderAdminHistoryPage(myUserId, userConfig);
                } else {
                    renderVisitHistoryPage(myUserId, userConfig, threadHistoryAccess);
                }
            } else if (adminFunctions) {
                adminFunctions.renderPunishHistoryPage(queryParams.uid * 1, userConfig);
            }
        }
        if (queryParams.uid) {
            enhanceUserProfilePages(queryParams.uid * 1, myUserId, userConfig, adminFunctions);
        } else {
            enhanceUserProfilePages(myUserId, myUserId, userConfig, adminFunctions);
        }

        if (action === 'topic') {
            enhanceTopicPostListDisplay(queryParams.uid * 1 || myUserId, myUserId, userConfig, threadHistoryAccess, threadCategoryConfigAccess, hideVisitedThreadToggler);
            setupInfiniteScroll_userTopics(userConfig, mainConfigAccess);
        } else if (action === 'post') {
            await enhanceReplyPostListDisplay(queryParams.uid * 1 || myUserId, myUserId, userConfig, userPinArea, mainConfigAccess, threadHistoryAccess, hideVisitedThreadToggler);
            setupInfiniteScroll_userReplies(userConfig, mainConfigAccess);
        }
        enhanceUserDetailPage(queryParams.action||'', queryParams.uid * 1, userConfig, userPinArea, adminFunctions);

        replaceTradeWithHistoryTab(myUserId, queryParams);
        if (adminFunctions) {
            replaceTradeWithPunishHistoryTab(myUserId, queryParams);
        }

        if (action) {
            document.body.classList.add('rinsp-uphp-' + action);
        }
    } else {
        if (userConfig.myUserHashId == null) {
            document.location.href = '/u.php';
        }
    }
    
    if (document.location.pathname === '/message.php') {
        if (action === '' || action === 'receivebox') {
            enhanceMessagingList(userConfig, userPinArea, true);
            setupInfiniteScroll_msgInbox(userConfig, mainConfigAccess);
        } else if (action === 'scout' || action === 'sendbox') {
            enhanceMessagingList(userConfig, userPinArea, false);
            setupInfiniteScroll_msgSent(userConfig, mainConfigAccess);
        } else if (action === 'chatlog') {
            setupInfiniteScroll_msgChatLog(userConfig, mainConfigAccess);
        } else if (action === 'write') {
            enhanceMessagingForm();
        } else if (action === 'read' || action === 'readsnd' || action === 'readscout') {
            enhanceMessageReadDisplay(userConfig);
            if (action === 'read' && threadHistoryAccess) {
                testRecordBestAnswerNotification(userConfig, threadHistoryAccess);
            }
        }
    }

    if (document.location.pathname === '/search.php') {
        enhanceSearchPage();
        setupInfiniteScroll_search(userConfig, mainConfigAccess);
        if (adminFunctions) {
            adminFunctions.addBatchDeleteFunction();
        }
    }

    if (document.location.pathname === '/read.php') {
        addSearchShortcut();
        setupInfiniteScroll_thread(userConfig, mainConfigAccess);
        addTitleLengthLimitIndicator();
    }

    if (document.location.pathname === '/plugin.php' || document.location.pathname === '/hack.php') {
        showAutoSpTaskStatus(myUserId, userConfig, mainConfigAccess);
    } else if (userConfig.autoSpTasks) {
        autoCompleteSpTasks(myUserId);
    }

    if (document.location.pathname === '/post.php') {
        addTitleLengthLimitIndicator();
    }

    if (document.location.pathname === '/thread.php' || document.location.pathname === '/thread_new.php') {
        enableForumAnnouncementFolding();
        addSearchShortcut();
        if (userConfig.showDefaultingPicWallOption) {
            addPicWallDefaultOption(queryParams.fid * 1);
        }
        if (document.location.pathname === '/thread.php') {
            setupInfiniteScroll_threadList(userConfig, mainConfigAccess);
        } else {
            setupInfiniteScroll_threadPicWall(userConfig, mainConfigAccess);
        }
    }

    if (userConfig.enhancePageTitle) {
        enhancePageTitle(queryParams, myUserId, userConfig, adminFunctions != null);
    }

    if (adminFunctions) {
        if (document.location.pathname === '/read.php') {
            if (inResourceArea) {
                adminFunctions.enableQuickPingFunction();
            }
            adminFunctions.addDeleteFormMetadata();
        } else if (document.location.pathname === '/masingle.php') {
            if (action === 'banuser') {
                adminFunctions.enhanceBanUserActionPage();
            } else {
                adminFunctions.enhanceReasonSelector();
                if (action === 'shield') {
                    adminFunctions.enhanceShieldActionPage();
                } else if (action === 'delatc') {
                    adminFunctions.enhanceDeleteReplyActionPage();
                }
            }
        } else if (document.location.pathname === '/mawhole.php') {
            adminFunctions.enhanceReasonSelector();
            adminFunctions.enhanceMawholeActionPage();
        } else if (document.location.pathname === '/operate.php') {
            adminFunctions.enhanceReasonSelector();
            if (action === 'showping') {
                if (queryParams.pid === 'tpc') {
                    adminFunctions.autoFillShowPingPage();
                }
                adminFunctions.enhancePingActionPage();
            }
        } else if (document.location.pathname === '/job.php') {
            if (action === 'endreward') {
                adminFunctions.enhanceEndRewardActionPage();
            }
        }
        if (document.location.pathname === '/thread.php') {
            const fid = queryParams.fid * 1;
            if (inResourceArea && fid > 0) {
                adminFunctions.addBatchPingFunction(fid);
            }
            adminFunctions.addBulkDeleteFormMetadata();
        }
    }

    if (userConfig.subCategoryInheritImageWallMode) {
        if (document.location.pathname === '/thread_new.php') {
            changeToImageWallThreadLinks();
        }
    }
    
    function tryRemoveCorruptDeployCookie() {
        try {
            const currentCookieString = document.cookie;
            if (!currentCookieString) {
                return;
            }
            
            const modified = currentCookieString
                .split(' ')
                .map(cookie => {
                    if (cookie.startsWith('deploy=') && cookie.length > 256) {
                        return 'deploy=\t\t\n';
                    }
                    return cookie;
                })
                .join(' ');
            
            if (currentCookieString !== modified) {
                if (DEV_MODE) {
                    console.info('dropped "deploy" cookie');
                }
                document.cookie = modified;
            }

        } catch (e) {
            if (DEV_MODE) {
                console.error(e);
            }
        }
    }
    tryRemoveCorruptDeployCookie();

    if (userConfig.siteNoticeSectionDefaultFolded) {
        const foldIcon = document.querySelector('.gongul #img_thread[src$="/cate_fold.gif"]');
        if (foldIcon) {
            if (DEV_MODE) {
                console.info('set notice default folded');
            }
            foldIcon.click();
        }
    }

    const watcherApi = initTopMenu();
    let ignoreList = await contentIgnoreListConfigAccess.read();
    let userFilter = await userFilterConfigAccess.read();
    initCurrentPost();

    function setupStickyUserInfoPopup() {
        let userInfoPopup = document.querySelector('body > .user-info');
        function bindUpdate() {
            const observer = new MutationObserver(() => {
                update();
            });
            observer.observe(userInfoPopup, { childList: false, subtree: false, attributes: true });
        }
        function update() {
            const top = Number.parseInt(userInfoPopup.style.top);
            if (document.documentElement.scrollTop > top && userInfoPopup.style.display !== 'none') {
                document.body.classList.add('rinsp-user-info-sticky-mode');
            } else {
                document.body.classList.remove('rinsp-user-info-sticky-mode');
            }
        }

        if (userInfoPopup == null) {
            const observer = new MutationObserver(() => {
                userInfoPopup = document.querySelector('body > .user-info');
                if (userInfoPopup != null) {
                    observer.disconnect();
                    observer.takeRecords();
                    bindUpdate();
                    update();
                }
            });
            observer.observe(document.body, { childList: true, subtree: true, attributes: false });
        } else {
            bindUpdate();
        }
    }
    if (userConfig.stickyUserInfo) {
        setupStickyUserInfoPopup();
    }

    document.addEventListener('mousedown', function(e) {
        if (!e.target) {
            return;
        }
        const enclosingPost = e.target.closest('.rinsp-post');
        if (enclosingPost) {
            enclosingPost.focus();
        }
        if (document.querySelector('.rinsp-dialog-modal-mask') != null) {
            return;
        }
        const withinMenu = e.target.closest('.rinsp-common-popup-menu');
        let withinMenuId = '-n/a-';
        if (withinMenu != null) {
            withinMenuId = withinMenu.getAttribute('id');
        }
        document.querySelectorAll('.rinsp-common-popup-menu').forEach(function(menu) {
            if (menu.getAttribute('id') !== withinMenuId) {
                menu.remove();
            }
        });
    });

    if (userConfig.autoCheckReplyOption) {
        autoCheckReply();
    }

    const stickyUserpicController = (function() {
        let posts = [];
        let schedule = null;
        return {
            setPosts(newPosts) {
                posts = newPosts;
            },
            update() {
                let activeFound = false;
                for (let post of posts) {
                    post.userPicElem.classList.remove('rinsp-userpic-sticky');
                    post.userPicElem.style.top = '';
                    if (!activeFound) {
                        const postRect = post.rootElem.getBoundingClientRect();
                        const visibleHeight = postRect.top + postRect.height;
                        if (postRect.top < 0 && visibleHeight > 0) {
                            const userPicElemStaticRect = post.userPicElem.getBoundingClientRect();
                            if (postRect.height - userPicElemStaticRect.height >= MIN_OVER_HEIGHT_STICKY_MODE_TRIGGER) {
                                const dummy = post.userPicElem.parentNode.querySelector('.rinsp-user-pic-dummy');
                                dummy.style.height = userPicElemStaticRect.height.toFixed(0) + 'px';
                                if (userConfig.stickyUserInfo) {
                                    post.userPicElem.classList.add('rinsp-userpic-sticky');
                                }
                                const userPicElemStickyRect = post.userPicElem.getBoundingClientRect();
                                const overflowHeight = userPicElemStickyRect.height - visibleHeight;
                                if (overflowHeight > 0) {
                                    post.userPicElem.style.top = '-' + overflowHeight.toFixed(0) + 'px';
                                } else {
                                    post.userPicElem.style.top = '';
                                }
                            }
                            activeFound = true;
                            continue;
                        }
                    }
                }
            },
            scheduleUpdate() {
                if (schedule != null)
                    return;
                schedule = setTimeout(() => {
                    this.update();
                    schedule = null;
                }, 0);
            }
        };
    })();

    let forceCleanUpdate = false;
    let enhancementMemoryObj = {};
    let postEnhancementActions = [];
    let lastCacheKey = null;

    const observer = createMutationObserver(applyEnhancement);
    await applyEnhancement(true);

    async function applyEnhancement(firstTime) {
        if (DEBUG_MODE) console.info('applyEnhancement');

        function disableSystemPostCollapse(post) {
            if (userConfig.disablePostCollapse) {
                return true;
            }
            if (userConfig.customUserpicBypassIgnoreList && !post.defaultUserPic) {
                return true;
            }
            if (post.postUid === myUserId && userConfig.disablePostCollapse$Self) {
                return true;
            }
            if (userConfig.customUserBypassIgnoreList && userConfig.customUserHashIdMappings['#' + post.postUid]) {
                return true;
            }
            return false;
        }
        function handleCollapseStateClasses(post) {
            post.rootElem.classList.remove('rinsp-sys-collapse-override');
            post.rootElem.classList.remove('rinsp-post-sys-collapsed');
            if (!post.contentDefaultIgnorable) {
                return;
            }
            if (disableSystemPostCollapse(post)) {
                post.rootElem.classList.add('rinsp-sys-collapse-override');
                const expandButton = post.rootElem.querySelector('.js-puremark-content ~ .bianji > a[onclick="return expandpuremark(this)"]');
                if (expandButton) {
                    expandButton.click();
                }
            } else {
                post.rootElem.classList.add('rinsp-post-sys-collapsed');
            }
        }

        let threadFilter = await threadFilterConfigAccess.read();

        function onConfigUpdate(updatedConfigs) {
            forceCleanUpdate = true;
            if (updatedConfigs) {
                if (updatedConfigs.threadFilter != null) {
                    threadFilter = updatedConfigs.threadFilter;
                }
                if (updatedConfigs.userFilter != null) {
                    userFilter = updatedConfigs.userFilter;
                }
                if (updatedConfigs.ignoreList != null) {
                    ignoreList = updatedConfigs.ignoreList;
                }
            }
        }
        function contentFilter_onConfigUpdate(updatedConfigs) {
            onConfigUpdate(updatedConfigs);
            observer.trigger();
        }

        function threadTitleFilter_onConfigUpdate(updatedConfigs) {
            onConfigUpdate(updatedConfigs);
            observer.trigger();
        }

        if (userConfig.stickyUserInfo) {
            document.addEventListener('scroll', () => {
                stickyUserpicController.scheduleUpdate();
            });
        }
        async function enhanceThreadView(tid, posts, userMap, userFilter, userPinArea, adminFunctions) {
            let gfPost = null;

            if (userConfig.stickyUserInfo) {
                stickyUserpicController.setPosts(posts);
            }

            userPinArea.clearState();
            const postCountsByUid = new Map();
            if (userConfig.showActiveRepliers) {
                const minReplyCount = userConfig.showActiveRepliers$min * 1 || 2;
                posts.forEach(post => {
                    if (post.floor > 0) {
                        const count = postCountsByUid.get(post.postUid);
                        postCountsByUid.set(post.postUid, (count||0) + 1);
                    }
                });
                postCountsByUid.forEach((count, uid) => {
                    if (count >= minReplyCount) {
                        if (!isUserBlacklisted(userFilter, uid)) {
                            const userInfo = userMap.get(uid);
                            userPinArea.watchUserLocally(uid, userInfo.nickName, userInfo.img);
                        }
                    }
                });
            }
            posts.forEach(post => {
                post.rootElem.classList.add('rinsp-post');
                if (post.floor === 0) {
                    gfPost = post;
                    post.rootElem.classList.add('rinsp-post-gf');
                }
                handleCollapseStateClasses(post);
            });

            if (inResourceArea && gfPost) {
                // add scored marker
                const markElem = gfPost.rootElem.querySelector('#mark_tpc');
                if (markElem && markElem.textContent.indexOf('评分记录') !== -1) {
                    gfPost.rootElem.classList.add('rinsp-post-scored');
                } else {
                    gfPost.rootElem.classList.add('rinsp-post-unscored');
                }
            }
            if (gfPost && gfPost.areaId === QUESTION_AND_REQUEST_AREA_ID) {
                const tac = document.querySelector('.tpc_content > .tips:not(.rinsp-uid-inspected) > .tac:last-child');
                if (tac) {
                    const winnerMatch = tac.textContent.match(/最佳答案获得者: ([a-z0-9]{8})/);
                    if (winnerMatch) {
                        const tips = tac.parentNode;
                        tips.classList.add('rinsp-uid-inspected');
                        const uhash = winnerMatch[1];
                        let replace = null;
                        if (userConfig.highlightMyself && uhash === userConfig.myUserHashId) {
                            (replace = newElem('span', 'rinsp-nickname-byme')).textContent = MY_NAME_DISPLAY;
                        } else {
                            // userPinArea

                            const fullName = userConfig.customUserHashIdMappings['@' + uhash];
                            if (fullName != null) {
                                (replace = newElem('span', 'rinsp-nickname-byother')).textContent = fullName[2];
                            }
                        }
                        if (replace) {
                            tac.hidden = true;
                            const tac2 = addElem(tips, 'div', 'tac');
                            tac2.appendChild(document.createTextNode('最佳答案获得者: '));
                            tac2.appendChild(replace);
                        }
                    }
                }
            }
            
            const punishRecordAccess = adminFunctions ? adminFunctions.createPunishRecordAccess() : null;
            annotateUsers(posts, myUserId, userConfig, userMap, {
                punishRecordAccess,
                changeCustomUserMapping
            });

            if (userConfig.enhanceSellFrame || userConfig.buyRefreshFree) {
                enhanceBuyButtons(userConfig, () => {
                    forceCleanUpdate = true;
                    observer.trigger();
                });
            }
            if (userConfig.replyRefreshFree) {
                const lastPost = posts.at(-1);
                applyReplyRefreshFree(tid, lastPost, myUserId);
            }
    
            applySubjectLineEnhancement(posts, userConfig);

            applyHookDirectives(posts, myUserId, userConfig);

            await applyContentFilter(posts, myUserId, userConfig, threadFilter, userFilter, ignoreList, contentFilter_onConfigUpdate);

            if (showResourceSpots) {
                let likePatternMatcher = null;
                if (threadFilter && threadFilter.likes.length > 0) {
                    likePatternMatcher = createKeywordMatcherFactory(threadFilter.likes);
                }

                posts.forEach(post => {
                    if (!post.rootElem.classList.contains('rinsp-filter-ignored')) {
                        userPinArea.offerPostContent(post, showResourceSpots, likePatternMatcher);
                    }
                });
            }
            await userPinArea.update();

            posts.forEach(post => {
                post.rootElem.parentNode.classList.remove('rinsp-thread-filter-pinned');
                if (userPinArea.isWatching(post.postUid)) {
                    if (userPinArea.isPinned(post.postUid)) {
                        post.rootElem.parentNode.classList.add('rinsp-thread-filter-pinned');
                    } else {
                        post.rootElem.parentNode.classList.add('rinsp-thread-filter-active-reply');
                    }
                    
                    const bannedText = post.rootElem.querySelector('.tpc_content div[id^="read_"] > span[style="color:black;background-color:#ffff66"]');
                    const banned = bannedText && bannedText.textContent.trim() === '用户被禁言,该主题自动屏蔽!';
                    const contentIgnored =  post.rootElem.classList.contains('rinsp-filter-ignored');
                    const locClass = post.floor === 0 ? 'gf' : banned ? 'banned' : contentIgnored ? 'ignoredmark' : null;
                    userPinArea.addLocation(post.postUid, post.floor === 0 ? 'GF' : `${post.floor}F`, locClass, () => {
                        if (contentIgnored && !ignoreContentToggler.isActive()) {
                            ignoreContentToggler.setActive(true);
                        }
                        post.rootElem.scrollIntoView({
                            behavior: 'smooth',
                            block: 'center'
                        });
                    });
                }
            });
            await userPinArea.update();

            if (adminFunctions) {
                adminFunctions.addPostAdminControl(posts, queryParams.page||1, userMap);
            }
        }

        if (document.location.pathname === '/read.php') {
            if (adminFunctions) {
                adminFunctions.addBatchPostSelectionControl();
            }
            const posts = getPosts(document);
            const cacheKey = 'posts:' + posts.length;
            if (forceCleanUpdate || cacheKey !== lastCacheKey) {
                lastCacheKey = cacheKey;
                if (posts.length > 0) {
                    const userMap = readEnhanceUserInfoMap(myUserId, userConfig);
                    const myself = userMap.get(myUserId);
                    if (myself) { // auto update my nick name if changed
                        if (userConfig.myNickName == null || myself.nickName !== userConfig.myNickName) {
                            userConfig = await mainConfigAccess.update(function(updatingUserConfig) {
                                updatingUserConfig.myNickName = myself.nickName;
                                return updatingUserConfig;
                            });
                        }
                    }

                    await enhanceThreadView(queryParams.tid * 1, posts, userMap, userFilter, userPinArea, adminFunctions);
                    if (threadHistoryAccess) {
                        recordThreadAccess(document, posts, myUserId, threadHistoryAccess, enhancementMemoryObj) // no need to wait
                            .catch(ex => console.error('recordThreadAccess FAILED', ex));
                    }
                    if (firstTime) {
                        const targetFloor = (document.location.hash.match(/^#fl-(\d+)$/)||[])[1] * 1;
                        if (targetFloor > 0) {
                            const targetElem = posts.filter(post => post.floor === targetFloor).map(post => post.rootElem)[0];
                            if (targetElem) {
                                postEnhancementActions.push(() => {
                                    targetElem.scrollIntoView({
                                        behavior: 'smooth',
                                        block: 'start'
                                    });
                                });
                            }
                        }
                    }
                } else {
                    const pageError = findErrorMessage(document);
                    if (pageError && pageError.indexOf('数据已被删除') !== -1 && (queryParams.tid * 1) > 0) {
                        recordThreadDeleted(queryParams.tid * 1, myUserId, threadHistoryAccess); // no need to wait
                    }
                }
            } else {
                if (DEBUG_MODE) console.info('skipped update', cacheKey);
            }
        } else {
            const threadList = getThreadList();
            if (threadList == null) {
                lastCacheKey = null;
            } else {
                const cacheKey = 'threads:' + threadList.length;
                if (forceCleanUpdate || cacheKey !== lastCacheKey) {
                    lastCacheKey = cacheKey;

                    for (let thread of threadList) {
                        if (thread.opHashId) {
                            await userHashLookupStore.set(Number.parseInt(thread.opHashId, 16), thread.op).catch(ex => null);
                        }
                    }
                    await showUserNames(userConfig, userHashLookupStore, myUserId);
                    if (userConfig.threadListDefaultOpenNewPage) {
                        threadList.forEach(thread => {
                            thread.row.querySelectorAll('a[href]:not([target])').forEach(el => el.setAttribute('target', '_blank'));
                        });
                    }

                    // unused: reserved for later use
                    /*
                    const pagesElem = document.querySelector('#main .t3 .pages');
                    if (pagesElem && document.querySelector('#main .rinsp-threadlist-options') == null) {
                        const t3 = pagesElem.closest('.t3');
                        const c = Array.from(t3.children).find(el => el.classList.contains('c'));
                        const optionsElem = newElem('span', 'fr rinsp-threadlist-options');
                        t3.insertBefore(optionsElem, c);
                    }
                    */

                    let settlementPostMatcher = null;
                    if (userConfig.hideSettlementPost) {
                        if (document.location.href.match(/\/thread(_new)?\.php\?fid-48[-.]/)) {
                            const settlementDetectionRules = threadFilter.settlementKeywords||[];
                            if (settlementDetectionRules.length > 0) {
                                settlementPostMatcher = createKeywordMatcherFactory(settlementDetectionRules);
                            }
                        }
                    }
                    await applyThreadListEnhancement(myUserId, userConfig, threadList, threadFilter, userFilter, userPinArea, shareTypeFilter, settlementPostMatcher, threadHistoryAccess, threadTitleFilter_onConfigUpdate, forceCleanUpdate);
                } else {
                    if (DEBUG_MODE) console.info('skipped update', cacheKey);
                }
            }
        }
    
        forceCleanUpdate = false;

        const actions = postEnhancementActions;
        postEnhancementActions = [];
        actions.forEach(action => {
            action();
        });

    }

    function getThreadList() {
        switch (document.location.pathname) {
            case '/thread_new.php':
                return readPicWallThreadList();
            case '/thread.php':
                return readThreadList();
            case '/search.php':
                if (document.querySelector('form[action="search.php?"]')) {
                    return null;
                } else {
                    return readSearchThreadList();
                }
            default:
                return null;
        }
    }

    userPinArea.addUpdateListener(() => {
        forceCleanUpdate = true;
        observer.trigger();
    });
    if (shareTypeFilter) {
        shareTypeFilter.addUpdateListener(() => {
            forceCleanUpdate = true;
            observer.trigger();
        });
    }

    observer.init(document.getElementById('main'), { childList: true, subtree: true, attributes: false });

    let hiddenStateChangeObserver = null;
    const postListForm = document.querySelector('#main > form[name="delatc"]');
    if (postListForm) {
        let lastHiddenPostCount = 0;
        hiddenStateChangeObserver = createMutationObserver(async () => {
            const hiddenPostCount = document.querySelectorAll('.t5.t2[hidden]').length;
            if (hiddenPostCount != lastHiddenPostCount) {
                forceCleanUpdate = true;
                lastHiddenPostCount = hiddenPostCount;
                postEnhancementActions.push(() => {
                    document.querySelectorAll('#main > form[name="delatc"] > .t5.t2[hidden] > .js-post').forEach(defaultHiddenPost => {
                        if (defaultHiddenPost.classList.contains('rinsp-filter-bypass')) {
                            return;
                        }
                        const postContainer = defaultHiddenPost.parentElement;
                        if (postContainer.hidden) {
                            defaultHiddenPost.classList.remove('rinsp-filter-bypass');
                        } else {
                            defaultHiddenPost.classList.add('rinsp-filter-bypass');
                        }
                    });
                });
                observer.trigger();
            }
        });
        hiddenStateChangeObserver.init(postListForm, { childList: true, subtree: true, attributes: true });

    }

    if (threadHistoryAccess) {
        // if history based styling is active, need to recheck on page activation
        document.addEventListener('visibilitychange', () => {
            if (!document.hidden) {
                forceCleanUpdate = true;
                observer.trigger();
            }
        });
    }

    if (DEV_MODE) console.info('(southplus-watcher) loaded');

    function getUserProfileHashId() {
        const elem = document.querySelector('#main #u-wrap #u-content #u-top h1');
        if (elem != null) {
            const userHashId = elem.textContent.trim();
            if (userHashId.match(/[0-9a-f]{8}/)) {
                return userHashId;
            }
        }
        return null;
    }

    async function enhanceUserProfilePages(userId, myUserId, userConfig, adminFunctions) {
        if (userId !== myUserId) {
            // add more action shortcuts
            const actionShortcutGrid = document.querySelector('#u-portrait ~ .bdbA > table > tbody');
            if (actionShortcutGrid) {
                const actionRow = addElem(actionShortcutGrid, 'tr');
                const cell = addElem(actionRow, 'td', null, { align: 'center' });
                addElem(cell, 'span', null, { style: 'display: inline-block; width: 14px' });
                cell.appendChild(document.createTextNode(' '));
                addElem(cell, 'a', null, { href: `message.php?action-chatlog-withuid-${userId}.html` }).textContent = '通信记录';
                addElem(actionRow, 'td');
                
                if (adminFunctions) {
                    const punishRecordAccess = adminFunctions.createPunishRecordAccess();
                    const record = await punishRecordAccess.getPunishRecord(userId);
                    if (record && record.logs.length > 0) {
                        const punishRecordRow = addElem(actionShortcutGrid, 'tr');
                        const punishRecordCell = addElem(punishRecordRow, 'td', 'rinsp-profile-punish-tag-cell', { colspan: '2' });
                        punishRecordCell.appendChild(renderPunishTags(userId, record));
                    }
                }
            }
        } else {
            document.querySelectorAll('#u-portrait img.pic').forEach(el => el.classList.add('rinsp-myavatar'));
        }
    }

    async function enhanceUserDetailPage(action, userId, userConfig, userPinArea, adminFunctions) {
        // add user memory action buttons
        let nickName = null;
        let uidCell = null;
        if (action === 'show') {
            for (let td of document.querySelectorAll('#u-profile table > tbody > tr > td')) {
                if (td.textContent === 'UID') {
                    uidCell = td.nextElementSibling;
                    if (!userId) {
                        userId = uidCell.textContent * 1;
                    }
                }
                if (td.textContent === '昵称') {
                    nickName = td.nextElementSibling.textContent.trim();
                    break;
                }
            }
        }
        const userHashId = getUserProfileHashId();
        if (userHashId != null) {
            if (uidCell) {
                const pinOper = addElem(uidCell, 'div', 'rinsp-user-action-pinuser-button');
                const update = () => {
                    pinOper.innerHTML = '';
                    delete pinOper.dataset.pinned;
                    if (userPinArea.isPinned(userId)) {
                        pinOper.dataset.pinned = '1';
                        addElem(pinOper, 'a', 'rinsp-user-action-pinuser-icon', { href: 'javascript:'})
                            .addEventListener('click', () => {
                                userPinArea.unwatchUser(userId);
                            });
                    } else {
                        addElem(pinOper, 'a', 'rinsp-user-action-pinuser-icon', { href: 'javascript:'})
                            .addEventListener('click', () => {
                                const img = document.querySelector('#u-portrait img.pic');
                                userPinArea.watchUser(userId, nickName, getImgSrc(img));
                            });
                    }
                };
                userPinArea.addUpdateListener(() => update());
                update();

                if (adminFunctions) {
                    uidCell.appendChild(document.createTextNode(' / '));
                    const showUserNameButton = addElem(uidCell, 'a', null, { href: 'javascript:'});
                    showUserNameButton.textContent = '显示用户名';
                    showUserNameButton.addEventListener('click', async () => {
                        const userNameDisplay = addElem(uidCell, 'span');
                        userNameDisplay.textContent = '⌛';
                        showUserNameButton.remove();
                        adminFunctions.findUserName(userId)
                            .then(userName => {
                                userNameDisplay.classList.add('rinsp-user-name-reveal');
                                userNameDisplay.textContent = userName;
                            })
                            .catch(ex => userNameDisplay.textContent = '⚠️' + ex);

                    });
                }
            }
    
            let customNameEntry = userConfig.customUserHashIdMappings['@' + userHashId];
            const parentElem = document.querySelector('#main #u-wrap #u-content #u-top h1');
            const button = addElem(parentElem, 'span', 'rinsp-user-map-icon');
            if (customNameEntry) {
                button.setAttribute('rinsp-nickname', customNameEntry[2]);
                button.classList.add('rinsp-user-mapped');
            }
            button.addEventListener('click', async () => {
                button.classList.add('rinsp-config-saving');
                changeCustomUserMapping(userId, userHashId, customNameEntry&&customNameEntry[2]||nickName||'')
                    .then(function(newName) {
                        if (newName == null) {
                            return;
                        }
                        if (newName) {
                            button.classList.add('rinsp-user-mapped');
                            button.setAttribute('rinsp-nickname', newName);
                        } else {
                            button.classList.remove('rinsp-user-mapped');
                            button.removeAttribute('rinsp-nickname');
                        }
                    })
                    .finally(() => {
                        button.classList.remove('rinsp-config-saving');
                    });
            });
        }
    
    }

    async function changeCustomUserMapping(userId, userHashId, nickName, bypassPrompt) {
        let chosenNickName = bypassPrompt ? nickName : prompt('记住昵称 - 输入空白以撤销记录', nickName);
        if (chosenNickName == null) { // cancelled
            return null;
        }
        chosenNickName = chosenNickName.trim();

        await mainConfigAccess.update(function(updatingUserConfig) {
            const currentMetadata = updatingUserConfig.customUserHashIdMappings['@' + userHashId];
            if (!chosenNickName && !currentMetadata) {
                return null;
            }
            if (chosenNickName) {
                const newMetadata = [userId * 1, userHashId, chosenNickName];
                updatingUserConfig.customUserHashIdMappings['#' + userId] = newMetadata;
                updatingUserConfig.customUserHashIdMappings['@' + userHashId] = newMetadata;
                return updatingUserConfig;
            } else {
                delete updatingUserConfig.customUserHashIdMappings['#' + currentMetadata[0]];
                delete updatingUserConfig.customUserHashIdMappings['@' + userHashId];
                return updatingUserConfig;
            }
        });
        return chosenNickName;
    }

    async function openClearHistoryMenu(anchor, onUpdate) {
        const items = [];

        function generateAction(task, confirmMessage, successMessage) {
            return async () => {
                if (!confirm('请确认 ' + confirmMessage))
                    return;
                const action = async () => {
                    await task();
                    return '✔️' + successMessage;
                };
                runWithProgressPopup(action, '清空中...', anchor, 1500)
                    .catch((ex) => alert(String(ex)))
                    .finally(() => onUpdate());
            };
        }

        items.push({
            label: '清空最近浏览列表',
            action: generateAction(async () => {
                await threadHistoryAccess.recentAccessStore.clear();
            }, '清空最近浏览列表', '已清空最近浏览列表')
        });

        items.push({
            label: '清空全部记录',
            action: generateAction(async () => {
                await threadHistoryAccess.historyStore.clear();
                await threadHistoryAccess.recentAccessStore.clear();
            }, '清空全部记录', '已清空全部记录')
        });

        setupPopupMenu({
            title: '清空帖子浏览记录',
            popupMenuId: 'CLEAR_HISTORY_MENU',
            width: 150,
            anchor,
            items
        });
    }

    async function initConfigPanel() {
        if (userConfig.myUserHashId == null) {
            const userHashId = getUserProfileHashId();
            if (userHashId != null) {
                userConfig = await mainConfigAccess.update(function(freshUserConfig) {
                    freshUserConfig.myUserHashId = userHashId;
                    return freshUserConfig;
                });
            }
        }

        const mainPanel = document.querySelector('#u-contentmain');
        const configPanel = addElem(mainPanel, 'div', 'rinsp-config-panel');
        const configHeading = addElem(configPanel, 'h5', 'u-h5');
        addElem(configHeading, 'span').textContent = `插件设置 (凛+ v${VERSION_TEXT})`;
        if (DEV_MODE) {
            const devToggler = addElem(configHeading, 'a', 'rinsp-dev-toggle r gray3');
            devToggler.textContent = '开发者模式';
            devToggler.addEventListener('click', () => {
                if (confirm('关闭开发者模式?')) {
                    localStorage.removeItem('RIN_PLUS_DEV_MODE');
                    window.location.reload();
                }
            });
        } else {
            const devToggler = addElem(configHeading, 'input', 'rinsp-dev-toggle r gray3', { size: 4 });
            devToggler.addEventListener('change', () => {
                if (devToggler.value === 'devmode') {
                    devToggler.value = '';
                    alert('开发者模式已打开');
                    localStorage.setItem('RIN_PLUS_DEV_MODE', '1');
                    window.location.reload();
                }
            });
        }
        const configList = addElem(configPanel, 'dl', 'rinsp-config-list');
        const introItem = addElem(configList, 'dt', 'rinsp-config-item-lv1 rinsp-config-item-checked');
        addElem(introItem, 'a', 'rinsp-intro-link', { href: INTRO_POST, target: '_blank' }).textContent = '🔗 捷径: 功能简介帖(茶館)';
        putCheckBoxItem('新功能更新后 自动打开简介帖', 'openIntroAfterUpdate', 2);

        addElem(configList, 'dt', 'rinsp-config-item-sep rinsp-config-item-lv1').textContent = '绑域名设置 (只同域名内适用)';
        putCheckBoxItem('暗黑模式', {
            get: () => darkModeEnabledProperty.get() != null,
            async update(enabled) {
                darkModeEnabledProperty.set(enabled ? '1' : null);
                applyDarkTheme();
            }
        });

        const darkModeOption = addElem(configList, 'dt', 'rinsp-config-item-lv2');
        addElem(darkModeOption, 'span').textContent = '🎨 样式 ';
        const darkModeSelect = addElem(darkModeOption, 'select');
        addElem(darkModeSelect, 'option', null, { value: 'darkbeige' }).textContent = '暗棕色';
        addElem(darkModeSelect, 'option', null, { value: 'hcdarkbeige' }).textContent = '暗棕色 (高反差)';
        addElem(darkModeSelect, 'option', null, { value: 'darkblue' }).textContent = '暗蓝色';
        addElem(darkModeSelect, 'option', null, { value: 'hcdarkblue' }).textContent = '暗蓝色 (高反差)';
        addElem(darkModeSelect, 'option', null, { value: 'darkred' }).textContent = '暗红色';
        addElem(darkModeSelect, 'option', null, { value: 'darkgrey' }).textContent = '暗灰色';
        darkModeSelect.value = darkModeThemeProperty.get();
        darkModeSelect.addEventListener('change', () => {
            darkModeThemeProperty.set(darkModeSelect.value);
            applyDarkTheme();
        });

        putCheckBoxItem('跟随系统偏好设置', {
            get: () => darkModeFollowsSystemProperty.get() === '1',
            async update(enabled) {
                darkModeFollowsSystemProperty.set(enabled ? '1' : null);
                applyDarkTheme();
            }
        }, 2);
        putCheckBoxItem('贤者模式 隐藏NSFW图片', {
            get: () => isSafeMode(),
            update: async (enabled) => setSafeMode(enabled)
        });
        putCheckBoxItem('贤者模式中不隐藏自己头像', {
            get: () => isSafeModeShowMyAvater(),
            update: async (enabled) => setSafeModeShowMyAvater(enabled)
        }, 2);
        putCheckBoxItem('提高页面加载速度 (日后论坛更新取代功能后移除)', {
            get: () => isFastLoadMode(),
            update: async (enabled) => setFastLoadMode(enabled)
        });
        putCheckBoxItem('延迟加载屏幕外的图片', {
            get: () => isFastLoadLazyImageMode(),
            update: async (enabled) => setFastLoadLazyImageMode(enabled)
        }, 2);

        addElem(configList, 'dt', 'rinsp-config-item-sep rinsp-config-item-lv1').textContent = '论坛设置';
        putCheckBoxItem('自动签到 (社区论坛任务)', 'autoSpTasks');
        if (userConfig.autoSpTasks) {
            const taskRecordAccess = initConfigAccess(myUserId, 'autosptask', {});
            const autoSpTasksSummary = addElem(configList, 'dt', 'rinsp-config-item-tip rinsp-config-item-lv2');
            taskRecordAccess.read().then(taskRecord => {
                if (taskRecord.lastCompleteSp > 0) {
                    const hr = (Date.now() - taskRecord.lastComplete) / 3600000;
                    autoSpTasksSummary.textContent = `💡 ${getAgeString(hr * 60)} 已领取 ${taskRecord.lastCompleteSp||'?'} SP | 记录中总共领取 ${taskRecord.totalSp||'?'} SP`;
                }
            });
        }
        putCheckBoxItem('自动域名跳转', {
            get: () => domainRedirectTarget != null,
            update: async (enabled) => setDomainRedirectEnabled(enabled)
        });
        putCheckBoxItem('增強网页标题 (令其更有意义)', 'enhancePageTitle');
        const searchBarItem = putCheckBoxItem('主菜单上添加快速搜索栏', 'showSearchBar');
        addElem(configList, 'dt', 'rinsp-config-item-tip rinsp-config-item-lv2').textContent = '💡 为了腾出空间 "社区论坛任务" 缩写为 "任务"';
        addElem(configList, 'dt', 'rinsp-config-item-tip rinsp-config-item-lv2').textContent = '💡 快速搜索使用已保存的默认设置 (发表主题时间、部分或完全匹配)';

        addElem(searchBarItem, 'span').textContent = '🔹位置 ';
        const searchBarAlignSelect = addElem(searchBarItem, 'select');
        addElem(searchBarAlignSelect, 'option', null, { value: 'default' }).textContent = '主菜单 最右边';
        addElem(searchBarAlignSelect, 'option', null, { value: 'menu-first' }).textContent = '主菜单 右边第一项';
        addElem(searchBarAlignSelect, 'option', null, { value: 'bar-center' }).textContent = '主菜单 空间置中';
        searchBarAlignSelect.value = userConfig.showSearchBar$align;
        searchBarAlignSelect.value = searchBarAlignSelect.value || 'default';
        searchBarAlignSelect.addEventListener('change', () => {
            configPanel.classList.add('rinsp-config-saving');
            mainConfigAccess.update(function(updatingUserConfig) {
                updatingUserConfig.showSearchBar$align = searchBarAlignSelect.value;
                return updatingUserConfig;
            })
            .finally(function() {
                configPanel.classList.remove('rinsp-config-saving');
                window.location.reload();
            });
        });


        
        const fontSizeOption = addElem(configList, 'dt', 'rinsp-config-item-lv1');
        addElem(fontSizeOption, 'span').textContent = '🗚 页面字体大小 ';
        const fontSizeSelect = addElem(fontSizeOption, 'select');
        addElem(fontSizeSelect, 'option', null, { value: '0' }).textContent = '论坛默认大小';
        addElem(fontSizeSelect, 'option', null, { value: '1' }).textContent = '列表标题大小 +1';
        addElem(fontSizeSelect, 'option', null, { value: '2' }).textContent = '列表标题及内容大小 +1';
        addElem(fontSizeSelect, 'option', null, { value: '3' }).textContent = '列表标题及内容大小 +2';
        fontSizeSelect.value = userConfig.textSize;
        fontSizeSelect.value = fontSizeSelect.value || '0';
        fontSizeSelect.addEventListener('change', () => {
            configPanel.classList.add('rinsp-config-saving');
            mainConfigAccess.update(function(updatingUserConfig) {
                updatingUserConfig.textSize = fontSizeSelect.value||0;
                return updatingUserConfig;
            })
            .finally(function() {
                configPanel.classList.remove('rinsp-config-saving');
                window.location.reload();
            });
        });

        const floatShortcutOption = addElem(configList, 'dt', 'rinsp-config-item-lv1');
        addElem(floatShortcutOption, 'span').textContent = '⚙️ 浮动式捷径窗口 ';
        const floatShortcutSelect = addElem(floatShortcutOption, 'select');
        addElem(floatShortcutSelect, 'option', null, { value: 'off' }).textContent = '不显示';
        addElem(floatShortcutSelect, 'option', null, { value: 'tr' }).textContent = '右上角';
        addElem(floatShortcutSelect, 'option', null, { value: 'tl' }).textContent = '左上角';
        floatShortcutSelect.value = userConfig.floatingShortcut;
        floatShortcutSelect.value = floatShortcutSelect.value || 'tr';
        floatShortcutSelect.addEventListener('change', () => {
            configPanel.classList.add('rinsp-config-saving');
            mainConfigAccess.update(function(updatingUserConfig) {
                updatingUserConfig.floatingShortcut = floatShortcutSelect.value;
                return updatingUserConfig;
            })
            .finally(function() {
                configPanel.classList.remove('rinsp-config-saving');
                window.location.reload();
            });
        });

        addElem(configList, 'dt', 'rinsp-config-item-lv1 rinsp-config-item-checked').textContent = '浮动功能捷径';
        putCheckBoxItem('浮动私信通知', 'showFloatingMessageIndicator', 2);
        putCheckBoxItem('浮动收藏按钮 (坛上内置功能)', 'showFloatingStoreThreadButton', 2);
        putCheckBoxItem('浮动关注按钮 (类似MARK++)', 'showFloatingWatchIndicator', 2);
        putCheckBoxItem('回到顶部按钮', 'showBackToTopButton', 2);

        addElem(configList, 'dt', 'rinsp-config-item-lv1 rinsp-config-item-checked').textContent = '自动无缝加载翻页';
        putCheckBoxItem('主题帖 - 列表', 'infiniteScroll$threads', 2);
        putCheckBoxItem('主题帖 - 图墙', 'infiniteScroll$threads_picwall', 2);
        putCheckBoxItem('用户主题', 'infiniteScroll$usertopics', 2);
        putCheckBoxItem('用户回复', 'infiniteScroll$userposts', 2);
        putCheckBoxItem('主题帖内容', 'infiniteScroll$thread_posts', 2);
        putCheckBoxItem('搜索结果', 'infiniteScroll$search', 2);
        putCheckBoxItem('收件箱', 'infiniteScroll$msg_inbox', 2);
        putCheckBoxItem('消息跟踪', 'infiniteScroll$msg_sent', 2);
        putCheckBoxItem('通信记录', 'infiniteScroll$msg_chatlog', 2);
        putCheckBoxItem('加载新页后取代地址 (即手动刷新时会停留在新的页数)', 'infiniteScrollReplaceURL', 2).classList.add('rinsp-config-item-sep');
        putCheckBoxItem('加载新页后滚动到新页顶部', 'infiniteScrollScrollToNewPage', 2);

        addElem(configList, 'dt', 'rinsp-config-item-lv1 rinsp-config-item-checked').textContent = '帖内标题';
        putCheckBoxItem('不灰暗化 (改为黑色)', 'siteThemeDarkerSubjectLine');
        putCheckBoxItem('隐藏多余的Re标题', 'hideRedundantReSubjectLine');
        putCheckBoxItem('增加可点击跳转回复原楼链接', 'addQuickJumpReSubjectLine');

        putCheckBoxItem('在列表上显示及加亮自己的主题及最后回复', 'highlightMyself').classList.add('rinsp-config-item-sep');
        putCheckBoxItem('列表内链接默认在新窗口打开', 'threadListDefaultOpenNewPage');
        putCheckBoxItem('发新帖吋自动选择 新回复站内通知', 'autoCheckReplyOption');
        putCheckBoxItem('色标出售框 (以及5sp以上警告)', 'enhanceSellFrame');
        putCheckBoxItem('购买免刷新', 'buyRefreshFree');
        putCheckBoxItem('回复免刷新', 'replyRefreshFree');
        putCheckBoxItem('显示网盘筛选栏', 'showShareTypeFilter');
        putCheckBoxItem('关注列表为空时不在主菜单上显示', 'hideWatchButtonIfEmpty');
        putCheckBoxItem('显示私信长度限制', 'showInputLimit');
        putCheckBoxItem('子分区承继当前图墙模式', 'subCategoryInheritImageWallMode');
        putCheckBoxItem('默认折叠版块公告', 'siteNoticeSectionDefaultFolded');
        putCheckBoxItem('显示默认图墙模式按钮', 'showDefaultingPicWallOption');
        putCheckBoxItem('隐藏 [-- 查看移动版 --]', 'hideMobileVerSwitch');

        addElem(configList, 'dt', 'rinsp-config-item-sep rinsp-config-item-lv1').textContent = '隐藏用户头像 (鼠标放在头像上才显示)';
        putCheckBoxItem('隐藏系统默认头像', 'hideDefaultUserpic');
        putCheckBoxItem('隐藏其他头像', 'hideOtherUserpic');
        putCheckBoxItem('收藏夹内用户除外', 'customUserBypassHideUserpic');
        putCheckBoxItem('自己头像除外', 'selfBypassHideUserpic');
        
        addElem(configList, 'dt', 'rinsp-config-item-sep rinsp-config-item-lv1').textContent = '其他用户头像及资料显示选项';
        putCheckBoxItem('替代弹出信息窗口内的快捷功能', 'useCustomUserInfoPopup');
        putCheckBoxItem('当前楼层用户头像常时可见', 'stickyUserInfo');
        putCheckBoxItem('显示更多用户信息 (头像下)', 'showExtendedUserInfo');
        putCheckBoxItem('显示及高亮负HP', 'showExtendedUserInfo$HP', 2);

        addElem(configList, 'dt', 'rinsp-config-item-sep rinsp-config-item-lv1 rinsp-config-item-checked').textContent = '显示焦点楼层快捷跳转栏';
        putCheckBoxItem('出售框', 'showResourceSpots$sells', 2);
        putCheckBoxItem('附件', 'showResourceSpots$attachments', 2);
        putCheckBoxItem('分享连接', 'showResourceSpots$shares', 2);
        putCheckBoxItem('图片', 'showResourceSpots$images', 2);
        putCheckBoxItem('其他连接', 'showResourceSpots$links', 2);
        putCheckBoxItem('喜欢关键词', 'showResourceSpots$likes', 2);
        putCheckBoxItem('浮动焦点列表 (页面左边)', 'showResourceSpotsFloating', 2).classList.add('rinsp-config-item-sep');
        
        putCheckBoxItem('减低高度占用量模式', 'heightReductionMode').classList.add('rinsp-config-item-sep');
        putCheckBoxItem('减低更多', 'heightReductionMode$LV2', 2);

        addElem(configList, 'dt', 'rinsp-config-item-sep rinsp-config-item-lv1 rinsp-config-item-checked').textContent = '调整论坛默认楼层折叠功能 (恢复论坛旧版模式)';
        putCheckBoxItem('完全不折叠', 'disablePostCollapse', 1, true);
        putCheckBoxItem('不折叠自己的楼层', 'disablePostCollapse$Self', 2);

        addElem(configList, 'dt', 'rinsp-config-item-lv1 rinsp-config-item-checked').textContent = '屏蔽内容/折叠 例外';
        putCheckBoxItem('不折叠及屏蔽收藏夹内用户', 'customUserBypassIgnoreList', 2);
        putCheckBoxItem('不折叠及屏蔽自订头像用户', 'customUserpicBypassIgnoreList', 2);


        addElem(configList, 'dt', 'rinsp-config-item-sep rinsp-config-item-lv1').textContent = '询问&求物区设置';
        putCheckBoxItem('加亮已有答案帖子', 'requestThreadHighlightEnded');
        putCheckBoxItem('显示额外悬赏', 'requestThreadShowExtraBounty');
        putCheckBoxItem('使用浏览记录显示更多资讯', 'requestThreadUseHistoryData');
        putCheckBoxItem('隐藏结算帖', 'hideSettlementPost');
        putCheckBoxItem('只灰暗化 不完全隐藏', 'hideSettlementPost$GreyoutOnly', 2);
        //        putCheckBoxItem('使用默认规则', 'hideSettlementPost$UseDefaultKeywords', 2);
        putCheckBoxItem('针对自己的结算帖除外', 'hideSettlementPost$HighlightMyself', 2);
        putCheckBoxItem('显示状态筛选操作 (在列表右边)', 'showRequestThreadFilters');
        putCheckBoxItem('求物区内不屏蔽用户回复', 'dontFilterRequestReplyByUser');
/*
        const ignoreSettlementItem = addElem(configList, 'dt', 'rinsp-config-item-lv2 rinsp-config-item-checked');
        const ignoreSettlementButton = addElem(ignoreSettlementItem, 'a');
        ignoreSettlementButton.textContent = '✏️编辑关键词列表';
        ignoreSettlementButton.addEventListener('click', function() {
            openIgnoreSettlementEditor();
        });
*/

        addElem(configList, 'dt', 'rinsp-config-item-sep rinsp-config-item-lv1 rinsp-config-item-checked').textContent = '改善帖列表上by乱码';
        putCheckBoxItem('改显示 by楼主 **如有记录时', 'replyShownAsByOp', 2);
        putCheckBoxItem('新帖无回复时不显示 by〷', 'hideZeroReply', 2);

        addElem(configList, 'dt', 'rinsp-config-item-sep rinsp-config-item-lv1 rinsp-config-item-checked').textContent = '🗞️记录已读帖子 - ⚠️占用资源较高';
        addElem(configList, 'dt', 'rinsp-config-item-tip rinsp-config-item-lv2').textContent = '💡 可搜索,主题及回复列表上也会显示记录中的完整标题';
        putCheckBoxItem('启用记录功能', 'keepVisitPostHistory', 2);
        putCheckBoxItem('主题及回复列表上显示原始标题', 'showInitialRememberedTitle', 2);
        const historySummaryItem = addElem(configList, 'dt', 'rinsp-config-item-tip rinsp-config-item-lv2');
        const historyActionItem = addElem(configList, 'dt', 'rinsp-config-item-lv2');
        const clearHistoryMenuButton = addElem(historyActionItem, 'a', 'rinsp-config-item-lv3');
        clearHistoryMenuButton.textContent = '🗑️ 打开清空菜单';
        clearHistoryMenuButton.addEventListener('click', async () => {
            openClearHistoryMenu(clearHistoryMenuButton, updateHistoryCount);
        });

        async function updateHistoryCount() {
            historySummaryItem.textContent = '💡 统计中...';
            const size = await threadHistoryAccess.historyStore.size();
            if (size > 0) {
                historySummaryItem.textContent = `💡 共大约 ${size} 条浏览记录`;
            } else {
                historySummaryItem.textContent = '💡 没有浏览记录';
            }
        }

        updateHistoryCount();


        const ignoreControlItem = addElem(configList, 'dt', 'rinsp-config-item-sep rinsp-config-item-lv1 rinsp-config-item-checked');
        const ignoreControlButton = addElem(ignoreControlItem, 'a');
        ignoreControlButton.textContent = '✏️ 编辑: 🚫屏蔽内容列表';
        ignoreControlButton.addEventListener('click', () => {
            openIgnoreListEditor();
        });
        addElem(configList, 'dt', 'rinsp-config-item-tip rinsp-config-item-lv2').textContent = '💡 南+内置的 (屏蔽此人) 会合并处理';
        addElem(configList, 'dt', 'rinsp-config-item-tip rinsp-config-item-lv2').textContent = '💡 自己的楼层不屏蔽';
        
        putCheckBoxItem('表情符号不予区分 (不同的表情都会一起屏蔽)', 'treatAllEmojiTheSameWay', 2);
        putCheckBoxItem('当有文字内容时不管其他表情 (例: 插眼)', 'ignoreContentUseTextOnly', 2);
        putCheckBoxItem('完全隐藏屏蔽内容的楼层', 'hideIgnoreContentPost', 2);
        addElem(configList, 'dt', 'rinsp-config-item-tip rinsp-config-item-lv3').textContent = '💡 不推荐 求帖有机会要选屏蔽内容为最佳以结帖';
        putCheckBoxItem('关注时屏蔽/折叠内容不予通知', 'watchSkipIgnoreContent', 2);
        putCheckBoxItem('收藏夹内用户的回复除外', 'customUserBypassWatchSkip', 3);

        const ignoreThreadItem = addElem(configList, 'dt', 'rinsp-config-item-sep rinsp-config-item-lv1 rinsp-config-item-checked');
        const ignoreThreadButton = addElem(ignoreThreadItem, 'a');
        ignoreThreadButton.textContent = '✏️ 编辑: 🚫屏蔽标题关键词列表';
        ignoreThreadButton.addEventListener('click', () => {
            openIgnoreThreadEditor();
        });
        putCheckBoxItem('完全隐藏被屏蔽的帖子', 'hideFilteredThread', 2);
        putCheckBoxItem('收藏夹内用户的帖子除外', 'customUserBypassThreadFilter', 2);

        const likeThreadItem = addElem(configList, 'dt', 'rinsp-config-item-lv1 rinsp-config-item-checked');
        const likeThreadButton = addElem(likeThreadItem, 'a');
        likeThreadButton.textContent = '✏️ 编辑: 💚喜欢帖子标题关键词列表 (屏蔽占优先权)';
        likeThreadButton.addEventListener('click', function() {
            openLikeThreadEditor();
        });
        putCheckBoxItem('浮动喜欢帖子列表', 'showFavThreadFloatingList', 2);
        putCheckBoxItem('显示标题前段文字', 'showFavThreadFloatingList$withTitle', 3);
        
        if (hasManagementRole) {
            addElem(configList, 'dt', 'rinsp-config-item-sep rinsp-config-item-lv1 rinsp-config-item-checked').textContent = '💯版主评分设置';
            putCheckBoxItem('免空区评分默认不发送通知', 'adminNoScoreNotifByDefault', 2);
            putCheckBoxItem('不显示 "格式不正" 快捷评分 (固定有通知)', 'adminHideMarkUnscoreButton', 2);
            putCheckBoxItem('不显示 "标记不评分" 快捷评分 (固定不通知)', 'adminHideMarkBadFormatButton', 2);
        }

        renderUserBookmarks();

        renderPinnedUsers();

        let userFilterConfig = await userFilterConfigAccess.read();
        renderUserBlacklist('屏蔽用户列表 - 屏蔽所有内容', entry => entry[1].hideReplies);
        renderUserBlacklist('屏蔽用户列表 - 只屏蔽主题帖', entry => entry[1].hideThreads && !entry[1].hideReplies);

        renderConfigManagement();

        function renderUserBookmarks() {
            const nickNameListHeading = addElem(configPanel, 'h5', 'u-h5');
            addElem(nickNameListHeading, 'span').textContent = '用户收藏夹';
            addElem(nickNameListHeading, 'span', null, { style: 'font-weight:normal'}).textContent = ' (亦记忆其昵称) 点击🔖打开菜单';
            const nickNameListHeadingControls = addElem(nickNameListHeading, 'span', 'r gray3');
            nickNameListHeadingControls.appendChild(document.createTextNode('顺序 '));
            const sortOrderSelect = addElem(nickNameListHeadingControls, 'select');
            addElem(sortOrderSelect, 'option', null, { value: 'uid' }).textContent = 'UID';
            addElem(sortOrderSelect, 'option', null, { value: 'nickname' }).textContent = '昵称';
            sortOrderSelect.value = userConfig.customUserOrderBy;
            sortOrderSelect.value = sortOrderSelect.value || 'uid';
            sortOrderSelect.addEventListener('change', () => {
                configPanel.classList.add('rinsp-config-saving');
                mainConfigAccess.update(function(updatingUserConfig) {
                    updatingUserConfig.customUserOrderBy = sortOrderSelect.value;
                    return updatingUserConfig;
                })
                .finally(function() {
                    configPanel.classList.remove('rinsp-config-saving');
                    window.location.reload();
                });
            });
    
            const nickNameMappingBlock = addElem(configPanel, 'dt', 'rinsp-nickname-list');

            let userItemComparator;
            if (userConfig.customUserOrderBy === 'nickname') {
                userItemComparator = comparator(2);
            } else {
                userItemComparator = comparator(item => item[0] * 1);
            }
    
            Object.entries(userConfig.customUserHashIdMappings)
                .filter(entry => entry[0][0]==='@')
                .map(entry => entry[1])
                .sort(userItemComparator)
                .forEach(function(entry) {
                    const userId = entry[0];
                    const userHashId = entry[1];
                    const userNickName = entry[2];
    
                    const userTag = addElem(nickNameMappingBlock, 'span', 'rinsp-user-tag', {
                        title: `${userNickName}\n${userHashId}`
                    });
                    nickNameMappingBlock.appendChild(document.createTextNode(' '));
                    const userTagIcon = addElem(userTag, 'span', 'rinsp-user-tag-icon');
    
                    const userLink = addElem(userTag, 'a', null, {
                        href: 'u.php?action-show-uid-' + userId + '.html',
                        target: '_blank'
                    });
                    userLink.textContent = userNickName;
                    userTagIcon.addEventListener('click', function() {
                        const deleteAction = () => {
                            if (confirm('取消收藏(' + userNickName + ')?')) {
                                configPanel.classList.add('rinsp-config-saving');
                                changeCustomUserMapping(userId, userHashId, '', true)
                                    .finally(function() {
                                        configPanel.classList.remove('rinsp-config-saving');
                                        userTag.remove();
                                    });
                            }
                        };
                        openBookmarkUserMenu(userTag, userId, userHashId, userNickName, deleteAction);
                    });
                });
        }

        async function renderPinnedUsers() {
            const pinListHeading = addElem(configPanel, 'h5', 'u-h5');
            addElem(pinListHeading, 'span').textContent = '焦点人物';
            configPanel.appendChild(putCheckBoxItem('焦点人物出现时才显示', 'hideInactivePinnedUsers'));
            const showActiveRepliersControl = putCheckBoxItem('自动显示当前页活跃回复者', 'showActiveRepliers');
            const showActiveRepliersMinControl = addElem(showActiveRepliersControl, 'a', null, { href: 'javascript:' });
            showActiveRepliersMinControl.addEventListener('click', async () => {
                let minCount = prompt('最低回复次数', '') * 1;
                if (minCount > 1) {
                    configPanel.classList.add('rinsp-config-saving');
                    mainConfigAccess.update(function(updatingUserConfig) {
                        updatingUserConfig.showActiveRepliers$min = minCount;
                        userConfig = updatingUserConfig;
                        return updatingUserConfig;
                    })
                    .finally(function() {
                        configPanel.classList.remove('rinsp-config-saving');
                        updateMinReplyDisplay();
                    });
                }
            });
            function updateMinReplyDisplay() {
                const minReplyCount = userConfig.showActiveRepliers$min > 1 ? userConfig.showActiveRepliers$min : 2;
                showActiveRepliersMinControl.textContent = ` [最低回复次数: ${minReplyCount}]`;
            }
            updateMinReplyDisplay();
            
            configPanel.appendChild(showActiveRepliersControl);
            const recordListBlock = addElem(configPanel, 'dt', 'rinsp-pinned-user-list');
            const pinnedUsersConfig = await pinnedUsersConfigAccess.read();
            Object.values(pinnedUsersConfig.users).forEach(record => {
                const item = addElem(recordListBlock, 'div', 'rinsp-pinuser-item');
                const face = addElem(item, 'div', 'rinsp-pinuser-item-face');
                addElem(face, 'img', null, { src: record.img });
                const label = addElem(face, 'div');
                addElem(label, 'a', null, { href: `u.php?action-show-uid-${record.uid}.html`, target: '_blank' }).textContent = record.nickName||`#${record.uid}`;
                const unpin = addElem(label, 'div', 'rinsp-pinuser-unpin-icon');
                unpin.addEventListener('click', async () => {
                    await pinnedUsersConfigAccess.update(function(updatingPinnedUsersConfig) {
                        delete updatingPinnedUsersConfig.users['#' + record.uid];
                        // clean up corrupt data ...
                        delete updatingPinnedUsersConfig.users['#NaN'];
                        delete updatingPinnedUsersConfig.users['#null'];
                        return updatingPinnedUsersConfig;
                    });
                    item.remove();
                });
            });

        }

        function renderUserBlacklist(heading, filter) {
            const blockedUserListHeading = addElem(configPanel, 'h5', 'u-h5');
            addElem(blockedUserListHeading, 'span').textContent = heading;
            addElem(blockedUserListHeading, 'span', null, { style: 'font-weight:normal'}).textContent = ' 点击🚫移除';
    
            let userItemComparator;
            if (userConfig.customUserOrderBy === 'nickname') {
                userItemComparator = comparator(item => item[1].name);
            } else {
                userItemComparator = comparator(item => item[0].substring(1) * 1);
            }

            const blockedUserListBlock = addElem(configPanel, 'dt', 'rinsp-user-blacklist');
            Object.entries(userFilterConfig.users)
                .filter(entry => entry[0][0]==='#')
                .sort(userItemComparator)
                .filter(filter)
                .forEach(function(entry) {
                    const userId = entry[0].substring(1) * 1;
                    const userNickName = entry[1].name;
                    const hideThreads = entry[1].hideThreads;
                    const hideReplies = entry[1].hideReplies;
                    if (!hideThreads && !hideReplies) {
                        return;
                    }
                    const userTag = addElem(blockedUserListBlock, 'span', 'rinsp-user-tag', {
                        title: userNickName + '\n' + (hideReplies ? '屏蔽所有内容' : '只屏蔽主题帖')
                    });
                    blockedUserListBlock.appendChild(document.createTextNode(' '));
                    const userTagDel = addElem(userTag, 'span', 'rinsp-user-tag-icon');
    
                    const userLink = addElem(userTag, 'a', null, {
                        href: 'u.php?action-show-uid-' + userId + '.html',
                        target: '_blank'
                    });
                    userLink.textContent = userNickName;
                    userTagDel.addEventListener('click', function() {
                        if (confirm('取消屏蔽(' + userNickName + ')?')) {
                            configPanel.classList.add('rinsp-config-saving');
                            userFilterConfigAccess
                                .update(function(userFilterConfig) {
                                    delete userFilterConfig.users[entry[0]];
                                    return userFilterConfig;
                                })
                                .finally(function() {
                                    configPanel.classList.remove('rinsp-config-saving');
                                    userTag.remove();
                                });
                        }
                    });
                });
        }
    
        function putCheckBoxItem(label, propertyNameOrAccessor, level, inverted) {
            let propertyAccess;
            if (typeof propertyNameOrAccessor === 'string') {
                propertyAccess = {
                    get() {
                        return !!userConfig[propertyNameOrAccessor];
                    },
                    update(newValue) {
                        return mainConfigAccess.update(function(updatingUserConfig) {
                            updatingUserConfig[propertyNameOrAccessor] = newValue;
                            return updatingUserConfig;
                        });
                    }
                };
            } else {
                propertyAccess = propertyNameOrAccessor;
            }
            const configItem = addElem(configList, 'dt', `rinsp-config-item-lv${level||1}`);
            const checkboxLabel = addElem(configItem, 'label');
            const checkbox = addElem(checkboxLabel, 'input', null, { type: 'checkbox' });
            checkboxLabel.appendChild(document.createTextNode(' ' + label));
            checkbox.checked = propertyAccess.get();
            updateStyle();
            function updateStyle() {
                const checkedValue = !inverted; // when inverted false is considered checked
                if (checkbox.checked === checkedValue) {
                    configItem.classList.add('rinsp-config-item-checked');
                } else {
                    configItem.classList.remove('rinsp-config-item-checked');
                }
            }
            checkbox.addEventListener('change', async function() {
                configPanel.classList.add('rinsp-config-saving');
                const newValue = checkbox.checked;
                propertyAccess.update(newValue)
                    .finally(function() {
                        configPanel.classList.remove('rinsp-config-saving');
                        updateStyle();
                    });
            });
            return configItem;
        }

        function renderConfigManagement() {
            const nickNameListHeading = addElem(configPanel, 'h5', 'u-h5');
            addElem(nickNameListHeading, 'span').textContent = '管理设置 (仅当前用户)';
            addElem(nickNameListHeading, 'span', null, { style: 'font-weight:normal'}).textContent = ' ⚠️进阶操作';
    
            const panel = addElem(configPanel, 'div', 'rinsp-config-manage-panel');
            const exportButton = addElem(panel, 'button', null, { type: 'button' });
            exportButton.textContent = '导出设置';
            exportButton.addEventListener('click', () => downloadUserSettings());
            
            addElem(panel, 'span').textContent = ' / ';
            
            const importButton = addElem(panel, 'button', null, { type: 'button' });
            importButton.textContent = '导入设置';
            addElem(panel, 'span').textContent = ' ';
            const uploadInput = addElem(panel, 'input', null, { type: 'file', accept: '.rpcfg', style: 'max-width: 18em' });
            importButton.addEventListener('click', () => importSettings(uploadInput));

        }
    }

    function rewriteUserRecentPostLinks(userConfig, userPinArea) {
        document.querySelectorAll('#minifeed .feed-list dt > a[href^="//"]').forEach(el => {
            const href = el.getAttribute('href');
            const tid = (href.match(/\/read.php\?tid-(\d+)\.html$/)||[])[1];
            if (tid) {
                el.setAttribute('href', `/read.php\?tid-${tid}.html`);
            }
        });
    }

    async function importSettings(input) {
        if (!input.value) {
            showMessagePopup('💡请选择文件', input, 1500);
            return;
        }
        async function readData() {
            const dataStream = new Blob([input.files[0]]).stream().pipeThrough(new DecompressionStream('gzip'));
            const rawData = await (await new Response(dataStream).blob()).text();
            return JSON.parse(rawData);
        }
        async function init() {
            const data = await readData();
            const userDataMap = new Map();
            Object.keys(data).forEach(key => {
                if (key[0] === '#') {
                    const id = key.substring(1) * 1;
                    if (id > 0) {
                        userDataMap.set(id, data[key]);
                    }
                }
            });
            if (userDataMap.size !== 1) {
                showMessagePopup('⚠️数据错误', null, 3000);
                return;
            }
            const importedUserId = Array.from(userDataMap.keys())[0];
            if (importedUserId !== myUserId) {
                if (!confirm('⚠️导入数据与用户ID不同,继续操作?')) {
                    return;
                }
            }

            setupPopupMenu({
                title: '已读取数据,请选择操作',
                width: 260,
                popupMenuId: 'CONFIG_IMPORT_ACTION_MENU',
                anchor: input,
                verticallyInverted: true,
                items: [
                    {
                        label: '❗操作前请先关闭其他页面❗',
                        cellClass: 'rinsp-alert-menu-message'
                    },
                    {
                        label: '⚠️注意,请先备份',
                        cellClass: 'rinsp-alert-menu-message'
                    },
                    {
                        label: '合并设置',
                        class: 'rinsp-alert-menu-button',
                        action: () => {
                            if (confirm('现有设置及资料会被改写!!确认执行?')) {
                                const action = async cnt => {
                                    const mergedUserData = await mergeUserData(myUserId, userDataMap.get(importedUserId));
                                    if (!confirm('合并准备完成,最终确认置换!?')) {
                                        return;
                                    }
                                    await importUserSettings(myUserId, mergedUserData);
                                    return '✔️导入完成,自动刷新窗口';
                                };
                                runWithProgressPopup(action, '导入中...', null, 5000)
                                    .then(() => document.location.reload())
                                    .catch(ex => { console.error(ex); alert('' + ex); });
                            }
                        }
                    },
                    {
                        label: '完全覆盖置换',
                        class: 'rinsp-alert-menu-button',
                        action: () => {
                            if (confirm('现有设置及资料会被覆盖!!确认执行?')) {
                                const action = async cnt => {
                                    await importUserSettings(myUserId, userDataMap.get(importedUserId));
                                    return '✔️导入完成,自动刷新窗口';
                                };
                                runWithProgressPopup(action, '导入中...', null, 5000)
                                    .then(() => document.location.reload())
                                    .catch(ex => { console.error(ex); alert('' + ex); });
                            }
                        }
                    },
                    {
                        label: '取消操作',
                        action: () => input.value = null
                    }
                ]
            });
        }
        init().catch(ex => showMessagePopup('⚠️' + ex, input, 3000));
    }

    function mergeArrayData(newData, currentData, keyFunc) {
        const set = new Set();
        newData.forEach(item => {
            set.add(keyFunc(item));
        });
        return currentData.filter(item => !set.has(keyFunc(item))).concat(newData);
    }

    async function importUserSettings(userId, userData) {
        const userKey = '#' + userId;
        const allKeys = await GM.listValues();
        const myKeys = allKeys.filter(key => key.split(':', 1)[0].endsWith(userKey));
        
        for (let key of myKeys) {
            await GM.deleteValue(key).catch(ex => null);
        }
        const entries = Object.entries(userData);
        for (let entry of entries) {
            let dbKey;
            if (entry[0].indexOf(':') === -1) {
                dbKey = entry[0] + userKey;
            } else {
                dbKey = entry[0].replace(':', userKey + ':');
            }
            await GM.setValue(dbKey, entry[1]);
        }
    }

    async function mergeUserData(userId, userData) {
        let currentUserData = await exportUserSettings(userId);

        function merger(configKey) {
            const newData = JSON.parse(userData[configKey]||'{}');
            const baseData = JSON.parse(currentUserData[configKey]||'{}');
            return {
                mergeArrayEntry(key, itemKeyFunc) {
                    newData[key] = mergeArrayData(newData[key]||[], baseData[key]||[], itemKeyFunc||(x=>x));
                    return this;
                },
                mergeKeyValueEntry(key) {
                    newData[key] = Object.assign({}, baseData[key]||{}, newData[key]||{});
                    return this;
                },
                getBaseData() {
                    return baseData;
                },
                getData() {
                    return newData;
                },
                replacing(targetData) {
                    targetData[configKey] = JSON.stringify(newData);
                }
            };
        }
        const newUserData = Object.assign({}, userData);

        merger(MAIN_CONFIG_KEY)
            .mergeKeyValueEntry('customUserHashIdMappings')
            .replacing(newUserData);

        merger(CONTENT_IGNORE_LIST_CONFIG_KEY)
            .mergeArrayEntry('terms')
            .mergeArrayEntry('exceptions')
            .replacing(newUserData);

        merger(SEARCH_CONFIG_KEY)
            .mergeKeyValueEntry('pinnedTopics')
            .replacing(newUserData);
        
        merger(THREAD_CUSTOM_CATEGORY_CONFIG_KEY)
            .mergeArrayEntry('keywords')
            .replacing(newUserData);
        
        merger(THREAD_FILTER_CONFIG_KEY)
            .mergeArrayEntry('dislikes')
            .mergeArrayEntry('likes')
            .mergeArrayEntry('settlementKeywords')
            .replacing(newUserData);

        merger(USER_FILTER_CONFIG_KEY)
            .mergeKeyValueEntry('users')
            .replacing(newUserData);

        merger(PINNED_USERS_CONFIG_KEY)
            .mergeKeyValueEntry('users')
            .replacing(newUserData);

        merger(SHARETYPE_FILTER_CONFIG_KEY)
            .mergeArrayEntry('hides')
            .replacing(newUserData);

        const my_watchlist_merger = merger('my_watchlist');
        newUserData['my_watchlist'] = JSON.stringify(Object.assign({}, my_watchlist_merger.getBaseData(), my_watchlist_merger.getData()));
        
        const tid_visit_history$map = new Map();
        Object.keys(userData).filter(key => key.startsWith('tid_visit_history:#')).forEach(key => {
            const itemMap = new Map();
            userData[key].split(' ').map(item => item.split(':')).forEach(pair => {
                const id = pair[0] * 1;
                if (id > 0) {
                    const count = (pair[1] * 1)||0;
                    itemMap.set(id, count);
                }
            });
            tid_visit_history$map.set(key, itemMap);
        });
        Object.keys(currentUserData).filter(key => key.startsWith('tid_visit_history:#')).forEach(key => {
            let itemMap;
            if (tid_visit_history$map.has(key)) {
                itemMap = tid_visit_history$map.get(key);
            } else {
                itemMap = new Map();
                tid_visit_history$map.set(key, itemMap);
            }
            currentUserData[key].split(' ').map(item => item.split(':')).forEach(pair => {
                const id = pair[0] * 1;
                if (id > 0) {
                    const count = (pair[1] * 1)||0;
                    itemMap.set(id, Math.max(count, itemMap.get(id)||0));
                }
            });
        });
        const tid_visit_history$index = Array.from(tid_visit_history$map.entries()).map(entry => entry[0].split('#', 2)[1] + ':' + entry[1].size).join(' ');
        Object.keys(newUserData).filter(key => key.startsWith('tid_visit_history:#')).forEach(key => delete newUserData[key]);
        tid_visit_history$map.forEach((itemMap, key) => {
            newUserData[key] = Array.from(itemMap.entries()).map(entry => entry[1] > 0 ? `${entry[0]}:${entry[1]}` : `${entry[0]}`).join(' ');
        });
        newUserData['tid_visit_history:index'] = tid_visit_history$index;

        const tid_recent$map = new Map();
        Object.keys(userData).filter(key => key.startsWith('tid_recent:#')).forEach(key => {
            const record = JSON.parse(userData[key]);
            tid_recent$map.set(record.tid, record);
        });
        Object.keys(currentUserData).filter(key => key.startsWith('tid_recent:#')).forEach(key => {
            const record = JSON.parse(currentUserData[key]);
            const dup = tid_recent$map.get(record.tid);
            if (dup == null || record.time > dup.time) {
                tid_recent$map.set(record.tid, record);
            }
        });
        const sortByVisitTime = comparator('time', true);
        const tid_recent$list = Array.from(tid_recent$map.values()).sort(sortByVisitTime).slice(0, MAX_RECENT_ACCESS_LOG);
        let tid_recent$index = '';
        if (tid_recent$list.length > 0) {
            let minTime = (tid_recent$list[0]||{}).time||0;
            tid_recent$list.forEach(record => {
                if (minTime > record.time) minTime = record.time;
            });
            tid_recent$index = tid_recent$list.map(record => record.tid + ':' + (record.time - minTime)).join(' ') + ' 0:' + minTime;
        }
        tid_recent$list.forEach(record => {
            newUserData['tid_recent:#' + record.tid] = JSON.stringify(record);
        });
        newUserData['tid_recent:recent'] = tid_recent$index;
        
        Object.keys(currentUserData)
            .filter(key => !key.startsWith('tid_visit_history:#') && !key.startsWith('tid_recent:#') && !Object_hasOwn(newUserData, key))
            .forEach(key => {
                newUserData[key] = currentUserData[key];
            });

        const user_punish$map = new Map();
        Object.keys(userData).filter(key => key.startsWith('user_punish:#')).forEach(key => {
            const record = JSON.parse(userData[key]);
            record.uid = key.substring(13) * 1;
            user_punish$map.set(key, record);
        });
        Object.keys(currentUserData).filter(key => key.startsWith('user_punish:#')).forEach(key => {
            const current = JSON.parse(currentUserData[key]);
            current.uid = key.substring(13) * 1;
            const dup = user_punish$map.get(key);
            if (dup == null) {
                user_punish$map.set(key, current);
            } else {
                dup.logs = mergeArrayData(current.logs, dup.logs, x=>x.time);
            }
        });
        user_punish$map.forEach((record, key) => {
            newUserData[key] = JSON.stringify(record);
        });
        return newUserData;
    }

    async function exportUserSettings(userId) {
        const userKey = '#' + userId;
        const allKeys = await GM.listValues();
        const myKeys = allKeys.filter(key => key.split(':', 1)[0].endsWith(userKey));
        const myData = {};
        for (let key of myKeys) {
            const value = await GM.getValue(key);
            const parts = key.split(':');
            parts[0] = parts[0].substring(0, parts[0].length - userKey.length);
            const localKey = parts.join(':');
            myData[localKey] = value;
        }
        return myData;
    }

    async function downloadUserSettings() {
        const action = async () => {
            const userData = await exportUserSettings(myUserId);
            const data = {};
            data['#' + myUserId] = userData;
            const utf8Bytes = new TextEncoder().encode(JSON.stringify(data));
            const chunks = [];
            const reader = new Blob([utf8Bytes]).stream().pipeThrough(new CompressionStream('gzip')).getReader();
            while (true) {
                const status = await reader.read();
                if (status.done) break;
                chunks.push(status.value);
            }
            const blob = new Blob(chunks, { type: 'application/octet-stream' });

            //const blob = new Blob([base64data], { type: 'text/plain;charset=utf-8' });
            
            const downloadLink = document.createElement('a');
            downloadLink.href = URL.createObjectURL(blob);
            downloadLink.download = `rinplus_${myUserId}_${new Date().toISOString().replace(/:/g, '').replace(/\.\d{3}Z$/, '')}.rpcfg`;
            downloadLink.click();
            URL.revokeObjectURL(downloadLink.href);
            return '✔️已触发下载';
        };
        await runWithProgressPopup(action, '导出中...', null, 1500)
            .catch((ex) => {
                console.error(ex);
                alert(String(ex));
            });
    }

    async function openIgnoreListEditor(callback) {
        openTextListEditor(IGNORE_LIST_POPUP_MENU_ID, {
            title: '🚫屏蔽内容列表',
            async read() {
                const ignoreList = await contentIgnoreListConfigAccess.read();
                return ignoreList.terms.sort().join('\n');
            },
            async save(textData) {
                return await contentIgnoreListConfigAccess.update(function(ignoreList) {
                    const newTerms = textData.split('\n').map(itm=>itm.trim()).filter(itm=>!!itm);
                    ignoreList.terms = newTerms;
                    return ignoreList;
                });
            }
        }, callback);
    }

    async function openIgnoreThreadEditor(callback) {
        openTextListEditor(THREAD_FILTER_POPUP_MENU_ID, {
            title: '🚫屏蔽标题关键词列表',
            async read() {
                const filterConfig = await threadFilterConfigAccess.read();
                return filterConfig.dislikes.sort().join('\n');
            },
            async save(textData) {
                return await threadFilterConfigAccess.update(function(filterConfig) {
                    const newKeywords = textData.toLowerCase().split('\n').map(itm=>itm.replace(/\s+/g, ' ').trim()).filter(itm=>!!itm);
                    filterConfig.dislikes = newKeywords;
                    return filterConfig;
                });
            }
        }, callback);
    }

    async function openLikeThreadEditor(callback) {
        openTextListEditor(THREAD_LIKE_POPUP_MENU_ID, {
            title: '💚喜欢帖子标题关键词列表',
            description: `每行一个关键字
正则表达式高级设置 (懂的都懂 不懂可无视) 格式: regex:/正则表达式语法/
      例子: regex:/[东南西北]\+/`,
            async read() {
                const filterConfig = await threadFilterConfigAccess.read();
                return filterConfig.likes.sort().join('\n');
            },
            async save(textData) {
                return await threadFilterConfigAccess.update(function(filterConfig) {
                    const newKeywords = textData.toLowerCase().split('\n').map(itm=>itm.replace(/\s+/g, ' ').trim()).filter(itm=>!!itm);
                    filterConfig.likes = newKeywords;
                    return filterConfig;
                });
            }
        }, callback);
    }

    function addPrivateMessageNotifier() {
        const pmElem = document.querySelector('#td_msg');
        const observer = new MutationObserver(function() {
            updatePrivateMessageStatus();
        });
        updatePrivateMessageStatus();
        observer.observe(pmElem, { childList: true, subtree: true, attributes: true });

        function updatePrivateMessageStatus() {
            const hasMessage = pmElem.querySelector('img[src="images/colorImagination/shortmail.gif"]') != null;
            pmNotificationElem.textContent = '';
            if (hasMessage) {
                pmNotificationElem.classList.add('rinsp-status-new');
                addElem(pmNotificationElem, 'img', null, { src: 'images/colorImagination/shortmail.gif', align: 'absmiddle' });
                pmNotificationElem.appendChild(document.createTextNode('\n未\n读\n私\n信'));
            } else {
                pmNotificationElem.classList.remove('rinsp-status-new');
            }
        }
    }

    function addTitleLengthLimitIndicator() {
        const titleField = document.querySelector('form[action="post.php?"] input[name="atc_title"]');
        if (titleField) {
            addLengthLimit(titleField, 300, 100, true, true);
        }
    }

    function addWatchMenuButton() {
        const userLogin = document.querySelector('#user-login #td_msg');
        if (userLogin != null) {
            const watchMenuElem = newElem('a', 'rinsp-watch-menuitem', {
                href: 'javascript:void(0)'
            });
            watchMenuElem.textContent = '关注';
            userLogin.after(watchMenuElem);
            return watchMenuElem;
        }
        return null;
    }

    function initTopMenu() {

        const watchMenuElem = addWatchMenuButton();
        if (watchMenuElem == null) {
            return;
        }
        watchNotificationElem.addEventListener('click', () => {
            document.body.scrollIntoView();
            watchMenuElem.click();
        });

        watchMenuElem.addEventListener('click', function() {
            showWatchMenu();
            return false;
        });

        async function tryCheck(override) {
            if (DEBUG_MODE) console.info('tryCheck (override=' + (override||null) + ')');
            let pending = [];
            let skipUntil = Date.now() + TEMP_INCREMENT;
            await updateWatchList(function(watchList) {
                const now = Date.now();
                const watchItems = Object.values(watchList).sort(comparator('lastChecked'));
                if (watchItems.length === 0) {
                    return null;
                }
                let checkableOldestItem = null;
                for (let watchItem of watchItems) {
                    if (isWatchExpired(watchItem)) {
                        continue;
                    }
                    if (typeof override === 'number') {
                        if (watchItem.id !== override) {
                            continue;
                        }
                    } else if (override !== true) {
                        let outdatedBy = now - watchItem.lastChecked;
                        if (watchItem.skipUntil > now) {
                            if (DEBUG_MODE) console.info('skip-for-now', watchItem.id);
                            continue;
                        }
                        if (outdatedBy < getScaledValue(watchItem, num(MIN_CHECK_INTERVAL), num(MAX_CHECK_INTERVAL)) * 60000) {
                            if (checkableOldestItem == null) {
                                if (outdatedBy >= getScaledValue(watchItem, num(MIN_CHECK_INTERVAL_OLDEST_ITEM), num(MAX_CHECK_INTERVAL_OLDEST_ITEM)) * 60000) {
                                    checkableOldestItem = watchItem;
                                }
                            }
                            continue;
                        }
                    }
                    pending.push(watchItem.id);
                    watchItem.skipUntil = skipUntil;
                }

                if (pending.length === 0 && checkableOldestItem != null) {
                    if (DEBUG_MODE) console.info('tryCheck - picked one oldest item: ' + checkableOldestItem.id);
                    pending.push(checkableOldestItem.id);
                    checkableOldestItem.skipUntil = skipUntil;
                }
                if (pending.length > 0) {
                    return watchList;
                } else {
                    return null;
                }
            });
            if (pending.length > 0) {
                if (DEBUG_MODE) console.info('tryCheck - ' + pending.length  + ' items');
                watchMenuElem.classList.add('rinsp-checking');
                await runCheck(pending, skipUntil)
                    .finally(function() {
                        watchMenuElem.classList.remove('rinsp-checking');
                    });
            } else {
                if (DEBUG_MODE) console.info('tryCheck - nothing to do');
            }
            await refreshWatchStatus();
        }

        async function runCheck(postIds, mySkipUntil) {
            let first = true;
            const ignoreList = await contentIgnoreListConfigAccess.read();
            async function checkWatchPost(watchList, watchPostId) {
                const watchItem = watchList['#' + watchPostId];
                if (watchItem == null) {
                    return null;
                }
                const now = Date.now();
                if (isWatchExpired(watchItem)) {
                    return null;
                }
                if (watchItem.skipUntil !== mySkipUntil && watchItem.skipUntil > now) {
                    return null;
                }
                watchItem.lastChecked = now;

                // enforce request rate control
                while (true) {
                    const rateInterval = Math.floor(now / MAX_ACC_REQUEST_RATE_BASE);
                    const lastRateRecord = (localStorage.getItem(REQUEST_RATE_RECORD_KEY)||'').split(':').map(item=>item*1);
                    if (rateInterval === lastRateRecord[0]) {
                        const accessCount = (lastRateRecord[1]*1)||0;
                        if (accessCount >= MAX_ACC_REQUEST_RATE_COUNT) {
                            if (DEBUG_MODE) console.info('HITTING RATE LIMIT - WAITING');
                            await sleep(MIN_REQUEST_DELAY);
                            continue;
                        }
                        localStorage.setItem(REQUEST_RATE_RECORD_KEY, `${rateInterval}:${accessCount + 1}`);
                    } else {
                        localStorage.setItem(REQUEST_RATE_RECORD_KEY, `${rateInterval}:1`);
                    }
                    break;
                }

                const lastPost = await fetchLastPost(watchPostId, ignoreList);
                if (lastPost.error) {
                    if (lastPost.errorMessage.indexOf('刷新不要') != -1) {
                        watchItem.skipUntil = Date.now() + MIN_REQUEST_DELAY;
                        return watchList;
                    } else {
                        watchItem.skipUntil = Number.MAX_SAFE_INTEGER;
                        watchItem.lastError = lastPost.errorMessage;
                        return watchList;
                    }
                }

                let bountyEnded = null;
                if (watchItem.bountyUntil > 0) {
                    if (lastPost.page > 1) {
                        await sleep(1500);
                        bountyEnded = !!(await fetchCheckBountyEnded(watchPostId));
                    } else {
                        bountyEnded = lastPost.bountyEnded;
                    }
                }

                if (lastPost.postUid === myUserId && lastPost.page == lastPost.maxPage && (lastPost.floor - watchItem.maxFloor) <= 1) {
                    // note: only can trust my own last post if the floor differs by one only
                    watchItem.lastVisitedPage = lastPost.page;
                    watchItem.lastVisitedFloor = lastPost.floor;
                    watchItem.lastVisitedPostId = lastPost.postId;
                }
                let lastUpdateIgnorable = null;
                if (userConfig.watchSkipIgnoreContent) {
                    const lastSeenPost = lastPost.postsByFloor[watchItem.lastVisitedFloor];
                    if (lastSeenPost != null && lastSeenPost.postId === watchItem.lastVisitedPostId) {
                        // last seen post is still valid and at the same floor
                        if (DEBUG_MODE) console.info('check floors', watchItem.lastVisitedFloor + 1, lastPost.floor);
                        for (let floor = watchItem.lastVisitedFloor + 1; floor <= lastPost.floor; floor++) {
                            if (lastUpdateIgnorable == null) {
                                lastUpdateIgnorable = true;
                            }
                            const newPost = lastPost.postsByFloor[floor];
                            if (!newPost.ignorable) {
                                lastUpdateIgnorable = false;
                                break;
                            }
                            if (userConfig.customUserBypassWatchSkip) {
                                if (userConfig.customUserHashIdMappings['#' + newPost.postUid] != null) {
                                    lastUpdateIgnorable = false;
                                    break;
                                }
                            }
                        }
                    }
                }
                if (DEBUG_MODE) console.info('ignorable', watchItem, lastUpdateIgnorable);
                watchItem.lastUpdateIgnorable = !!lastUpdateIgnorable;
                watchItem.maxPage = lastPost.maxPage;
                watchItem.maxFloor = lastPost.floor;
                watchItem.maxFloorApprox = lastPost.maxPage !== lastPost.page;
                watchItem.lastPostId = lastPost.postId;
                watchItem.lastPostUid = lastPost.postUid;
                watchItem.lastPostTime = lastPost.postTime;
                watchItem.skipUntil = 0;
                watchItem.lastError = null;
                if (bountyEnded) {
                    watchItem.bountyUntil = -1;
                }
                return watchList;
            }
            async function checkAndUpdateWatchPost(watchPostId) {
                if (DEBUG_MODE) console.info('checking ... ' + watchPostId);
                watchMenuElem.classList.add('rinsp-checking');
                await updateWatchList(
                    async function(watchList) {
                        const updatedWatchList = await checkWatchPost(watchList, watchPostId);
                        if (updatedWatchList != null) {
                            refreshWatchStatus(updatedWatchList);
                        }
                        return updatedWatchList;
                    })
                    .catch(e => {
                        console.warn(`error when checking post ${watchPostId}:`, e);
                        return true;
                    })
                    .finally(() => {
                        pendingWatchItemChecks.delete(watchPostId);
                        if (pendingWatchItemChecks.size === 0) {
                            watchMenuElem.classList.remove('rinsp-checking');
                        }
                        if (document.getElementById(WATCHER_POPUP_MENU_ID)) {
                            showWatchMenu();
                        }
                    });
                if (DEBUG_MODE) console.info('done ... ' + watchPostId);
            }
            const checkablePostIds = [];
            postIds.forEach(postId => {
                if (!pendingWatchItemChecks.has(postId)) {
                    pendingWatchItemChecks.add(postId);
                    checkablePostIds.push(postId);
                }
            });
            if (document.getElementById(WATCHER_POPUP_MENU_ID)) {
                showWatchMenu();
            }
            for (let watchPostId of checkablePostIds) {
                if (first) {
                    //console.info('wait');
                    await sleep(MIN_REQUEST_DELAY);
                } else {
                    first = false;
                }
                await checkAndUpdateWatchPost(watchPostId);
            }
        }

        async function refreshWatchStatus(initWatchList) {
            const watchList = initWatchList ? initWatchList : await readWatchList();
            let errorCount = 0;
            let updateCount = 0;
            let totalCount = 0;
            for (let watchItem of Object.values(watchList)) {
                const itemStatus = getWatchItemStatus(watchItem);
                if (itemStatus === 'new') {
                    updateCount++;
                } else if (itemStatus === 'error') {
                    errorCount++;
                }
                totalCount++;
            }
            if (totalCount > 0) {
                watchMenuElem.classList.add('rinsp-status-enabled');
            } else {
                watchMenuElem.classList.remove('rinsp-status-enabled');
            }
            
            watchNotificationElem.textContent = '';
            if (updateCount > 0) {
                watchMenuElem.classList.add('rinsp-status-new');
                watchMenuElem.dataset.newCount = updateCount;
                watchNotificationElem.classList.add('rinsp-status-new');
                addElem(watchNotificationElem, 'div', 'rinsp-notification-item-count').textContent = String(updateCount);
                watchNotificationElem.appendChild(document.createTextNode('关\n注\n更\n新'));
            } else {
                watchMenuElem.dataset.newCount = '';
                watchMenuElem.classList.remove('rinsp-status-new');
                watchNotificationElem.classList.remove('rinsp-status-new');
            }
            if (errorCount > 0) {
                watchMenuElem.classList.add('rinsp-status-error');
                watchNotificationElem.classList.add('rinsp-status-error');
                watchNotificationElem.textContent = '⚠️\n关\n注\n出\n错';
            } else {
                watchMenuElem.classList.remove('rinsp-status-error');
                watchNotificationElem.classList.remove('rinsp-status-error');
            }
        }

        document.addEventListener('visibilitychange', function() {
            if (document.visibilityState === 'visible') {
                setTimeout(function() {
                    if (document.visibilityState === 'visible') {
                        tryCheck();
                    }
                }, 3000);
            }
        });

        refreshWatchStatus().then(() => {
            setTimeout(function() {
                tryCheck();
            }, 3000);
        });

        setInterval(function() {
            if (document.visibilityState === 'visible') {
                tryCheck();
            }
        }, num(MIN_CHECK_INTERVAL_OLDEST_ITEM) * 60000 + 1000);


        const currentThreadWatchControls = [];
        let currentThreadWatching = false;
        const currentThread = {
            register(ctrl) {
                currentThreadWatchControls.push(ctrl);
            },
            isWatching() {
                return currentThreadWatching;
            },
            setWatching(watching) {
                currentThreadWatching = watching;
                currentThreadWatchControls.forEach(ctrl => ctrl.setWatching(watching));
            },
            setBusy(busy) {
                currentThreadWatchControls.forEach(ctrl => ctrl.setBusy(busy));
            }
        };
        return {
            currentThread,
            tryCheck
        };
    }

    function getWatchItemStatus(watchItem) {
        if (watchItem.lastError) {
            return 'error';
        }
        if (watchItem.lastUpdateIgnorable) {
            return 'ignorable';
        }
        if (watchItem.lastPostUid !== myUserId && watchItem.maxFloor != watchItem.lastVisitedFloor) {
            return 'new';
        }
    }

    async function showWatchMenu() {
        closePopupMenu(WATCHER_POPUP_MENU_ID);
        const popupMenu = createPopupMenu(WATCHER_POPUP_MENU_ID, document.querySelector('#user-login'));
        const watchList = await readWatchList();
        popupMenu.renderContent(async function(borElem) {
            const watchItems = Object.values(watchList);
            if (watchItems.length === 0) {
                borElem.setAttribute('style', 'padding:13px 30px');
                borElem.textContent = '关注列表为空';
                return;
            }

            watchItems.sort(function(x, y) {
                const xNew = x.lastVisitedFloor != x.maxFloor;
                const yNew = y.lastVisitedFloor != y.maxFloor;
                if (xNew !== yNew) {
                    return xNew ? -1 : 1;
                }
                return x.timeAdded - x.timeAdded;
            });

            const tableElem = addElem(borElem, 'table', null, {
                width: '800',
                cellspacing: '0',
                cellpadding: '0',
                style: 'table-layout:fixed'
            });

            const colgroup = addElem(tableElem, 'colgroup');
            addElem(colgroup, 'col', null, { style: 'width: 100px' });
            addElem(colgroup, 'col');
            addElem(colgroup, 'col', null, { style: 'width: 100px' });
            addElem(colgroup, 'col', null, { style: 'width: 120px' });
            addElem(colgroup, 'col', null, { style: 'width: 120px' });
            addElem(colgroup, 'col', null, { style: 'width: 50px' });
            addElem(colgroup, 'col', null, { style: 'width: 20px' }); // allow for scrollbar
            const tbodyElem = addElem(tableElem, 'tbody');

            const trElem1 = addElem(tbodyElem, 'tr');

            const thElem1_1 = addElem(trElem1, 'th', 'h', {
                colspan: '6'
            });
            addElem(trElem1, 'th', 'h');

            const frElem1 = addElem(thElem1_1, 'span', 'fr', {
                style: 'margin-top:2px;cursor:pointer'
            });
            frElem1.addEventListener('click', function() {
                closePopupMenu(WATCHER_POPUP_MENU_ID);
            });

            addElem(frElem1, 'img', null, {
                src: 'images/close.gif'
            });
            thElem1_1.appendChild(document.createTextNode('关注列表'));

            const trElemAct = addElem(tbodyElem, 'tr', 'tr2 tac');
            const actCell = addElem(trElemAct, 'td', 'rinsp-action-bar', { colspan: '6' });
            addElem(trElemAct, 'td');
            const checkNowButton = addElem(actCell, 'span', 'rinsp-check-now');
            checkNowButton.textContent = '🔃立即检查';
            checkNowButton.addEventListener('click', function() {
                closePopupMenu(WATCHER_POPUP_MENU_ID);
                watcherApi.tryCheck(true);
            });
            let maxReachedWarnElem = null;
            if (watchItems.length >= num(MAX_WATCH_ITEM_COUNT)) {
                maxReachedWarnElem = addElem(actCell, 'span', 'rinsp-limit-warning').textContent = '关注数量已达上限';
            }

            const trElem2 = addElem(tbodyElem, 'tr', 'tr2 tac');
            addElem(trElem2, 'td').textContent = '版块名称';
            addElem(trElem2, 'td').textContent = '标题';
            addElem(trElem2, 'td').textContent = '悬赏剩时';
            addElem(trElem2, 'td').textContent = '最近回复';
            addElem(trElem2, 'td').textContent = '最近检查';
            addElem(trElem2, 'td').textContent = '删';
            addElem(trElem2, 'td');

            const now = Date.now();
            watchItems.forEach(function(watchItem) {
                const trElem3 = addElem(tbodyElem, 'tr', 'tr3 tac');
                addElem(trElem3, 'td').textContent = watchItem.areaName;
                const titleCell = addElem(trElem3, 'td', 'tal');
                let postUrl = `read.php?tid-${watchItem.id}.html`;
                if (watchItem.lastVisitedPage > 1) {
                    postUrl = `read.php?tid-${watchItem.id}-fpage-0-toread--page-${watchItem.lastVisitedPage}.html`;
                }
                addElem(titleCell, 'a', null, {
                    href: postUrl,
                    target: '_blank'
                }).textContent = watchItem.postTitle;
                const bountyCell = addElem(trElem3, 'td', 'rinsp-bounty-cell');
                const expired = isWatchExpired(watchItem);
                if (expired) {
                    trElem3.classList.add('rinsp-expired');
                    bountyCell.textContent = '关注已过期';
                } else if (watchItem.areaName === REQUEST_ZONE_NAME) {
                    if (watchItem.bountyUntil === -1) {
                        bountyCell.classList.add('rinsp-bounty-answered');
                        bountyCell.textContent = '已有答案';
                    } else {
                        const hourLeft = ((watchItem.bountyUntil||0) - now) / 3600000;
                        if (hourLeft <= 0) {
                            bountyCell.classList.add('rinsp-bounty-ended');
                            if (watchItem.bountyUntil != null) {
                                const ageString = getAgeString((now - watchItem.bountyUntil) / 60000);
                                bountyCell.textContent = `已过期 (${ageString})`;
                            } else {
                                bountyCell.textContent = '已过期';
                            }
                        } else {
                            bountyCell.textContent = `${Math.floor(hourLeft.toFixed(0))}小时`;
                        }
                    }

                }
                const lastReplyCell = addElem(trElem3, 'td', 'rinsp-lastreply-cell');
                const floorElem = newElem('span', 'rinsp-lastreply-floor');
                if (watchItem.lastPostUid === myUserId) {
                    floorElem.textContent = MY_NAME_DISPLAY;
                    lastReplyCell.classList.add('rinsp-reply-self');
                } else if (watchItem.maxFloor > 0) {
                    if (watchItem.maxFloorApprox) {
                        floorElem.textContent = `${watchItem.maxFloor}楼+`;
                    } else {
                        floorElem.textContent = `${watchItem.maxFloor}楼`;
                    }
                } else {
                    floorElem.textContent = '无回复';
                }
                if (watchItem.lastPostTime > 0) {
                    lastReplyCell.appendChild(document.createTextNode(`${getAgeString((now - watchItem.lastPostTime) / 60000)} / `));
                }
                lastReplyCell.appendChild(floorElem);
                
                const statusCell = addElem(trElem3, 'td', 'rinsp-status-cell');
                statusCell.textContent = getStatusText(watchItem);
                if (expired) {
                    statusCell.setAttribute('title', '下次检查: 已关闭');
                } else {
                    const checkMin = getScaledValue(watchItem, num(MIN_CHECK_INTERVAL), num(MAX_CHECK_INTERVAL));
                    const checkOneMin = getScaledValue(watchItem, num(MIN_CHECK_INTERVAL_OLDEST_ITEM), num(MAX_CHECK_INTERVAL_OLDEST_ITEM));
                    if (pendingWatchItemChecks.has(watchItem.id)) {
                        trElem3.classList.add('rinsp-checking');
                        statusCell.setAttribute('title', '正在检查');
                    } else {
                        if (checkMin >= 60) {
                            if (checkOneMin >= 60) {
                                statusCell.setAttribute('title', '下次检查: ' + formatRangeText((checkOneMin / 60).toFixed(0), (checkMin / 60).toFixed(0)) + ' 小时');
                            } else {
                                statusCell.setAttribute('title', '下次检查: ' + checkOneMin.toFixed(0) + ' 分钟 - ' + (checkMin / 60).toFixed(0) + ' 小时');
                            }
                        } else {
                            statusCell.setAttribute('title', '下次检查: ' + formatRangeText(checkOneMin.toFixed(0), checkMin.toFixed(0)) + ' 分钟');
                        }
                        statusCell.addEventListener('click', function() {
                            watcherApi.tryCheck(watchItem.id);
                        });
                    }
        
                }
                const delCell = addElem(trElem3, 'td');
                const delButton = addElem(delCell, 'img', null, {
                    src: 'images/post/c_editor/del.gif',
                    style: 'cursor:pointer'
                });
                delButton.addEventListener('click', function() {
                    if (watchItem.lastError == null && !expired) {
                        if (!confirm('从关注列表中移除?以后不会收到更多回复通知')) {
                            return;
                        }
                    }
                    showNativeSpinner();
                    updateWatchList(watchList => {
                        const postIdKey = '#' + watchItem.id;
                        if (!Object_hasOwn(watchList, postIdKey)) {
                            return null;
                        }
                        delete watchList[postIdKey];
                        return watchList;
                    })
                    .then(function() {
                        trElem3.remove();
                        if (maxReachedWarnElem != null) {
                            maxReachedWarnElem.remove();
                            maxReachedWarnElem = null;
                        }
                        if (tableElem.querySelectorAll('tr.tr3.tac').length === 0) {
                            closePopupMenu(WATCHER_POPUP_MENU_ID);
                        }
                    })
                    .finally(() => closeNativeSpinner());
                });
                addElem(trElem3, 'td');
                const status = getWatchItemStatus(watchItem);
                if (status != null) {
                    trElem3.classList.add('rinsp-' + status);
                }

            });
        });

        function formatRangeText(minText, maxText) {
            if (minText === maxText) {
                return minText;
            } else {
                return minText + ' - ' + maxText;
            }
        }
    }

    function initCurrentPost() {
        const currentPostLink = document.querySelector('#breadcrumbs .crumbs-item.current strong > a[href^="read.php?tid-"]');
        if (currentPostLink == null) {
            return;
        }

        const areaLink = currentPostLink.closest('.crumbs-item').previousElementSibling;
        const fid = Number.parseInt(areaLink.getAttribute('href').split('?fid-', 2)[1]);
        const areaName = areaLink.textContent.trim();
        const postTitle = currentPostLink.textContent.trim();
        const postId = Number.parseInt(currentPostLink.getAttribute('href').substring(13));
        if (Number.isNaN(postId)) {
            return;
        }
        const postIdKey = '#' + postId;

        const pageArr = readCurrentAndMaxPage(document);
        const currentPage = pageArr[0];
        let maxPage = pageArr[1];

        if (userConfig.showFloatingStoreThreadButton) {
            addFloatingFavorButton(topControlContainer, postId, favorThreadsCacheAccess);
        }
        if (userConfig.showFloatingWatchIndicator) {
            const watchButton = addElem(topControlContainer, 'a', 'rinsp-excontrol-item rinsp-excontrol-item-watch rinsp-excontrol-item-ticker');
            watchButton.textContent = '🔥\n关\n注';
            watchButton.addEventListener('click', () => {
                if (watcherApi.currentThread.isWatching()) {
                    showNativeSpinner();
                    removeFromWatchList()
                        .finally(() => closeNativeSpinner());
                } else {
                    showNativeSpinner();
                    addToWatchList()
                        .finally(() => closeNativeSpinner());
                }
            });
            
            watcherApi.currentThread.register({
                setWatching(watching) {
                    if (watching) {
                        watchButton.classList.add('rinsp-active');
                    } else {
                        watchButton.classList.remove('rinsp-active');
                    }
                },
                setBusy(busy) {
                    if (busy) {
                        watchButton.classList.add('rinsp-running');
                    } else {
                        watchButton.classList.remove('rinsp-running');
                    }
                }
            });
        }

        function initIfFirstPage() {
            const shouElem = document.querySelector('.tr1.r_one .tipad .readbot .shou');
            if (shouElem == null) {
                return;
            }

            const watchElem = document.createElement('li');
            watchElem.classList.add('rinsp-watch');
            watchElem.textContent = '关注: ';
            shouElem.parentNode.appendChild(watchElem);

            watcherApi.currentThread.register({
                setWatching(watching) {
                    if (watching) {
                        watchElem.classList.add('rinsp-active');
                    } else {
                        watchElem.classList.remove('rinsp-active');
                    }
                },
                setBusy(busy) {
                    if (busy) {
                        watchElem.classList.add('rinsp-running');
                    } else {
                        watchElem.classList.remove('rinsp-running');
                    }
                }
            });
            watchElem.addEventListener('click', function() {
                if (watcherApi.currentThread.isWatching()) {
                    showNativeSpinner();
                    removeFromWatchList()
                        .finally(() => closeNativeSpinner());
                } else {
                    showNativeSpinner();
                    addToWatchList()
                        .finally(() => closeNativeSpinner());
                }
            });

        }

        async function removeFromWatchList() {
            watcherApi.currentThread.setBusy(true);
            await updateWatchList(function(watchList) {
                if (!Object_hasOwn(watchList, postIdKey)) {
                    return null;
                }
                delete watchList[postIdKey];
                return watchList;
            });
            watcherApi.currentThread.setWatching(false);
            watcherApi.currentThread.setBusy(false);
        }

        async function addToWatchList() {
            watcherApi.currentThread.setBusy(true);
            let fp = null;
            if (currentPage === 1) {
                const onlyGfLink = document.querySelector('#td_tpc a[href^="read.php?tid-' + postId + '-uid-"]');
                if (onlyGfLink != null) {
                    fp = document;
                }
            }
            if (fp == null) {
                fp = await fetchGetPage(`${document.location.origin}/read.php?tid-${postId}.html`);
            }
            
            const onlyGfLink = fp.querySelector('#td_tpc a[href^="read.php?tid-' + postId + '-uid-"]');
            if (onlyGfLink == null) {
                return;
            }
            const gfUid = Number.parseInt(onlyGfLink.getAttribute('href').replace(/.*-uid-/, ''));
            const opDate = new Date(fp.querySelector('#td_tpc > .tiptop > .fl.gray[title^="发表于:"]').textContent.trim()) * 1;

            let bountyUntil = null;
            if (fid === QUESTION_AND_REQUEST_AREA_ID) {
                const bountyStatus = getBountyStatus(fp);
                bountyUntil = bountyStatus ? bountyStatus.bountyUntil : null;
            }

            const lastPost = await getLastPost();
            let added = false;
            await updateWatchList(watchList => {
                if (Object.keys(watchList).length >= num(MAX_WATCH_ITEM_COUNT)) {
                    return null;
                }
                watchList[postIdKey] = {
                    id: postId,
                    owner: gfUid,
                    areaName: areaName,
                    postTitle: postTitle,
                    maxPage: lastPost.page,
                    maxFloor: lastPost.floor,
                    maxFloorApprox: false,
                    lastVisitedPage: lastPost.page,
                    lastVisitedFloor: lastPost.floor,
                    lastVisitedPostId: lastPost.postId,
                    lastPostId: lastPost.postId,
                    lastPostUid: lastPost.postUid,
                    lastPostTime: lastPost.postTime,
                    lastUpdateIgnorable: false,
                    lastChecked: Date.now(),
                    skipUntil: 0,
                    lastError: null,
                    bountyUntil: bountyUntil,
                    timeOpened: opDate,
                    timeAdded: Date.now()
                };
                added = true;
                return watchList;
            });
            watcherApi.currentThread.setBusy(false);
            watcherApi.currentThread.setWatching(added);
            if (!added) {
                showWatchMenu();
            }
        }
        if (currentPage === 1) {
            initIfFirstPage();
        }

        async function update(alsoUpdateSavedData) {
            const watchList = await readWatchList();
            const watchItem = watchList[postIdKey];
            watcherApi.currentThread.setWatching(watchItem != null);
            if (watchItem != null && alsoUpdateSavedData) {
                if (DEBUG_MODE) console.info('mark read', postId);
                // only use the current page if applicable to update the current watch item
                let bountyUntil = null;
                if (watchItem.bountyUntil > 0) {
                    if (currentPage === 1) {
                        const bountyStatus = getBountyStatus(document);
                        if (bountyStatus != null) {
                            bountyUntil = bountyStatus.bountyUntil;
                        }
                    }
                }
                if (currentPage === maxPage) {
                    const lastPost = await getLastPost();
                    watchItem.postTitle = postTitle;
                    watchItem.maxPage = maxPage;
                    watchItem.maxFloor = lastPost.floor;
                    watchItem.maxFloorApprox = false;
                    watchItem.lastVisitedPage = lastPost.page;
                    watchItem.lastVisitedFloor = lastPost.floor;
                    watchItem.lastVisitedPostId = lastPost.postId;
                    watchItem.lastPostId = lastPost.postId;
                    watchItem.lastPostUid = lastPost.postUid;
                    watchItem.lastPostTime = lastPost.postTime;
                    watchItem.lastUpdateIgnorable = false;
                    watchItem.lastChecked = Date.now();
                    watchItem.skipUntil = 0;
                    watchItem.lastError = null;
                    if (bountyUntil != null) {
                        watchItem.bountyUntil = bountyUntil;
                    }
                    await writeWatchList(watchList);
                } else if (bountyUntil != null && watchItem.bountyUntil != bountyUntil) {
                    watchItem.bountyUntil = bountyUntil;
                    await writeWatchList(watchList);
                }
            }
        }

        update(true);

        async function getLastPost() {
            //console.info('getLastPost', currentPage, maxPage);
            if (currentPage === maxPage) {
                return findLastPost(document, null, null); // ignore filters not needed
            } else {
                return fetchLastPost(postId);
            }
        }

    }

    async function fetchCheckBountyEnded(postId) {
        const doc = await fetchGetPage(`${document.location.origin}/read.php?tid-${postId}.html`);
        const bountyStatus = getBountyStatus(doc);
        return bountyStatus == null ? null : bountyStatus.ended === 2;
    }

    async function fetchLastPost(postId, ignoreList) {
        const url = `${document.location.origin}/read.php?tid-${postId}-page-e-fpage-1.html`;
        const doc = await fetchGetPage(url);
        const lastPost = findLastPost(doc, userFilter, ignoreList);
        if (DEBUG_MODE) console.info('fetchLastPost', url, lastPost);
        if (lastPost == null) {
            return {
                error: 'PAGE_GONE',
                errorMessage: findErrorMessage(doc)
            };
        } else {
            return lastPost;
        }
    }

    function getStatusText(watchItem) {
        //console.info('getStatusText', watchItem);
        if (watchItem.lastError) {
            return watchItem.lastError;
        }
        const unreadFloor = watchItem.maxFloor - watchItem.lastVisitedFloor;
        if (unreadFloor === 0) {
            const ageMins = (Date.now() - watchItem.lastChecked) / 60000;
            let ageString = getAgeString(ageMins);
            return `${ageString}`;
        }
        if (unreadFloor < 0) {
            return '有新回复';
        }
        if (watchItem.lastUpdateIgnorable) {
            return '屏蔽 ' + unreadFloor + ' 回复';
        } else {
            return '有 ' + unreadFloor + ' 新回复';
        }
    }

    function findLastPost(doc, userFilter, ignoreList) {

        const posts = getPosts(doc);
        if (posts.length === 0) {
            return null;
        }
        const postsByFloor = Object.create(null);
        let highestFloor = 0;
        posts.forEach(post => {
            if (post.floor > highestFloor) {
                highestFloor = post.floor;
            }
            postsByFloor[post.floor] = {
                floor: post.floor,
                postId: post.postId,
                postUid: post.postUid,
                postUname: post.postUname,
                postTime: post.postTime,
                ignorable: false
            };
        });
        const lastPost = postsByFloor[highestFloor];
        const pageArr = readCurrentAndMaxPage(doc);

        if (ignoreList != null) {
            const ignoreContentMatcher = createIgnoreContentMatcher(ignoreList);
            posts.forEach(post => {
                if (post.floor > 0) {
                    const [ignoreState] = annotateIgnorablePost(post, myUserId, userConfig, userFilter, ignoreContentMatcher);
                    if (ignoreState) {
                        postsByFloor[post.floor].ignorable = true;
                    }
                }
            });
        }

        let bountyEnded = null;
        if (pageArr[0] === 1) {
            const bountyStatus = getBountyStatus(doc);
            bountyEnded = bountyStatus == null ? null : bountyStatus.ended === 2;
        }

        return {
            page: pageArr[0],
            maxPage: pageArr[1],
            floor: lastPost.floor,
            postsByFloor: postsByFloor,
            postId: lastPost.postId,
            postUid: lastPost.postUid,
            postTime: lastPost.postTime,
            bountyEnded: bountyEnded
        };
    }

    async function readWatchList() {
        try {
            let sSavedData = await GM.getValue('my_watchlist#' + myUserId);
            let watchList = JSON.parse(sSavedData);
            Object.values(watchList).forEach(watchItem => {
                if (watchItem.timeOpened == null) {
                    watchItem.timeOpened = Date.now(); // backward-compatibility
                }
            });
            return watchList;
        } catch (e) {}
        return {};
    }

    async function writeWatchList(watchList) {
        await GM.setValue('my_watchlist#' + myUserId, JSON.stringify(watchList));
        return true;
    }

    async function updateWatchList(updater) {
        async function execute() {
            const watchList = await readWatchList();
            const updatedWatchList = await updater(watchList);
            if (updatedWatchList != null) {
                await writeWatchList(updatedWatchList);
            }
        }
        await runWithLock('watchlist-update', 1500, execute);
    }

    async function testRecordBestAnswerNotification(userConfig, threadHistoryAccess) {
        const canditateThreadLink = document.querySelector('#info_base .set-table2 > tbody > tr > td > b + a[href^="//"][href$=".html"]');
        if (canditateThreadLink == null) {
            return;
        }
        const tid = (canditateThreadLink.getAttribute('href').match(/.*\/read\.php\?tid-(\d+)\.html$/)||[])[1] * 1;
        if (Number.isNaN(tid)) {
            return;
        }
        if (!canditateThreadLink.closest('td').textContent.startsWith('您的回复被设为最佳答案!')) {
            return;
        }
        const lastAccessRecord = await threadHistoryAccess.recentAccessStore.get(tid);
        if (lastAccessRecord && lastAccessRecord.bounty && lastAccessRecord.bounty.winner == null) {
            lastAccessRecord.bounty.winner = userConfig.myUserHashId;
            lastAccessRecord.bounty.ended = 2;
            await threadHistoryAccess.recentAccessStore.put(tid, lastAccessRecord);
            if (DEBUG_MODE) {
                console.info(`bounty winner recorded (tid=${tid})`);
            }
        }
    }

    function enhanceMessageReadDisplay(userConfig) {
        const userLink = document.querySelector('#info_base .set-table2 > tbody > tr:first-child > td > a[href^="u.php?action-show-uid-"]');
        if (userLink == null) {
            return;
        }
        const userId = Number.parseInt(userLink.getAttribute('href').substring(22));
        if (userId === 0) {
            userLink.textContent = SYSTEM_SENDER_DISPLAY_NAME;
            userLink.closest('table').classList.add('rinsp-system-message-view');
        } else {
            const userRecord = userConfig.customUserHashIdMappings['#' + userId];
            if (userRecord != null) {
                userLink.closest('td').classList.add('rinsp-message-user-mapped');
                const currentName = userLink.textContent;
                userLink.textContent = userRecord[2];
                userLink.setAttribute('title', '现在昵称: ' + currentName);
            }
        }

    }

    function enhanceMessagingList(userConfig, userPinArea, inboxMode) {
        document.querySelectorAll('#set-content .set-tab-box #info_base .set-table2').forEach(list => {

            let messages = [];

            const selectionPanel = newElem('div', 'rinsp-message-selection-panel');
            list.parentNode.insertBefore(selectionPanel, list);

            if (inboxMode) {
                const selectSysMsgButton = addElem(selectionPanel, 'div', 'rinsp-selection-button');
                selectSysMsgButton.textContent = '系统信息';
                selectSysMsgButton.addEventListener('click', () => {
                    messages.filter(message=>!!message.checkbox).forEach(m => m.checkbox.checked = m.userId === 0 || SYSTEM_MESSAGE_TITLES.has(m.title));
                });
    
                addElem(selectionPanel, 'div', 'rinsp-selection-sep').textContent = '|';
            }

            const selectAllButton = addElem(selectionPanel, 'div', 'rinsp-selection-button');
            selectAllButton.textContent = '全选';
            selectAllButton.addEventListener('click', () => {
                messages.filter(message=>!!message.checkbox).forEach(m => m.checkbox.checked = true);
            });

            const deselectAllButton = addElem(selectionPanel, 'div', 'rinsp-selection-button');
            deselectAllButton.textContent = '清除';
            deselectAllButton.addEventListener('click', () => {
                messages.filter(message=>!!message.checkbox).forEach(m => m.checkbox.checked = false);
            });


            const observer = createMutationObserver(async () => {
                apply();
            });
            observer.init(list, { childList: true, subtree: false, attributes: false });
            observer.trigger();
            userPinArea.addUpdateListener(() => {
                observer.trigger();
            });
        
            function apply() {
                messages.length = 0;
                userPinArea.clearState();
                list.querySelectorAll('tbody > tr > td > a[href^="u.php?action-show-uid-"]').forEach(userLink => {
                    const row = userLink.closest('tr');
                    const messageLink = row.querySelector('a[href^="message.php?action-read-mid-"],a[href^="message.php?action-readscout-mid-"],a[href^="message.php?action-readsnd-mid-"]');
                    if (messageLink == null) {
                        return;
                    }
                    const checkbox = row.querySelector('td > input[name="delid[]"]');
                    const userId = Number.parseInt(userLink.getAttribute('href').substring(22));
                    if (userId === 0) {
                        userLink.textContent = SYSTEM_SENDER_DISPLAY_NAME;
                        row.classList.add('rinsp-system-message-row');
                    } else {
                        const userCell = userLink.closest('td');
                        userCell.classList.remove('rinsp-message-user-pinned');
                        userCell.classList.remove('rinsp-message-user-mapped');
                        let pinnedUser = userPinArea.getPinned(userId);
                        if (pinnedUser) {
                            userCell.classList.add('rinsp-message-user-pinned');
                            userPinArea.addLocation(userId, '⚪', null, () => {
                                userLink.closest('tr').scrollIntoView({
                                    behavior: 'smooth',
                                    block: 'center'
                                });
                            });
                        }
                        let userRecord = userConfig.customUserHashIdMappings['#' + userId];
                        if (userRecord != null) {
                            if (!pinnedUser) {
                                userCell.classList.add('rinsp-message-user-mapped');
                            }
                            const currentName = userLink.textContent;
                            userLink.textContent = userRecord[2];
                            userLink.setAttribute('title', '现在昵称: ' + currentName);
                        }

                    }
                    const title = messageLink.textContent.trim();
                    messages.push({
                        row,
                        checkbox,
                        userLink,
                        userId,
                        title
                    });
                });
                selectionPanel.hidden = messages.length === 0;
                userPinArea.update();
            }


        }); 
    }

    function enhanceMessagingForm() {
        const form = document.querySelector('#main form[name="FORM"][action="message.php"]');
        if (form == null) {
            return null;
        }

        const touidmsg = form.querySelector('input[name="touidmsg"]');
        const touid = touidmsg.value * 1;
        if (touid > 0) {
            const nicknameCell = touidmsg.closest('tr').nextElementSibling.querySelector('td.td1 + td');
            const homeButton = newElem('a', 'abtn', { href: `u.php?action-show-uid-${touid}.html`, target: '_blank' });
            homeButton.textContent = '用户资料';
            nicknameCell.appendChild(homeButton);

            nicknameCell.appendChild(document.createTextNode(' '));

            const historyButton = newElem('a', 'abtn', { href: `message.php?action-chatlog-withuid-${touid}.html`, target: '_blank' });
            historyButton.textContent = '通信记录';
            nicknameCell.appendChild(historyButton);

        }

        if (userConfig.showInputLimit) {
            const input = form.querySelector('input[name="msg_title"]');
            if (input) {
                input.value = truncateByByteLength(input.value, 75, '...');
                addLengthLimit(input, 75);
            }
            const textarea = form.querySelector('textarea[name="atc_content"]');
            if (textarea) {
                addLengthLimit(textarea, 1500);
            }
            form.addEventListener('submit', evt => {
                if (form.querySelector('.rinsp-input-max-exceeded')) {
                    evt.preventDefault();
                    return false;
                }
            });
        }
    }

    function addLengthLimit(input, byteLimit, charLimit, warnOnly, skipRe) {
        const hint = newElem('span', 'rinsp-input-max-hint');
        input.after(hint);
        input.addEventListener('input', () => update());
        input.addEventListener('change', () => update());
        function update() {
            hint.classList.remove('rinsp-input-max-exceeded');
            input.classList.remove('rinsp-input-max-exceeded');
            hint.textContent = '';
            let valueTrim = input.value.trim();
            if (skipRe && valueTrim.startsWith('Re:')) {
                valueTrim = valueTrim.substring(3);
            }
            if (charLimit > 0) {
                if (valueTrim.length > charLimit) {
                    hint.textContent =  `${valueTrim.length} (过长) / ${charLimit}`;
                    hint.classList.add('rinsp-input-max-exceeded');
                    input.classList.add('rinsp-input-max-exceeded');
                    return;
                } else if (!warnOnly) {
                    hint.textContent =  `${valueTrim.length} / ${charLimit}`;
                }
            }
            if (byteLimit > 0) {
                const curLength = getByteLength(valueTrim);
                if (curLength > byteLimit) {
                    hint.textContent =  `${curLength} (过长) / ${byteLimit}字节`;
                    hint.classList.add('rinsp-input-max-exceeded');
                    input.classList.add('rinsp-input-max-exceeded');
                    return;
                } else if (!warnOnly) {
                    if (charLimit === 0) {
                        hint.textContent =  `${curLength} / ${byteLimit}字节`;
                    }
                }
            }
        }
        update();
    }

    function changeToImageWallThreadLinks() {
        document.querySelectorAll('ul.threadlist li a[href^="thread.php?"]').forEach(link => {
            link.setAttribute('href', 'thread_new' + link.getAttribute('href').substring(6));
        });
    }

    async function addPicWallDefaultOption(fid) {
        const toggleButton = document.querySelector('#breadcrumbs > .fr > a[href^="thread"]');
        if (toggleButton == null || !['[点击进入图墙模式]', '[点击进入列表模式]'].includes(toggleButton.textContent.trim())) {
            return;
        }
        const toggle = newElem('span', 'fr');
        toggleButton.parentElement.parentElement.insertBefore(toggle, toggleButton.parentElement.nextElementSibling);

        const labelElem = addElem(toggle, 'label');
        const checkbox = addElem(labelElem, 'input', null, { type: 'checkbox' });
        checkbox.checked = readPinnedPicWallFids().includes(fid);
        addElem(labelElem, 'span', null, { style: 'vertical-align: middle; margin-right: 0.5em' }).textContent = '默认图墙模式';
        checkbox.addEventListener('change', () => {
            const fids = readPinnedPicWallFids().filter(x=>x!==fid);
            if (checkbox.checked) {
                fids.push(fid);
            }
            localStorage.setItem(PIC_WALL_PREF_KEY, fids.join(' '));
            update();
        });
        update();

        function update() {
            if (checkbox.checked && toggleButton.getAttribute('href').startsWith('thread.php?')) {
                toggleButton.parentElement.setAttribute('style', 'opacity: 0.1; pointer-events:none');
            } else {
                toggleButton.parentElement.setAttribute('style', '');
            }
        }

    }

    async function enableForumAnnouncementFolding() {
        const anchor = document.querySelector('td.tac > img[src="images/colorImagination/thread/anc.gif"]');
        if (anchor == null) {
            return;
        }
        const th = anchor.parentNode.nextElementSibling;
        if (th == null || !th.textContent.trim().startsWith('论坛公告:')) {
            return;
        }
        const row = anchor.closest('tr.tr3');
        row.classList.add('rinsp-top-announcement-row');
        let announcementRows = [];
        const normalTopicMarker = document.querySelector('.rinsp-top-announcement-row ~ tr.tr2 > .tac');
        if (normalTopicMarker && normalTopicMarker.textContent.trim() === '普通主题') {
            const endMarkerRow = normalTopicMarker.closest('.tr2');
            let testRow = row;
            while ((testRow = testRow.nextElementSibling) !== endMarkerRow) {
                announcementRows.push(testRow);
            }
        } else {
            let testRow = row;
            while ((testRow = testRow.nextElementSibling) != null) {
                if (testRow.querySelector('td + td > img[src="images/colorImagination/file/headtopic_3.gif"]') == null) {
                    break;
                }
                announcementRows.push(testRow);
            }
            
        }
        if (announcementRows.length === 0) {
            return;
        }
        const toggler = newElem('a', 'rinsp-announcement-toggler');
        const img = newElem('img');
        toggler.appendChild(img);
        th.insertBefore(toggler, th.childNodes[0]);

        function updateFoldingState(folded) {
            if (folded) {
                row.classList.add('rinsp-announcement-folded');
                announcementRows.forEach(r => r.classList.add('rinsp-announcement-row-folded'));
                img.setAttribute('src', 'images/colorImagination/index/cate_open.gif');
            } else {
                row.classList.remove('rinsp-announcement-folded');
                announcementRows.forEach(r => r.classList.remove('rinsp-announcement-row-folded'));
                img.setAttribute('src', 'images/colorImagination/index/cate_fold.gif');
            }
        }
        updateFoldingState(userConfig.siteAnnouncementSectionDefaultFolded);
        toggler.addEventListener('click', async () => {
            const newFoldState = !row.classList.contains('rinsp-announcement-folded');
            updateFoldingState(newFoldState);
            await mainConfigAccess.update(function(updatingUserConfig) {
                updatingUserConfig.siteAnnouncementSectionDefaultFolded = newFoldState;
                return updatingUserConfig;
            });
        });
    }

    async function enhanceSearchPage() {
        const searchForm = document.querySelector('#main form[action="search.php?"]');
        if (searchForm == null) {
            return;
        }

        const searchPref = await searchConfigAccess.read();
        const keywordInput = searchForm.querySelector('input[name="keyword"]');
        if (keywordInput != null) {
            keywordInput.classList.add('rinsp-search-keyword-input');
            enhanceKeywordInput(keywordInput, searchPref);
            setTimeout(function () {
                keywordInput.focus();
            }, 0);
        }
    
        const dateRangeSelect = searchForm.querySelector('select[name="sch_time"]');
        if (dateRangeSelect != null) {
            enhanceSearchRangeSelect(dateRangeSelect, searchPref);
        }

        const topicSelect = searchForm.querySelector('select[name="f_fid"]');
        if (topicSelect != null) {
            enhanceSearchTopicSelect(topicSelect, searchPref);
        }
        const prefillParamString = (document.location.hash.match(/^#prefill\((.*)\)$/)||[])[1];
        if (prefillParamString) {
            const prefillData = {};
            try {
                const array = JSON.parse(`[${decodeURIComponent(prefillParamString)}]`);
                for (let i = 0; i < array.length - 1; i += 2) {
                    prefillData[array[i]] = array[i + 1];
                }
            } catch (ignore) {}

            if (prefillData.fid > 0) {
                topicSelect.value = String(prefillData.fid);
            }
            if (prefillData.keyword) {
                keywordInput.value = prefillData.keyword;
            }
            if (prefillData.pwuser) {
                searchForm.querySelector('input[name="pwuser"]').value = prefillData.pwuser;
            }
        }
    }

    async function enhanceKeywordInput(keywordInput, searchPref) {
        let defaultSearchAll = !!searchPref.defaultSearchAll;
    
        const parentCell = keywordInput.closest('th');
        const setDefaultButton = document.createElement('span');
        setDefaultButton.classList.add('rinsp-fav-search-setdefault-mode');
        setDefaultButton.classList.add('rinsp-hide');
        setDefaultButton.textContent = '➕设为默认';
        setDefaultButton.addEventListener('click', async function () {
            await searchConfigAccess.update(function(updatingSearchPref) {
                updatingSearchPref.defaultSearchAll = searchAllRadioOption.checked;
                return updatingSearchPref;
            });
            defaultSearchAll = searchAllRadioOption.checked;
            redraw();
        });
        parentCell.appendChild(setDefaultButton);
        const searchAllRadioOption = parentCell.querySelector('input[name="method"][value="AND"]');
        parentCell.querySelectorAll('input[name="method"]').forEach(function (methodRadioOption) {
            methodRadioOption.addEventListener('change', function () {
                redraw();
            });
        });
        function redraw() {
            if (defaultSearchAll === searchAllRadioOption.checked) {
                setDefaultButton.classList.add('rinsp-hide');
            } else {
                setDefaultButton.classList.remove('rinsp-hide');
            }
        }
        if (defaultSearchAll) {
            searchAllRadioOption.checked = true;
        }
        setTimeout(function () {
            redraw();
        }, 0); // account for browser auto-fill
    }
    
    async function enhanceSearchRangeSelect(rangeSelect, searchPref) {
        const parentCell = rangeSelect.closest('th');
        const setDefaultButton = addElem(parentCell, 'span', 'rinsp-fav-search-setdefault-range rinsp-hide');
        setDefaultButton.textContent = '➕设为默认';
        let currentDefaultRange = searchPref.defaultTimeRange||'all';
        function update() {
            if (rangeSelect.value === currentDefaultRange) {
                setDefaultButton.classList.add('rinsp-hide');
            } else {
                setDefaultButton.classList.remove('rinsp-hide');
            }
        }
        setDefaultButton.addEventListener('click', function () {
            parentCell.classList.add('rinsp-temp-disabled');
            async function execute() {
                await searchConfigAccess.update(function(updatingSearchPref) {
                    updatingSearchPref.defaultTimeRange = rangeSelect.value;
                    currentDefaultRange = rangeSelect.value;
                    return updatingSearchPref;
                });
            }
            execute().finally(function () {
                parentCell.classList.remove('rinsp-temp-disabled');
                update();
            });
        });
        rangeSelect.addEventListener('change', function () {
            update();
        });
        rangeSelect.value = currentDefaultRange;
    }

    async function enhanceSearchTopicSelect(topicSelect, searchPref) {
        const parentCell = topicSelect.closest('th');
        const pin = addElem(parentCell, 'span', 'rinsp-fav-search-area-pin');
        const originalContent = topicSelect.innerHTML;

        await update(searchPref.pinnedTopics || {});

        async function update(topicMappings) {
            let entries = Object.entries(topicMappings);

            let favOptGroup = null;
            if (entries.length > 0) {
                favOptGroup = newElem('optgroup', 'rinsp-fav-search-area-group', { label: '版块捷径' });
                entries.forEach(function (entry) {
                    const favOption = addElem(favOptGroup, 'option', null, {
                        value: entry[0].substring(1)
                    });
                    favOption.textContent = '🔖' + entry[1];
                });
            }
            const fullOptGroup = newElem('optgroup', null, { label: '目录树' });
            topicSelect.querySelectorAll('option').forEach(function (option, i) {
                if (i > 0) {
                    fullOptGroup.appendChild(option);
                } else {
                    option.classList.add('rinsp-fav-search-area-all');
                }
            });
            if (favOptGroup != null) {
                topicSelect.appendChild(favOptGroup);
            }
            topicSelect.appendChild(fullOptGroup);
        }

        pin.addEventListener('click', function () {
            parentCell.classList.add('rinsp-temp-disabled');
            async function execute() {
                const current = getCurrentItem();
                let newSearchPref = await searchConfigAccess.update(function(updatingSearchPref) {
                    const updatingMappings = updatingSearchPref.pinnedTopics || {};
                    if (current.fav) {
                        delete updatingMappings['$' + current.value];
                    } else {
                        let chosenLabel = prompt('显示名称', current.label.trim().replace(/^.*? /, ''));
                        chosenLabel = chosenLabel ? chosenLabel.trim() : '';
                        if (!chosenLabel) {
                            return;
                        }
                        updatingMappings['$' + current.value] = chosenLabel;
                    }
                    updatingSearchPref.pinnedTopics = updatingMappings;
                    return updatingSearchPref;
                });
                if (newSearchPref != null) {
                    topicSelect.innerHTML = originalContent;
                    update(newSearchPref.pinnedTopics||{});
                    topicSelect.value = current.value;
                    updatePin();
                }
            }
            execute().finally(function () {
                parentCell.classList.remove('rinsp-temp-disabled');
            });
        });
        function updatePin() {
            const current = getCurrentItem();
            if (current == null || current.value === 'all') {
                pin.textContent = '';
            } else {
                pin.textContent = current.fav ? '❌删除捷径' : '➕存为捷径';
            }
        }
        function getCurrentItem() {
            let options = topicSelect.querySelectorAll('option[value="' + topicSelect.value + '"]');
            if (options.length === 0) {
                return null;
            }
            return {
                fav: options.length > 1,
                label: options[options.length - 1].textContent,
                value: topicSelect.value
            };
        }
        updatePin();
        topicSelect.addEventListener('change', function () {
            updatePin();
        });
    }

    function annotateIgnorablePost(post, myUserId, userConfig, userFilter, ignoreContentMatcher) {
        post.rootElem.classList.remove('rinsp-filter-default-ignorable');
        post.rootElem.classList.remove('rinsp-filter-ignored-bykeyword');
        post.rootElem.classList.remove('rinsp-filter-ignored-byuid');
        post.rootElem.classList.remove('rinsp-my-post');
        post.rootElem.classList.remove('rinsp-filter-bypass');
        post.rootElem.classList.add('rinsp-filter-added');
        if (post.postUid === myUserId) {
            post.rootElem.classList.add('rinsp-my-post');
            post.rootElem.classList.add('rinsp-filter-bypass');
        }
        let ignoreState = false;

        let userFilterRule;
        if (userConfig.dontFilterRequestReplyByUser && post.areaId === QUESTION_AND_REQUEST_AREA_ID) {
            userFilterRule = null;
        } else {
            userFilterRule = userFilter.users['#' + post.postUid];
        }
        let userBlacklisted = userFilterRule && userFilterRule.hideReplies;
        if (post.postDefaultHidden) {
            ignoreState = true;
        } else if (userBlacklisted) {
            post.rootElem.classList.add('rinsp-filter-ignored-byuid');
            ignoreState = true;
        } else {
            if (userConfig.customUserpicBypassIgnoreList && !post.defaultUserPic) {
                post.rootElem.classList.add('rinsp-filter-bypass');
            } else if (userConfig.customUserBypassIgnoreList) {
                // user blacklist still wins over bookmarked users
                if (userConfig.customUserHashIdMappings['#' + post.postUid] != null) {
                    post.rootElem.classList.add('rinsp-filter-bypass');
                }
            }
        }
        let effectiveContent = null;
        if (ignoreContentMatcher != null) {
            if (post.contentDefaultIgnorable) {
                effectiveContent = DEFAULT_IGNORABLE_MARKER_TAG;
            } else {
                effectiveContent = getEffectiveTextContent(post.contentElem, !!userConfig.ignoreContentUseTextOnly, !!userConfig.treatAllEmojiTheSameWay);
            }
            //if (DEBUG_MODE) console.info(item, 'effectiveContent: `' + effectiveContent + '`');
            if (!ignoreState && effectiveContent != null && ignoreContentMatcher.matches(effectiveContent)) {
                ignoreState = true;
                post.rootElem.classList.add('rinsp-filter-ignored-bykeyword');
            }
        }
    
        if (ignoreState) {
            post.rootElem.classList.add('rinsp-filter-ignored');
        } else {
            post.rootElem.classList.remove('rinsp-filter-ignored');
        }
        return [ignoreState, effectiveContent];
    }

    async function applyContentFilter(allPosts, myUserId, userConfig, threadFilter, userFilter, ignoreList, updateCallback) {
        const posts = allPosts.slice();
        if (posts.length === 0) {
            return;
        }

        function onUpdate(updatedConfigs) {
            // reset annotated state
            posts.forEach(post => {
                post.rootElem.parentNode.removeAttribute('rinsp-filter-ignored-cont');
                post.rootElem.classList.remove('rinsp-filter-added');
            });
            updateCallback(updatedConfigs);
        }
    
        posts.forEach(post => {
            // add block user
            const contentTh = post.contentElem.closest('th');
            contentTh.querySelectorAll('.rinsp-filter-block-menu-item,.rinsp-filter-block-menu-sep').forEach(el => el.remove());
            const blockMenu = contentTh.querySelector('.tiptop .fr > .dropdown > .dropdown-content');
            if (blockMenu) {
                const menuItem = blockMenu.previousElementSibling;
                menuItem.classList.remove('rinsp-filter-block-menu-item-active');
                const dislikeUserRule = userFilter.users['#' + post.postUid];
                const hideThreads = dislikeUserRule && dislikeUserRule.hideThreads;
                const hideReplies = dislikeUserRule && dislikeUserRule.hideReplies;
                
                addElem(blockMenu, 'div', 'rinsp-filter-block-menu-sep');
                const blockThreadButton = addElem(blockMenu, 'a', 'rinsp-filter-block-menu-item');
                if (hideThreads && !hideReplies) {
                    menuItem.classList.add('rinsp-filter-block-menu-item-active');
                    blockThreadButton.classList.add('rinsp-filter-block-menu-item-active');
                    blockThreadButton.textContent = '🚫只主题帖';
                    blockThreadButton.addEventListener('click', () => {
                        promptRemoveUserBlock(post.postUid, post.postUname, onUpdate);
                    });
                } else {
                    blockThreadButton.textContent = '只主题帖';
                    blockThreadButton.addEventListener('click', () => {
                        promptBlockOp(post.postUid, post.postUname, onUpdate);
                    });
                }

                const blockReplyButton = addElem(blockMenu, 'a', 'rinsp-filter-block-menu-item');
                if (hideReplies) {
                    menuItem.classList.add('rinsp-filter-block-menu-item-active');
                    blockReplyButton.classList.add('rinsp-filter-block-menu-item-active');
                    blockReplyButton.textContent = '🚫所有内容';
                    blockReplyButton.addEventListener('click', () => {
                        promptRemoveUserBlock(post.postUid, post.postUname, onUpdate);
                    });
                } else {
                    blockReplyButton.textContent = '所有内容';
                    blockReplyButton.addEventListener('click', () => {
                        promptBlockComplete(post.postUid, post.postUname, onUpdate);
                    });
                }
            }
        });

        // GF: add keyword filter
        if (posts[0].floor === 0) {
            const opPost = posts[0];
            const threadFilterExecutor = initThreadFilterExecutor(myUserId, threadFilter, userFilter, onUpdate, true);
            const subjectElem = opPost.rootElem.querySelector('#subject_tpc');
            const threadLikeModel = {
                tid: opPost.tid,
                row: opPost.rootElem,
                op: opPost.postUid,
                opName: opPost.postUname,
                opElem: null,
                titleCell: subjectElem,
                title: getEffectiveThreadTitle(subjectElem.textContent)
            };
            threadFilterExecutor.clear(threadLikeModel);
            threadFilterExecutor.run(threadLikeModel);
        }

        function ignoreButtonDef() {
            return {
                create(post) {
                    const ignoreElem = document.createElement('li');
                    const actionBar = post.rootElem.querySelector('.tipad .fl.readbot');
                    actionBar.appendChild(ignoreElem);
                    return ignoreElem;
                },
                restoreLabel: '已屏蔽',
                hideLabel: '屏蔽内容'
            };
        }
        function defaultIgnorableButtonDef() {
            return {
                create(post) {
                    const ignoreElem = document.createElement('span');
                    const actionBar = post.rootElem.querySelector('.tiptop > .fr');
                    const insertBeforeNode = actionBar.childNodes[0];
                    actionBar.insertBefore(ignoreElem, insertBeforeNode);
                    return ignoreElem;
                },
                restoreLabel: '已屏蔽',
                hideLabel: '屏蔽内容'
            };
        }
        const ignoreContentMatcher = createIgnoreContentMatcher(ignoreList);
        posts.forEach(post => {
            if (post.floor === 0) {
                annotateIgnorablePost(post, myUserId, userConfig, userFilter, null);
                return;
            }
            const [ignoreState, effectiveContent] = annotateIgnorablePost(post, myUserId, userConfig, userFilter, ignoreContentMatcher);
            if (post.contentDefaultIgnorable) {
                post.rootElem.classList.add('rinsp-filter-default-ignorable');
            }
            if (post.postDefaultHidden) {
                return;
            }

            // add block content
            post.rootElem.querySelectorAll('.rinsp-ignore-switch').forEach(el => el.remove());

            setupIgnoreButton(ignoreButtonDef());
            if (post.postCollapsed) {
                setupIgnoreButton(defaultIgnorableButtonDef());
            }

            function setupIgnoreButton(buttonDef) {
                if (effectiveContent != null && !post.rootElem.classList.contains('rinsp-filter-ignored-byuid')) {
                    let ignoreElem = buttonDef.create(post);
                    ignoreElem.classList.add('rinsp-ignore-switch');
                    const ignoreLabel = addElem(ignoreElem, 'a', null, { href: 'javascript:void(0)', title: '关键内容: \n' + effectiveContent });
    
                    ignoreLabel.addEventListener('click', async function() {
                        if (DEBUG_MODE) console.info('effective content = ' + effectiveContent);
                        ignoreElem.classList.add('rinsp-config-saving');
                        const newIgnoredState = !ignoreState;
                        const updatedIgnoreList = await contentIgnoreListConfigAccess.update(function(ignoreList) {
                            const newTerms = ignoreList.terms.filter(term => term !== effectiveContent);
                            if (newIgnoredState) {
                                newTerms.push(effectiveContent);
                            }
                            if (ignoreList.terms.length === newTerms.length) {
                                // no change
                                return;
                            }
                            ignoreList.terms = newTerms;
                            return ignoreList;
                        })
                        .finally(function() {
                            ignoreElem.classList.remove('rinsp-config-saving');
                        });
        
                        if (updatedIgnoreList != null) {
                            onUpdate({ ignoreList: updatedIgnoreList });
                        }
                        return false;
                    });
                    ignoreLabel.textContent = ignoreState ? buttonDef.restoreLabel : buttonDef.hideLabel;
                }
            }

        });

        const ignoredPosts = posts.filter(post => post.rootElem.classList.contains('rinsp-filter-ignored'));
        const ignoreCount = ignoredPosts.length;
        document.querySelectorAll('.r_two[rinsp-filter-group-size]').forEach(el=>el.removeAttribute('rinsp-filter-group-size'));
        document.querySelectorAll('.rinsp-filter-ignored-range-end-summary').forEach(el=>el.remove());
        if (ignoreCount > 0) {
            const groups = [];
            ignoredPosts.filter(post => !post.rootElem.classList.contains('rinsp-filter-bypass')).forEach(function(post) {
                const floor = post.floor;
                const forcedSingleton = post.rootElem.parentNode.previousElementSibling.matches('div');
                const thisSingleGroup = { elems: [{ row: post.rootElem, floor: post.floorElem }], floorMin: floor, floorMax: floor, singleton: forcedSingleton };
                if (groups.length === 0) {
                    groups.push(thisSingleGroup);
                } else {
                    const last = groups[groups.length - 1];
                    if (floor == last.floorMax + 1 && !forcedSingleton && !last.singleton) {
                        last.elems.push({ row: post.rootElem, floor: post.floorElem });
                        last.floorMax = floor;
                    } else {
                        groups.push(thisSingleGroup);
                    }
                }
            });
            groups.forEach(group => {
                group.elems[0].row.parentNode.removeAttribute('rinsp-filter-ignored-cont');
                group.elems[0].floor.closest('tr').querySelector('.r_two').removeAttribute('rinsp-filter-group-size');
                group.elems[0].floor.querySelectorAll('.rinsp-filter-ignored-range-end-summary').forEach(el => el.remove());
                if (group.floorMin === group.floorMax) {
                    return;
                }
                group.elems[0].floor.closest('tr').querySelector('.r_two').setAttribute('rinsp-filter-group-size', String(group.elems.length));
                const rangeEnd = newElem('span', 'rinsp-filter-ignored-range-end-summary');
                rangeEnd.appendChild(document.createTextNode(' - '));
                rangeEnd.appendChild(group.elems[group.elems.length - 1].floor.cloneNode(true));
                group.elems[0].floor.after(rangeEnd);

                const rangeEnd2 = newElem('span', 'rinsp-filter-ignored-range-end-summary');
                rangeEnd2.appendChild(document.createTextNode(' - '));
                const dateEndElem = group.elems[group.elems.length - 1].floor.closest('.fl').nextElementSibling.cloneNode(true);
                dateEndElem.classList.remove('fl');
                dateEndElem.classList.remove('gray');
                rangeEnd2.appendChild(dateEndElem);
                group.elems[0].floor.closest('.fl').nextElementSibling.appendChild(rangeEnd2);
                
                group.elems[0].floor.closest('tr').querySelector('.r_two').setAttribute('rinsp-filter-group-size', String(group.elems.length));
                group.elems.slice(1).forEach(el=>el.row.parentNode.setAttribute('rinsp-filter-ignored-cont', '1'));
            });
        }

        ignoreContentToggler.setCount(ignoreCount);
        if (ignoreCount === 0) {
            document.body.classList.remove('rinsp-filter-peek-mode');
        }
    }

    function createIgnoreContentMatcher(ignoreList) {
        const ignoreSet = new Set();
        ignoreList.terms.forEach(term => {
            const matchableText = term
                .split(/(<[^>]+>)/)
                .map(chunk => {
                    if (chunk.length === 0 || chunk[0] === '<') {
                        return chunk;
                    } else {
                        return simplifyText(chunk);
                    }
                })
                .join('');
            ignoreSet.add(matchableText);
        });
        return {
            matches(effectiveContent) {
                return ignoreSet.has(effectiveContent);
            }
        };
    }

    async function updateUserBlockRule(opId, opName, data, callback) {
        const updatedUserFilterConfig = await userFilterConfigAccess.update(function(userFilterConfig) {
            const userIdKey = '#' + opId;
            if (data) {
                const updateData = Object.assign({
                    name: opName
                }, data);
                if (userFilterConfig.users[userIdKey] == null) {
                    userFilterConfig.users[userIdKey] = updateData;
                } else {
                    userFilterConfig.users[userIdKey] = Object.assign(userFilterConfig.users[userIdKey], updateData);
                }
            } else {
                delete userFilterConfig.users[userIdKey];
            }
            return userFilterConfig;
        });
        callback({ userFilter: updatedUserFilterConfig });
    }

    async function promptBlockOp(opId, opName, callback) {
        if (confirm(`屏蔽此用户(${opName}) 所有主题帖?`)) {
            updateUserBlockRule(opId, opName, { hideThreads: true, hideReplies: false }, callback);
        }
    }

    async function promptRemoveUserBlock(opId, opName, callback) {
        if (confirm(`取消屏蔽此用户(${opName})?`)) {
            updateUserBlockRule(opId, opName, null, callback);
        }
    }

    async function promptBlockComplete(opId, opName, callback) {
        if (confirm(`屏蔽此用户(${opName}) 所有主题及回复?`)) {
            updateUserBlockRule(opId, opName, { hideThreads: true, hideReplies: true }, callback);
        }
    }

    async function promptEditKeywords(message, initValues, initText, attr, callback) {
        const input = prompt(message, initText == null ? initValues.slice(0).sort().join(' ') : initText);
        if (input == null) {
            return;
        }
        const keywords = input.toLowerCase().trim().split(' ').filter(s=>!!s);
        const updatedFilterConfig = await threadFilterConfigAccess.update(function(filterConfig) {
            const set = new Set(filterConfig[attr]);
            initValues.forEach(kw => set.delete(kw));
            keywords.forEach(kw => set.add(kw));
            filterConfig[attr] = Array.from(set);
            return filterConfig;
        });
        callback({ threadFilter: updatedFilterConfig });
    }

    async function removeBlockThread(tid, callback) {
        const updatedFilterConfig = await threadFilterConfigAccess.update(function(filterConfig) {
            const set = new Set(filterConfig.dislikes);
            set.delete(getTidMatchDirective(tid));
            filterConfig.dislikes = Array.from(set);
            return filterConfig;
        });
        callback({ threadFilter: updatedFilterConfig });
    }

    async function promptBlockThread(tid, callback) {
        if (confirm('屏蔽此帖?')) {
            const updatedFilterConfig = await threadFilterConfigAccess.update(function(filterConfig) {
                const set = new Set(filterConfig.dislikes);
                set.add(getTidMatchDirective(tid));
                filterConfig.dislikes = Array.from(set);
                return filterConfig;
            });
            callback({ threadFilter: updatedFilterConfig });
        }
    }

    async function removeLikeThread(tid, callback) {
        const updatedFilterConfig = await threadFilterConfigAccess.update(function(filterConfig) {
            const set = new Set(filterConfig.likes);
            set.delete(getTidMatchDirective(tid));
            filterConfig.likes = Array.from(set);
            return filterConfig;
        });
        callback({ threadFilter: updatedFilterConfig });
    }

    async function promptLikeThread(tid, callback) {
        if (confirm('标记此帖?')) {
            const updatedFilterConfig = await threadFilterConfigAccess.update(function(filterConfig) {
                const set = new Set(filterConfig.likes);
                set.add(getTidMatchDirective(tid));
                filterConfig.likes = Array.from(set);
                return filterConfig;
            });
            callback({ threadFilter: updatedFilterConfig });
        }
    }

    function initRequestThreadEnhancementExecutor(myUserId, userConfig, settlementPostMatcher) {
        const bountySteps = [10000, 5000, 2000, 1000, 500, 200, 100];
        
        function isSettlementPost(title) {
            if (userConfig.hideSettlementPost$UseDefaultKeywords) {
                if (title.match(DEFAULT_SETTLEMENT_BLACKLIST_PATTERN) != null) {
                    return false;
                }
                let titleReduced = title.replace(/@[^@ ]+/, '');
                const hasTarget = titleReduced.length < title.length;
                titleReduced = titleReduced.replace(DEFAULT_SETTLEMENT_STOPWORD_PATTERN, '');
                const beforeLength = titleReduced.length;
                titleReduced = titleReduced.replace(DEFAULT_SETTLEMENT_KEYWORD_PATTERN, '');
                const keywordLength = beforeLength - titleReduced.length;
                titleReduced = titleReduced.trim();
                if (hasTarget && titleReduced.length === 0)
                    return true;
                if (titleReduced.length <= DEFAULT_SETTLEMENT_TITLE_MAX_OTHER_TEXT_AMOUNT) {
                    if (hasTarget || keywordLength > 0) {
                        return true;
                    }
                }
            }
            if (settlementPostMatcher != null) {
                const matches = settlementPostMatcher.match(title.toLowerCase());
                if (matches.size > 0) {
                    return true;
                }
            }
            return false;
        }

        return {
            async run(thread) {
                this.clear(thread);
                const areaTag = thread.titleCell.querySelector(`.s8[href^="thread.php?fid-${QUESTION_AND_REQUEST_AREA_ID}-type-"]`);
                if (areaTag == null) {
                    return;
                }
                thread.row.classList.add('rinsp-request-thread');
                const title = thread.titleCell.querySelector('h3 > a').textContent;
                const bountyElem = thread.titleCell.querySelector('h3 > .s1');
                if (bountyElem) {
                    const betterBountyDisplay = newElem('span', 'rinsp-request-bounty');
                    bountyElem.parentNode.insertBefore(betterBountyDisplay, bountyElem);
    
                    const opDate = new Date((thread.row.querySelector('td[id^="td_"] + td > .f10.gray2')||{}).textContent||'') * 1;
                    const nowRoundDay = Math.floor(Date.now() / 86400000) * 86400000;
                    
                    const expired = nowRoundDay - opDate >= 30 * 86400000; // 30 days request expiry
                    const statusText = expired ? '已超时' : '悬赏金额';
                    if (expired) {
                        thread.row.classList.add('rinsp-request-thread-expired');
                    } else {
                        thread.row.classList.add('rinsp-request-thread-ongoing');
                    }
                    const baseBounty = (bountyElem.textContent.match(/ *— 悬赏金额:*(\d+) *— */)||[])[1]*1||0;
                    const extraBounty = parseSpAmount(title);
    
                    addElem(betterBountyDisplay, 'span').textContent = `— ${statusText}:`;
                    addElem(betterBountyDisplay, 'span', 'rinsp-base-bounty').textContent = String(baseBounty);
                    if (extraBounty > baseBounty) {
                        addElem(betterBountyDisplay, 'span', 'rinsp-extra-bounty').textContent = '+' + extraBounty;
                    }
                    addElem(betterBountyDisplay, 'span').textContent = ' —';
                    const totalBounty = extraBounty > baseBounty ? extraBounty : baseBounty;
                    thread.row.setAttribute('rinsp-request-bounty-steps', bountySteps.filter(amt => totalBounty > amt).join(' ') + ' min');
    
                } else {
                    if (userConfig.requestThreadUseHistoryData && threadHistoryAccess) {
                        const lastAccessRecord = await threadHistoryAccess.recentAccessStore.get(thread.tid);
                        if (lastAccessRecord && lastAccessRecord.bounty && lastAccessRecord.bounty.winner === userConfig.myUserHashId) {
                            thread.row.classList.add('rinsp-request-thread-won');
                        }
                    }
                    thread.row.classList.add('rinsp-request-thread-ended');
                }
    
                if (hasMentionedMe(title, myUserId, userConfig)) {
                    if (userConfig.hideSettlementPost$HighlightMyself) {
                        thread.row.classList.add('rinsp-request-settlement-bypass');
                    }
                    thread.row.classList.add('rinsp-thread-mention-me');
                }
                
                if (thread.op === myUserId) {
                    thread.row.classList.add('rinsp-thread-byme');
                }
                if (userConfig.hideSettlementPost) {
                    if (isSettlementPost(thread.title)) {
                        if (thread.op === myUserId) {
                            thread.row.classList.add('rinsp-request-settlement-bypass');
                        }
                        thread.row.classList.add('rinsp-request-settlement-thread');
                    }
                }
            },
            clear(thread) {
                thread.row.removeAttribute('rinsp-request-bounty-steps');
                thread.row.classList.remove('rinsp-request-thread');
                thread.row.classList.remove('rinsp-request-thread-ongoing');
                thread.row.classList.remove('rinsp-request-thread-ended');
                thread.row.querySelectorAll('.rinsp-request-bounty').forEach(el => el.remove());
                thread.row.classList.remove('rinsp-thread-mention-me');
                thread.row.classList.remove('rinsp-thread-byme');
                thread.row.classList.remove('rinsp-request-settlement-bypass');
                thread.row.classList.remove('rinsp-request-settlement-thread');
            }
        };
    }
    
    function isScoreMismatched(score, defaultRating, bold) {
        if (score <= 1) {
            return false;
        }
        if (defaultRating.baseTotalScore === 0) {
            return false;
        }
        const fid = getCurrentPageFid();
        const diffAllowance = SCORE_DIFF_ALLOWANCE[`fid=${fid}`] || SCORE_DIFF_ALLOWANCE['fid=*'];
        if (defaultRating.ownBought || defaultRating.ownTranslate || bold) {
            const diff = score - defaultRating.baseTotalScore * 10;
            if (diff < defaultRating.baseTotalScore + 10) {
                return false;
            }
            if (score <= diffAllowance[1]) {
                return false;
            }
        }
        const diff = score - defaultRating.baseTotalScore;
        if (diff < defaultRating.baseTotalScore / 2 + diffAllowance[0]) {
            return false;
        }
        return true;
    }

    function initThreadHistoryExecutor(threadHistoryAccess) {
        return {
            async runBatch(threads) {
                const threadsById = new Map();
                threads.forEach(thread => {
                    threadsById.set(thread.tid, thread);
                });
                const tids = Array.from(threadsById.keys());
    
                const lastSeenRecords = await threadHistoryAccess.historyStore.getBatch(tids);
                lastSeenRecords.forEach((record, i) => {
                    const thread = threadsById.get(tids[i]);
                    this.clear(thread);
                    if (record != null) {
                        thread.row.classList.add('rinsp-thread-visited');
                        if (thread.replyCount > record) {
                            thread.row.classList.add('rinsp-thread-visited-update');
                        }
                    }
                });
            },
            async run(thread) {
                await this.runBatch([thread]);
            },
            clear(thread) {
                thread.row.classList.remove('rinsp-thread-visited');
                thread.row.classList.remove('rinsp-thread-visited-update');
            }
        };
    }

    function initMarkPinnedUserThreadExecutor() {
        return {
            async runBatch(threads) {
                threads.forEach(thread => {
                    this.clear(thread);
                    if (userPinArea.isWatching(thread.op)) {
                        thread.row.classList.add('rinsp-thread-user-pinned');
                        userPinArea.addLocation(thread.op, '⚪', null, () => {
                            thread.row.scrollIntoView({
                                behavior: 'smooth',
                                block: 'center'
                            });
                        });
                    }
                });
            },
            clear(thread) {
                thread.row.classList.remove('rinsp-thread-user-pinned');
            }
        };
    }

    function initMarkPaywellThreadExecutor() {
        return {
            run(thread) {
                this.clear(thread);
                if (thread.areaId != null) {
                    if (PAYWALL_AREA_IDS.has(thread.areaId)) {
                        thread.row.classList.add('rinsp-thread-paywall');
                    }
                }
            },
            clear(thread) {
                thread.row.classList.remove('rinsp-thread-paywall');
            }
        };
    }

    function initClosedThreadFilterExecutor() {
        return {
            run(thread) {
                this.clear(thread);
                if (thread.row.querySelector('td > a > img[src="images/colorImagination/thread/topicclose.gif"]')) {
                    thread.row.classList.add('rinsp-thread-filter-closed');
                }
            },
            clear(thread) {
                thread.row.classList.remove('rinsp-thread-filter-closed');
            }
        };
    }

    function initScoredThreadFilterExecutor() {
        return {
            run(thread) {
                this.clear(thread);
                let scoreElem = thread.titleCell.querySelector('h3 + .gray.tpage');
                if (scoreElem) {
                    const score = (scoreElem.textContent.match(/ *( [+-]\d+ ) */)||[])[1] * 1;
                    
                    const defaultRating = getDefaultRating(thread.titleCell.querySelector('h3 > a').textContent);
                    if (isScoreMismatched(score, defaultRating, thread.titleCell.querySelector('h3 > a > b') != null)) {
                        thread.row.classList.add('rinsp-thread-filter-miscored');
                    } else {
                        thread.row.classList.add('rinsp-thread-filter-scored');
                    }
                } else {
                    thread.row.classList.add('rinsp-thread-filter-unscored');
                }
            },
            clear(thread) {
                thread.row.classList.remove('rinsp-thread-filter-scored');
                thread.row.classList.remove('rinsp-thread-filter-unscored');
            }
        };
    }

    function initCategorizeShareTypeThreadExecutor(shareTypeFilter) {
        return {
            async runBatch(threads) {
                shareTypeFilter.clearState();
                let rejectedChainSize = 0;
                let lastRejected = null;
                threads.slice().reverse().forEach(thread => {
                    if (thread.areaId > 0 && !RESOURCE_AREA_IDS.has(thread.areaId) && !PAYWALL_AREA_IDS.has(thread.areaId)) {
                        rejectedChainSize = 0;
                        lastRejected = null;
                        return;
                    }
                    this.clear(thread);
                    const accepted = shareTypeFilter.addAndAcceptThread(thread.tid, thread.title);
                    if (accepted) {
                        rejectedChainSize = 0;
                        lastRejected = null;
                    } else {
                        rejectedChainSize++;
                        thread.row.classList.add('rinsp-thread-filter-masked-bysharetype');
                        if (lastRejected) {
                            thread.row.setAttribute('rinsp-sharetype-chain', rejectedChainSize);
                            thread.titleCell.setAttribute('rinsp-sharetype-chain', rejectedChainSize);
                            lastRejected.titleCell.removeAttribute('rinsp-sharetype-chain');
                            lastRejected.row.classList.add('rinsp-thread-filter-masked-bysharetype-collapse');
                        }
                        lastRejected = thread;
                    }
                });
                await shareTypeFilter.update();
            },
            clear(thread) {
                thread.row.removeAttribute('rinsp-sharetype-chain');
                thread.titleCell.removeAttribute('rinsp-sharetype-chain');
                thread.row.classList.remove('rinsp-thread-filter-masked-bysharetype');
                thread.row.classList.remove('rinsp-thread-filter-masked-bysharetype-collapse');
            }
        };
    }

    async function openBookmarkUserMenu(anchor, userId, userHashId, userNickName, deleteAction) {
        setupPopupMenu({
            title: userNickName,
            width: 160,
            popupMenuId: USER_BOOKMARK_ACTION_POPUP_MENU_ID,
            anchor,
            items: [
                {
                    label: '发短消息',
                    class: 'rinsp-mailto-user-bookmark-button',
                    action: `message.php?action-write-touid-${userId}.html`
                },
                {
                    label: '通信记录',
                    action: `message.php?action-chatlog-withuid-${userId}.html`
                },
                {
                    label: '移除收藏',
                    class: 'rinsp-alert-menu-button',
                    action: deleteAction
                }
            ]
        });
    }

    async function openFilterMenu(anchor, thread, onClose, onUpdate, unblockable) {
        const fid = getCurrentPageFid();
        const items = [];

        const dlsiteMatch = thread.title.match(/[RBVrbv][Jj](?:01)?\d{6}/);
        if (dlsiteMatch) {
            const code = dlsiteMatch[0].toUpperCase();
            items.push({
                label: `${code} (官网)`,
                action: `https://www.dlsite.com/maniax/work/=/product_id/${code}.html`, // dlsite will handle redirection
                target: '_blank'
            });
        }

        items.push({
            label: '搜索 (全部版块)',
            action: `search.php#prefill("keyword",${JSON.stringify(thread.title)})`,
            target: '_blank'
        });

        items.push({
            label: '搜索 (此版块内)',
            action: `search.php#prefill("fid",${fid},"keyword",${JSON.stringify(thread.title)})`,
            target: '_blank'
        });

        items.push({
            label: '添加屏蔽词',
            class: 'rinsp-thread-filter-dislike-button',
            action: () => promptEditKeywords('请输入屏蔽关键词 用空格分开', [], thread.title, 'dislikes', onUpdate)
        });

        items.push({
            label: '添加喜欢项目',
            class: 'rinsp-thread-filter-like-button',
            action: () => promptEditKeywords('请输入喜欢的关键词 用空格分开', [], thread.title, 'likes', onUpdate)
        });

        if (thread.row.classList.contains('rinsp-thread-filter-like-bytid')) {
            items.push({
                label: '移除标记',
                class: 'rinsp-thread-filter-liketid-button',
                action: () => removeLikeThread(thread.tid, onUpdate)
            });
        } else {
            items.push({
                label: '标记此帖',
                class: 'rinsp-thread-filter-liketid-button',
                action: () => promptLikeThread(thread.tid, onUpdate)
            });
        }

        if (thread.row.classList.contains('rinsp-thread-filter-dislike-bytid')) {
            items.push({
                label: '取消屏蔽此帖',
                class: 'rinsp-thread-filter-hidetid-button',
                action: () => removeBlockThread(thread.tid, onUpdate)
            });
        } else if (!unblockable) {
            items.push({
                label: '🛇只屏蔽此帖',
                class: 'rinsp-thread-filter-hidetid-button',
                action: () => promptBlockThread(thread.tid, onUpdate)
            });
        }

        if (thread.row.classList.contains('rinsp-thread-filter-dislike-byuid')) {
            items.push({
                label: '取消屏蔽用户',
                class: 'rinsp-thread-filter-hideop-button',
                action: () => promptRemoveUserBlock(thread.op, thread.opName, onUpdate)
            });
        } else if (!unblockable) {
            items.push({
                label: '🛇屏蔽所有主题帖',
                class: 'rinsp-thread-filter-hideop-button',
                action: () => promptBlockOp(thread.op, thread.opName, onUpdate)
            });
        }

        setupPopupMenu({
            title: '帖子功能选择',
            popupMenuId: THREAD_FILTER_ACTION_POPUP_MENU_ID,
            width: 150,
            anchor,
            rightAligned: true,
            onClose,
            items
        });
    }

    function getTidMatchDirective(tid) {
        return 'tid:<' + tid + '>';
    }

    function initThreadFilterExecutor(myUserId, threadFilter, userFilter, onUpdate, isGF) {
        const dislikePatternMatcher = createKeywordMatcherFactory(threadFilter.dislikes);
        const likePatternMatcher = createKeywordMatcherFactory(threadFilter.likes);
        return {
            run(thread) {
                this.clear(thread);
                const unblockable = thread.op === myUserId || THREAD_FILTER_EXEMPTED_USERS.has(thread.op);
                const tidDirective = getTidMatchDirective(thread.tid);
                const matchableTitle = thread.title.toLowerCase() + ' ' + tidDirective;
                const filterMenuButton = addElem(thread.titleCell, 'div', 'rinsp-thread-filter-menu-button');
        
                filterMenuButton.addEventListener('click', () => {
                    thread.row.classList.add('rinsp-thread-selected');
                    openFilterMenu(filterMenuButton, thread, () => thread.row.classList.remove('rinsp-thread-selected'), onUpdate, unblockable);
                });
    
                thread.titleCell.appendChild(document.createTextNode(' ')); // put a space to avoid copying to include button text
                const dislikeUserRule = userFilter.users['#' + thread.op];
                if (dislikeUserRule != null) {
                    if (dislikeUserRule.hideThreads) {
                        thread.row.classList.add('rinsp-thread-filter-dislike');
                        thread.row.classList.add('rinsp-thread-filter-dislike-byuid');
                    }
                }
                if (dislikePatternMatcher != null) {
                    const matches = dislikePatternMatcher.match(matchableTitle);
                    if (matches.size > 0) {
                        thread.row.classList.add('rinsp-thread-filter-dislike');
                        thread.row.classList.add('rinsp-thread-filter-dislike-bytitle');
                        const filteredWords = addElem(thread.titleCell, 'div', 'rinsp-thread-filter-dislike-words');
                        if (matches.has(tidDirective)) {
                            thread.row.classList.add('rinsp-thread-filter-dislike-bytid');
                            filteredWords.textContent = '🚫此帖已屏蔽';
                            filteredWords.addEventListener('click', async () => {
                                if (confirm('取消屏蔽此帖?')) {
                                    removeBlockThread(thread.tid, onUpdate);
                                }
                            });
                        } else {
                            filteredWords.textContent = '🚫' + Array.from(matches).sort().join(' ').replace(/^(.{20}).+/, '$1...');
                            filteredWords.addEventListener('click', async () => {
                                const staticKeywords = Array.from(matches).filter(w => w !== tidDirective && dislikePatternMatcher.isStaticKeyword(w));
                                promptEditKeywords('请编辑屏蔽关键词 用空格分开', Array.from(staticKeywords), null, 'dislikes', onUpdate);
                            });
                        }
                        if (unblockable) {
                            thread.row.classList.add('rinsp-thread-filter-bypass');
                        } else if (userConfig.customUserBypassThreadFilter) {
                            if (userConfig.customUserHashIdMappings['#' + thread.op] != null) {
                                thread.row.classList.add('rinsp-thread-filter-bypass');
                            }
                        }
                    }
                }
                if (likePatternMatcher != null) {
                    const matches = likePatternMatcher.match(matchableTitle);
                    if (matches.size > 0) {
                        thread.row.classList.add('rinsp-thread-filter-like');
                        const filteredWords = addElem(thread.titleCell, 'div', 'rinsp-thread-filter-like-words');
                        if (matches.has(tidDirective)) {
                            thread.row.classList.add('rinsp-thread-filter-like-bytid');
                            filteredWords.textContent = '💚标记帖';
                            filteredWords.addEventListener('click', async () => {
                                if (confirm('移除标记?')) {
                                    removeLikeThread(thread.tid, onUpdate);
                                }
                            });
                            if (userConfig.showFavThreadFloatingList) {
                                if (isGF) {
                                    userPinArea.addLocationHighlight(0, '💚:标记帖');
                                } else {
                                    let label = userConfig.showFavThreadFloatingList$withTitle ? '标记帖 - ' + thread.title : '标记帖';
                                    userPinArea.addLocationHighlight(null, '💚:' + label, thread.row, `/read.php?tid-${thread.tid}.html`);
                                }
                            }
                        } else {
                            const matchList = Array.from(matches).sort();
                            filteredWords.textContent = '💚' + matchList.join(' ').replace(/^(.{20}).+/, '$1...');
                            filteredWords.addEventListener('click', async () => {
                                const staticKeywords = matchList.filter(w => likePatternMatcher.isStaticKeyword(w));
                                promptEditKeywords('请编辑喜欢的关键词 用空格分开', staticKeywords, null, 'likes', onUpdate);
                            });
                            if (userConfig.showFavThreadFloatingList) {
                                if (isGF) {
                                    userPinArea.addLocationHighlight(0, '💚:' + matchList[0]);
                                } else {
                                    let label = userConfig.showFavThreadFloatingList$withTitle ? matchList[0] + ' - ' + thread.title : matchList[0];
                                    userPinArea.addLocationHighlight(null, '💚:' + label, thread.row, `/read.php?tid-${thread.tid}.html`);
                                }
                            }
                        }
                    }
                }
    
                if (thread.opElem) {
                    if (thread.row.classList.contains('rinsp-thread-filter-dislike-byuid')) {
                        const removeFilterOpButton = newElem('div', 'rinsp-thread-filter-unhideop-button');
                        removeFilterOpButton.textContent = '🚫';
                        thread.opElem.after(removeFilterOpButton);
                        removeFilterOpButton.addEventListener('click', async () => {
                            promptRemoveUserBlock(thread.op, thread.opElem.textContent.trim(), onUpdate);
                        });
                    }
                }
            },
            clear(thread) {
                thread.row.classList.remove('rinsp-thread-filter-like');
                thread.row.classList.remove('rinsp-thread-filter-like-bytid');
                thread.row.classList.remove('rinsp-thread-filter-dislike');
                thread.row.classList.remove('rinsp-thread-filter-dislike-bytitle');
                thread.row.classList.remove('rinsp-thread-filter-dislike-bytid');
                thread.row.classList.remove('rinsp-thread-filter-dislike-byuid');
                thread.row.querySelectorAll('.rinsp-thread-filter-like-words,.rinsp-thread-filter-dislike-words,.rinsp-thread-filter-menu-button,.rinsp-thread-filter-unhideop-button').forEach(el => el.remove());
            }
        };
    }

    async function applyThreadListEnhancement(myUserId, userConfig, threadList, threadFilter, userFilter, userPinArea, shareTypeFilter, settlementPostMatcher, threadHistoryAccess, updateCallback, fresh) {
        if (DEBUG_MODE) console.info('thread-list', threadList);

        const executors = [
            initScoredThreadFilterExecutor(),
            initClosedThreadFilterExecutor(),
            initThreadFilterExecutor(myUserId, threadFilter, userFilter, onUpdate),
            initRequestThreadEnhancementExecutor(myUserId, userConfig, settlementPostMatcher),
            initMarkPaywellThreadExecutor(),
            initMarkPinnedUserThreadExecutor(userPinArea),
        ];
        
        if (shareTypeFilter) {
            executors.push(initCategorizeShareTypeThreadExecutor(shareTypeFilter));
        }
        if (threadHistoryAccess) {
            executors.push(initThreadHistoryExecutor(threadHistoryAccess));
        }

        function onUpdate(updatedConfigs) {
            updateCallback(updatedConfigs);
        }

        let newThreadList;
        if (fresh || threadList.some(thread => !thread.row.classList.contains('rinsp-thread-inspected'))) {
            threadList.forEach(thread => {
                thread.row.classList.remove('rinsp-thread-inspected');
            });
            newThreadList = threadList;
        } else {
            return;
        }

        userPinArea.clearState();
        for (let executor of executors) {
            if (typeof executor.runBatch === 'function') {
                await executor.runBatch(newThreadList);
            } else {
                const awaits = [];
                for (let thread of newThreadList) {
                    const returnValue = executor.run(thread);
                    if (returnValue instanceof Promise) {
                        awaits.push(returnValue);
                    }
                }
                if (awaits.length > 0) {
                    await Promise.allSettled(awaits);
                }
            }
        }
        await userPinArea.update();

        newThreadList.forEach(thread => {
            thread.row.classList.add('rinsp-thread-inspected');
        });

        const dislikeThreadCount = document.querySelectorAll('.rinsp-thread-filter-dislike').length;
        ignoreDislikeThreadToggler.setCount(dislikeThreadCount);

        const settlementCount = document.querySelectorAll('.rinsp-request-settlement-thread:not(.rinsp-request-settlement-bypass)').length;
        ignoreSettlementToggler.setCount(settlementCount);

        ignorePaywellToggler.setCount(document.querySelectorAll('.rinsp-thread-paywall').length);
        if (hideAnsweredRequestToggler) {
            hideAnsweredRequestToggler.setCount(document.querySelectorAll('.rinsp-request-thread.rinsp-request-thread-ended').length);
        }
        if (hideUnansweredRequestToggler) {
            hideUnansweredRequestToggler.setCount(document.querySelectorAll('.rinsp-request-thread.rinsp-request-thread-ongoing').length);
        }
        if (hideExpiredRequestToggler) {
            hideExpiredRequestToggler.setCount(document.querySelectorAll('.rinsp-request-thread.rinsp-request-thread-expired').length);
        }
        if (hideVisitedThreadToggler) {
            hideVisitedThreadToggler.setCount(document.querySelectorAll('.rinsp-thread-visited').length);
        }
        hideClosedThreadToggler.setCount(document.querySelectorAll('.rinsp-thread-filter-closed').length);
    }
    
} // END: init()

let forumUserBlacklistCache = {
    data: {
        banPosts: new Set(),
        banAvatars: new Set(),
    },
    key: ''
};

function getForumUserBlacklist() {
    if (forumUserBlacklistCache.key === localStorage.config||'') {
        return forumUserBlacklistCache.data;
    }
    const banPosts = new Set();
    const banAvatars = new Set();
    try {
        const data = JSON.parse(localStorage.config||'{}');
        for (let entry of Object.entries(data.blist||{})) {
            for (let mode of entry[1].level||[]) {
                if (mode === 'topic') {
                    banPosts.add(entry[0] * 1);
                } else if (mode === 'avatar') {
                    banAvatars.add(entry[0] * 1);
                }
            }
        }
    } catch (ex) {}
    const data = {
        banPosts, banAvatars
    };
    forumUserBlacklistCache = {
        data: data,
        key: localStorage.config||''
    };
    return data;
}
function isUserBlacklisted(userFilter, uid) {
    const userFilterRule = userFilter.users['#' + uid];
    if (userFilterRule && userFilterRule.hideReplies) {
        return true;
    }
    return getForumUserBlacklist().banPosts.has(uid * 1);
}

function getPosts(doc, max) {
    const tidRefLink = doc.querySelector('#td_tpc .tiptop .fr a[href^="read.php?tid-"], th[id^="td_"] .tiptop .fr a[href^="read.php?tid-"]');
    if (tidRefLink == null) {
        return [];
    }
    const tid = Number.parseInt(tidRefLink.getAttribute('href').substring(13));
    const currentLevel = doc.querySelector('#breadcrumbs > a.crumbs-item[href^="thread.php?fid-"] + .crumbs-item.current');
    let areaId = null;
    if (currentLevel) {
        areaId = Number.parseInt(currentLevel.previousElementSibling.getAttribute('href').substring(15));
    }
    function getAuthor(mainCell) {
        const nameElem = mainCell.closest('tr').querySelector('.user-pic ~ div a[href^="u.php?action-show-uid-"] > strong');
        return [Number.parseInt(nameElem.parentNode.getAttribute('href').substring(22)), nameElem.textContent.trim(), nameElem];
    }
    function getFloorElem(mainCell) {
        return mainCell.querySelector('.tiptop > .fl > .s3');
    }

    function getFloor(floorElem) {
        const flText = floorElem.textContent.trim();
        return flText === 'GF' ? 0 : Number.parseInt(flText.substring(1));
    }
    function getDate(mainCell) {
        return new Date(mainCell.querySelector('.tiptop > .fl + .fl.gray[title^="发表于"]').textContent.trim()) * 1;
    }

    const posts = [];
    doc.querySelectorAll('th.r_one[id^=td_] .tpc_content > div[id^="read_"]').forEach(function(contentElem) {
        if (max != null && posts.length >= max) {
            return;
        }
        const rootElem = contentElem.closest('.t2.t5 > table');
        if (rootElem == null) {
            return;
        }
        const mainCell = contentElem.closest('th.r_one[id^=td_]');
        const floorElem = getFloorElem(mainCell);
        const [postUid, postUname, nameElem] = getAuthor(mainCell);

        const userPicImg = rootElem.querySelector('.user-pic a[href^="u.php?action-show-uid-"] > img');
        const userPic = userPicImg.closest('.user-pic');
        const userPicImgSrc = getImgSrc(userPicImg);
        let defaultUserPic = false;
        if (userPicImgSrc.startsWith('images/face/')) {
            if (userPicImgSrc === 'images/face/none.gif') {
                defaultUserPic = 'none';
            } else {
                defaultUserPic = 'preset';
            }
        }

        const postDefaultHidden = rootElem.parentNode.getAttribute('hidden') != null;
        const puremarkContent = rootElem.querySelector('.tiptop > .js-puremark-content');
        const postCollapsed = puremarkContent != null && (puremarkContent.getAttribute('style')||'').match(/\bdisplay *: *none(?:;|\b)/) == null;

        const postEid = contentElem.getAttribute('id');
        const postId = postEid.split('_')[1] * 1;
        const post = {
            tid,
            areaId,
            postId,
            postUid,
            postUname,
            postTime: getDate(mainCell),
            floor: getFloor(floorElem),
            floorElem: getFloorElem(mainCell),
            rootElem,
            mainCell,
            contentElem,
            userNameElem: nameElem,
            userPicElem: userPic,
            defaultUserPic,
            postDefaultHidden,
            contentDefaultIgnorable: puremarkContent != null,
            postCollapsed
        };
        posts.push(post);
    });
    return posts;
}

function readThreadList() {
    const threadList = [];
    let divider = document.querySelector('tr.tr2 > .tac[colspan="6"]');
    let prefixSelector = '';
    if (divider && divider.textContent.trim() === '普通主题') {
        divider = divider.closest('tr');
        divider.classList.add('rinsp-thread-section-divider');
        prefixSelector = '.rinsp-thread-section-divider ~ ';
        let row = divider.previousElementSibling;
        while (row != null && row.classList.contains('t_one')) {
            const elem = row.querySelector('td[id^="td_"] > h3 > a[href^="read.php?tid-"]');
            if (elem && elem.closest('td').querySelector('a.s8[href^="thread.php?fid-"]') != null) {
                collect(elem);
            }
            row = row.previousElementSibling;
        }
        threadList.reverse();
    } else {
        divider = null;
    }
    document.querySelectorAll(prefixSelector + '.t_one > td[id^="td_"] > h3 > a[href^="read.php?tid-"], .rinsp-infscroll-divider ~ tbody .t_one > td[id^="td_"] > h3 > a[href^="read.php?tid-"], .spp-infinite-scroll-divider ~ tbody .t_one > td[id^="td_"] > h3 > a[href^="read.php?tid-"]').forEach(elem => {
        collect(elem);
    });
    function collect(link) {
        const tid = Number.parseInt(link.getAttribute('href').substring(13));
        const row = link.closest('tr');
        const opElem = row.querySelector('td > a.bl[href^="u.php?action-show-uid-"]');
        const countElem = row.querySelector('td.f10 > .s8');
        let replyCount = 0;
        let hitCount = 0;
        if (countElem) {
            const match = countElem.parentElement.textContent.trim().match(/^(\d+)[\u00A0 ]*\/[\u00A0 ]*(\d+)$/);
            if (match) {
                replyCount = match[1] * 1;
                hitCount = match[2] * 1;
            }
        }
        const byCell = row.querySelector('td.tal.y-style > .f10 + br + .gray2');
        const replyHashId = byCell ? (byCell.textContent.trim().match(/^by: ([a-f0-9]{8})$/)||[])[1]||null : null;
        threadList.push({
            tid,
            row: row,
            title: getEffectiveThreadTitle(link.textContent),
            titleCell: link.closest('td'),
            opElem: opElem,
            opName: opElem ? opElem.textContent.trim() : null,
            op: opElem ? Number.parseInt(opElem.getAttribute('href').substring(22)) : null,
            opHashId: replyCount === 0 ? replyHashId : null,
            replyCount,
            hitCount
        });
    }
    return threadList;
}

function readPicWallThreadList() {
    const threadList = [];
    document.querySelectorAll('#wall ul.stream > li > .inner').forEach(elem => {
        const row = elem.parentNode;
        const link = elem.querySelector('.section-title > a[href^="./read.php?tid-"]');
        const tid = Number.parseInt(link.getAttribute('href').substring(15));
        const countElem = elem.querySelector('.section-text > span + span');
        let replyCount = 0;
        let hitCount = 0;
        if (countElem) {
            const match = countElem.textContent.trim().match(/:[\u00A0 ]*(\d+)[\u00A0 ]*\/[\u00A0 ]*(\d+)$/);
            if (match) {
                replyCount = match[1] * 1;
                hitCount = match[2] * 1;
            }
        }
        const opElem = elem.querySelector('.section-intro a.bl[href^="u.php?action-show-uid-"]');
        threadList.push({
            tid,
            row,
            title: getEffectiveThreadTitle(link.textContent),
            titleCell: link.parentNode,
            opElem: opElem,
            opName: opElem ? opElem.textContent.trim() : null,
            op: opElem ? Number.parseInt(opElem.getAttribute('href').substring(22)) : null,
            replyCount,
            hitCount
        });
    });
    return threadList;
}

function readSearchThreadList() {
    const threadList = [];
    document.querySelectorAll('table > tbody > tr.tr3.tac > th > a[href^="read.php?tid-"]').forEach(link => {
        const tid = Number.parseInt(link.getAttribute('href').substring(13));
        const row = link.closest('tr');
        const areaElem = row.querySelector('th + td > a[href^="thread.php?fid-"]');
        const areaId = areaElem ? Number.parseInt(areaElem.getAttribute('href').substring(15)) : null;
        const areaName = areaElem ? areaElem.textContent.trim() : null;
        const opElem = row.querySelector('td.smalltxt > a[href^="u.php?action-show-uid-"]');
        const replyCountElem = opElem.parentElement.nextElementSibling;
        const hitCountElem = replyCountElem.nextElementSibling;
        threadList.push({
            tid,
            row: row,
            titleCell: link.closest('th'),
            title: getEffectiveThreadTitle(link.textContent),
            opElem: opElem,
            opName: opElem ? opElem.textContent.trim() : null,
            op: opElem ? Number.parseInt(opElem.getAttribute('href').substring(22)) : null,
            opHashId: opElem ? opElem.textContent.trim() : null,
            replyCount: replyCountElem.textContent.trim() * 1,
            hitCount: hitCountElem.textContent.trim() * 1,
            areaId: areaId,
            areaName: areaName
        });
    });
    return threadList;
}

function getEffectiveThreadTitle(rawTitle) {
    return rawTitle.replace(/^(\[(?:[三二]次元R18相关|全年龄正常向)\] *)+/g, '').replace(/\s+/g, ' ').trim();
}

function getEffectiveTextContent(rootNode, preferTextOnly, treatAllEmojiTheSameWay) {
    if (rootNode.textContent.length > MAX_IGNORE_CONTENT_TEXT_LENGTH || rootNode.innerHTML.length > MAX_IGNORE_CONTENT_HTML_LENGTH) {
        return null;
    }
    const contentList = [];
    let externalImg = false;
    let complexData = false;
    function visit(node) {
        let endContent = null;
        if (node.classList.contains('jumbotron')) {
            contentList.push('<SELL>');
            complexData = true;
            return;
        }
        switch (node.tagName) {
            case 'HR':
                return;
            case 'BR':
                contentList.push(' ');
                return;
            case 'SCRIPT':
            case 'STYLE':
            case 'BUTTON':
            case 'INPUT':
            case 'SELECT':
            case 'FORM':
            case 'VIDEO':
            case 'OBJECT':
            case 'FRAME':
            case 'IFRAME':
                contentList.push('<' + node.tagName + '>');
                complexData = true;
                return;
            case 'IMG':
                const imgSrc = getImgSrc(node)||'';
                if (imgSrc.startsWith('images/post/smile/')) {
                    let token;
                    if (treatAllEmojiTheSameWay) {
                        token = `<EMOJI>`;
                    } else {
                        const emojiId = imgSrc.substring(18).replace(/\//g, ':').replace(/\.[a-z]+$/, '').toUpperCase();
                        token = `<EMOJI:${emojiId}>`;
                    }
                    if (contentList.length === 0 || contentList[contentList.length - 1] !== token) {
                        contentList.push(token); // skip consecutive emoji
                    }
                } else {
                    contentList.push('<EXT_IMG>');
                    externalImg = true;
                }
                return;
            case 'FONT':
            case 'CENTER':
            case 'I':
            case 'U':
            case 'B':
            case 'SUP':
            case 'SUB':
            case 'CODE':
            case 'H1':
            case 'H2':
            case 'H3':
            case 'H4':
            case 'H5':
            case 'H6':
            case 'H7':
            case 'H8':
            case 'H9':
            case 'P':
            case 'SPAN':
            case 'DIV':
                break;
            case 'BLOCKQUOTE':
                if (node.classList.contains('blockquote3')) {
                    contentList.push('<QUOTE>');
                    complexData = true;
                    return;
                }
                break;
            case 'A':
            default:
                contentList.push(`<${node.tagName}>`);
                endContent = `<${node.tagName}_END>`;
                break;
            }
        for (const childNode of node.childNodes) {
            if (childNode.nodeType === 3) {
                const text = childNode.textContent.toLowerCase();
                if (text.length > 0) {
                    const normText = simplifyText(text);
                    if (normText) {
                        contentList.push(normText);
                    }
                }
            } else if (childNode.nodeType === 1) {
                visit(childNode);
                if (complexData) {
                    return;
                }
            }
        }
        if (endContent != null) {
            contentList.push(endContent);
        }
    }
    visit(rootNode);
    if (complexData) {
        return null;
    }
    if (preferTextOnly) {
        const textOnly = contentList.filter(s=>s[0]!=='<').join('').replace(/\s+/g, ' ').trim();
        if (textOnly) {
            const textWithoutSymbols = textOnly.replace(/!/g, ' ').replace(/ +/g, ' ').trim();
            if (textWithoutSymbols) {
                return textWithoutSymbols;
            }
            return textOnly;
        } else if (externalImg) {
            return null;
        }
    }
    return contentList.join('')
        .replace(/\s+/g, ' ')
        .trim() || '<EMPTY>';
}

function simplifyText(text) {
    const chars = Array.from(text);
    const normText = chars
        .reduce((arr, itm) => {
            if (arr.length === 0 || arr[arr.length - 1] !== itm) {
                arr.push(itm);
            }
            return arr;
        }, [])
        .join('');
    return normText.trim().replace(/\s+/g, ' ').replace(/[~⁉`‘’'~!!??。,、,.<>]+/g, '!');
}

function num(v) {
    return atob(v)*1;
}

function escapeRegExp(string) {
    return string.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
}

async function tryOpenUpdateSummary() {
    const lastVersion = await GM.getValue('last_version') * 1||0;
    if (VERSION_MAJOR > lastVersion) {
        await GM.setValue('last_version', String(VERSION_MAJOR));
        GM.notification({
            title: `南家功能强化(凛+) 已更新 ver.${VERSION_MAJOR}`,
            text: "请参阅功能简介帖"
        });
        if (confirm(`凛+ 已更新 ${VERSION_MAJOR},打开功能简介帖?(新页面)`)) {
            GM.openInTab(INTRO_POST, false);
        }
    }
}

const allowingHookPosters = [1597333055699695, 501998265301213];

function applyHookDirectives(posts, myUserId, userConfig) {
    
    function handleDirective_if_not_installed(hook) {
        const minVersion = (hook.getAttribute('sparg-min-ver')||'0') * 1;
        if (VERSION_FULL >= minVersion) {
            hook.setAttribute('hidden', '');
            hook.textContent = '';
        }
    }

    function handleDirective_if_installed(hook) {
        const minVersion = (hook.getAttribute('sparg-min-ver')||'0') * 1;
        if (VERSION_FULL >= minVersion) {
            hook.removeAttribute('hidden');
        }
    }

    function handleDirective_value_userhash(hook) {
        const prefix = hook.getAttribute('sparg-prefix')||'';
        const length = hook.getAttribute('sparg-length') * 1 || 6;
        const hashNum = cyrb53(prefix + myUserId);
        let hashStr = btoa(hashNum).replace(/[\+\/=]/g, '').split('').reverse().join('');
        while (hashStr.length < length) {
            hashStr = hashStr + hashStr;
        }
        hook.textContent = hashStr.substring(0, length).toUpperCase();
    }

    function handleDirective_value_version(hook) {
        hook.textContent = VERSION_TEXT;
    }

    function handleDirective(hook, directive) {
        switch (directive) {
            case 'when-not-installed':
                return handleDirective_if_not_installed(hook);
            case 'when-installed':
                return handleDirective_if_installed(hook);
            case 'value-userhash':
                return handleDirective_value_userhash(hook);
            case 'value-version':
                return handleDirective_value_version(hook);
        }
    }

    posts
        .filter(post => allowingHookPosters.indexOf(cyrb53(''+post.postUid, 233)) !== -1)
        .forEach(post => {
            post.contentElem.querySelectorAll('.rinsp-sp-hook').forEach(hook => {
                hook.classList.remove('rinsp-sp-hook');
                const directive = hook.getAttribute('sp-directive');
                hook.removeAttribute('sp-directive');
                if (directive) {
                    handleDirective(hook, directive);
                }
            });
        });
}

function applyReplyRefreshFree(tid, lastPost, myUserId) {
    const pagination = document.querySelector('.pages .pagesone');
    let onLastPage = true;
    if (pagination) {
        const match = pagination.textContent.match(/Pages: (\d+)\/(\d+)/);
        if (match && match[1] !== match[2]) {
            onLastPage = false;
        }
    }
    const replyForm = document.FORM;
    if (replyForm == null || replyForm.getAttribute('action') !== 'post.php?') {
        return;
    }
    if (replyForm.getAttribute('onsubmit') == null) {
        return;
    }
    
    const lastPageUrl = `${document.location.origin}/read.php?tid=${tid}&page=e&#a`;

    replyForm.addEventListener('submit', evt => {
        let ok = 1;
        try {
            ok = unsafeWindow.checkpost(replyForm);
        } catch (ignore) {
            console.warn('fail to execute unsafeWindow.checkpost', ignore);
        }
        if (!ok) {
            evt.preventDefault();
            return false;
        }

        executeSubmit()
            .catch(err => {
                console.error(err);
                document.location.href = lastPageUrl;
            })
            .finally(() => {
                replyForm.classList.remove('rinsp-refresh-free-submitting');
                // unlock and reset form for resubmission
                replyForm.Submit.disabled = false;
                replyForm.encoding = 'multipart/form-data';
                if (ok === true) { // meaning checkpost was executed successfully
                    unsafeWindow.cnt = 0;
                }
            });
        evt.preventDefault();
        return false;
    });
    replyForm.classList.add('rinsp-reply-refresh-free');
    replyForm.removeAttribute('onsubmit');

    let lastPostId = lastPost.postId;
    async function executeSubmit() {
        replyForm.classList.add('rinsp-refresh-free-submitting');
        const resp = await fetch(`${document.location.origin}/post.php`, {
            method: 'POST',
            mode: 'same-origin',
            credentials: 'same-origin',
            body: new FormData(replyForm)
        });
        if (!resp.ok) {
            throw Error('request failed');
        }
        const postFeedback = await resp.text();
        const parser = new DOMParser();
        const postFeedbackDoc = parser.parseFromString(postFeedback, 'text/html');
        const err = findErrorMessage(postFeedbackDoc);
        if (err) {
            alert(err.replace(/\s+/g, ' '));
            return;
        }

        if (onLastPage) {
            const newDoc = await fetchGetPage(lastPageUrl);
            const knownLastPost = newDoc.querySelector(`#td_${lastPostId||'tpc'}`);
            if (knownLastPost == null) {
                document.location.href = lastPageUrl;
                return;
            }
            const newRows = [];
            let myNewPostIdAttr = null;
            let row = knownLastPost.closest('.t5.t2');
            if (row.nextElementSibling && row.nextElementSibling.classList.contains('menu')) {
                row = row.nextElementSibling;
                if (row.nextElementSibling && row.nextElementSibling.matches('a[name="a"]')) {
                    row = row.nextElementSibling;
                }
            }
            while ((row = row.nextElementSibling) != null) {
                newRows.push(row.outerHTML);
                if (row.querySelector(`.user-pic a[href="u.php?action-show-uid-${myUserId}.html"]`)) {
                    myNewPostIdAttr = row.querySelector('th[id^="td_"]').getAttribute('id');
                }
            }

            if (myNewPostIdAttr == null) {
                // can't find new post, just load last page instead
                document.location.href = lastPageUrl;
            }

            const postList = lastPost.rootElem.closest('form');
            const tmpElem = newElem('div');
            tmpElem.innerHTML = newRows.join('\n');
            Array.from(tmpElem.children).forEach(node => postList.appendChild(node));
            tmpElem.remove();
            lastPostId = Number.parseInt(Array.from(postList.querySelectorAll('.t5.t2 th[id^="td_"]')).at(-1).getAttribute('id').substring(3));
            setTimeout(() => {
                document.querySelector(`#${myNewPostIdAttr}`).closest('.t5.t2').scrollIntoView({
                    behavior: 'smooth',
                    block: 'center'
                });
            });

            document.FORM.atc_content.value = '';
            document.FORM.querySelector('#attach').innerHTML = '';
            try {
                if (unsafeWindow.newAtt) { // attachment might be disabled, check if handler exists first
                    unsafeWindow.newAtt.create();
                }
            } catch (ignore) {}

        } else {
            document.location.href = lastPageUrl;
        }
        
    }
}

function applySubjectLineEnhancement(posts, userConfig) {
    let threadTitle = null;
    if (userConfig.hideRedundantReSubjectLine) {
        threadTitle = (document.querySelector('#breadcrumbs .crumbs-item.current > strong')||{}).textContent;
        if (threadTitle) {
            threadTitle = threadTitle.replace(/\s|\u00A0|&[a-z;]{0,5}$/g, '').trim();
        }
    }
    const pageArr = readCurrentAndMaxPage(document);
    const currentPage = pageArr[0];
    const maxPage = pageArr[1];
    const form = document.querySelector('form[name="delatc"]');
    let pageSize = 0;
    if (currentPage === 1) {
        if (maxPage > currentPage) {
            for (let child of form.children) {
                if (child.matches('.rinsp-infscroll-divider')) {
                    break;
                }
                if (child.matches('.t5.t2')) {
                    pageSize++;
                }
            }
        }
    } else {
        pageSize = posts[0].floor / (currentPage - 1);
    }
    function getFloorTarget(targetFloor) {
        let targetElem = posts.filter(post => post.floor === targetFloor).map(post => post.rootElem)[0];
        let targetPage = pageSize === 0 ? 1 : Math.floor(targetFloor / pageSize)+1;
        let hash = targetFloor > 0 ? `#fl-${targetFloor}` : '';
        if (targetPage > 1) {
            return { url: `/read.php?tid-${posts[0].tid}-fpage-0-toread--page-${targetPage}.html${hash}`, targetElem };
        } else {
            return { url: `/read.php?tid-${posts[0].tid}.html${hash}`, targetElem };
        }
    }

    posts.forEach(post => {
        if (post.rootElem.classList.contains('rinsp-subject-inspected')) {
            return;
        }
        post.rootElem.classList.add('rinsp-subject-inspected');
        const subjectLine = post.mainCell.querySelector('.h1.fl > h1[id^="subject_"]');
        if (subjectLine) {
            if (post.floor === 0) {
                const textNode = subjectLine.childNodes[0];
                if (textNode) {
                    const prefix = textNode.textContent.match(/^(\[(?:二次元R18相关|三次元R18相关|全年龄正常向)\])/);
                    if (prefix) {
                        textNode.textContent = textNode.textContent.substring(prefix[1].length);
                        const classTag = newElem('span', 'rinsp-subject-class');
                        classTag.textContent = prefix[1];
                        subjectLine.insertBefore(classTag, textNode);
                    }
                }
            }
            if (threadTitle && post.floor > 0) {
                const subject = subjectLine.textContent.replace(/\s|\u00A0|&[a-z;]{0,5}$/g, '').trim();
                if (subject.startsWith('Re:') && threadTitle.startsWith(subject.substring(3))) {
                    subjectLine.parentNode.classList.add('rinsp-subject-redundant');
                }
            }
        }
        function addClickableLine(targetLine, floor, match) {
            const target = getFloorTarget(floor);
            addElem(targetLine, 'span').textContent = match[1];
            const link = addElem(targetLine, 'a', 'rinsp-subject-floor-link', { href: target.url, target: '_blank' });
            link.textContent = match[2];
            addElem(targetLine, 'span').textContent = match[3];
            if (target.targetElem) {
                link.addEventListener('click', evt => {
                    evt.preventDefault();
                    target.targetElem.scrollIntoView({
                        behavior: 'smooth',
                        block: 'start'
                    });
                });
            }
        }
        if (userConfig.addQuickJumpReSubjectLine) {
            if (subjectLine && subjectLine.childNodes.length === 1 && subjectLine.childNodes[0].nodeType === 3) {
                const match = subjectLine.textContent.match(/^((?:Re:)?回 )(\d+楼|楼主)(\(.+ 的帖子)$/);
                if (match) {
                    subjectLine.textContent = '';
                    const floor = Number.parseInt(match[2])||0;
                    addClickableLine(subjectLine, floor, match);
                    if (userConfig.hideRedundantReSubjectLine && floor === 0) {
                        subjectLine.classList.add('rinsp-subject-redundant');
                    }
                }
            }
            post.mainCell.querySelectorAll('.blockquote3 > .quote2 + div').forEach(quote => {
                if (quote.childNodes.length > 1 && quote.childNodes[1].nodeType === 1 && quote.childNodes[1].tagName === 'BR') {
                    const textNode = quote.childNodes[0];
                    if (textNode.nodeType === 3) {
                        const match = textNode.textContent.match(/^(引用 *)(第\d+楼|楼主)(.*)$/);
                        if (match) {
                            const floor = Number.parseInt(match[2].substring(1))||0;
                            const quoteLine = newElem('span');
                            addClickableLine(quoteLine, floor, match);
                            quote.childNodes[0].replaceWith(quoteLine);
                        }
                    }
                }
            });
        }
});
}

function getPostMetadata(postContent) {
    const row = postContent.closest('form > .t5.t2');
    if (row == null) return null;
    const link = row.querySelector('.tiptop a[href^="read.php?tid-"]');
    const match = link.getAttribute('href').match(/\?tid-(\d+)(?:-uid-(\d+))?\.html/);
    let tid = match[1] * 1;
    let uid;
    if (match[2] == null) {
        uid = Number.parseInt(link.closest('tr').querySelector('.user-pic a[href^="u.php?action-show-uid-"]').getAttribute('href').substring(22));
    } else {
        uid = match[2] * 1;
    }
    const pid = row.querySelector('.tpc_content > div[id^="read_"]').getAttribute('id').substring(5);
    return {
        tid, uid, pid
    };
}

function enhanceBuyButtons(userConfig, triggerRefresh) {
    
    document.querySelectorAll('span[id^="att_"]:not(.rinsp-att-enhanced) > a[href^="job.php?action-download-"]').forEach(function(attLink) {
        const root = attLink.parentNode;
        root.classList.add('rinsp-att-enhanced');
        const price = (root.textContent.match(/ 售价:(\d+)SP币/)||[])[1] * 1;
        if (Number.isNaN(price) || price <= 0) {
            return;
        }
        const href = attLink.getAttribute('href');
        attLink.dataset.price = String(price);
        attLink.removeAttribute('href');
        attLink.setAttribute('buyhref', href);
        root.classList.add('rinsp-att-sell');
        if (userConfig.enhanceSellFrame && price > 5) {
            root.classList.add('rinsp-att-sell-high');
        }
        attLink.addEventListener('click', () => {
            if (attLink.getAttribute('href'))
                return;
            if (price > 5) {
                let msg = `警告: 高额附件 ${price} SP 请确认购买?`;
                if (!confirm(msg)) {
                    return false;
                }
            }
            attLink.setAttribute('href', href);
        });
    });

    document.querySelectorAll('.quote.jumbotron > .s3 + .btn-danger:not(.rinsp-sell-relay-button):not(.rinsp-sell-enhanced)').forEach(function(buyButton) {
        const tpc = buyButton.closest('.tpc_content');
        const sellFrame = buyButton.closest('h6.jumbotron');
        buyButton.classList.add('rinsp-sell-enhanced');
        sellFrame.querySelectorAll('.rinsp-sell-relay-button').forEach(btn => btn.remove()); // remove any existing relay button and recreate again
        const relayButton = newElem('input', 'btn btn-danger rinsp-sell-relay-button', { type: 'button' });
        buyButton.parentNode.insertBefore(relayButton, buyButton);
        let priceMatch = sellFrame.querySelector('.s3').textContent.match(/此帖售价 (\d+) SP币,已有 (\d+) 人购买/);
        let price = priceMatch[1] * 1;
        if (userConfig.enhanceSellFrame) {
            if (price === 0) {
                relayButton.setAttribute('value', '免费');
                sellFrame.classList.add('rinsp-sell-free');
            } else if (price <= 5) {
                relayButton.setAttribute('value', `愿意购买, 售价 ${price} SP`);
                sellFrame.classList.add('rinsp-sell-5');
            } else if (price <= 100) {
                relayButton.setAttribute('value', `警告: 高额 ${price} SP 出售 (欢迎举报)`);
                sellFrame.classList.add('rinsp-sell-100');
            } else if (price >= 99999) {
                relayButton.setAttribute('value', '只楼主可见');
                sellFrame.classList.add('rinsp-sell-99999');
            } else {
                relayButton.setAttribute('value', `警告: 高额 ${price} SP 出售 (欢迎举报)`);
                sellFrame.classList.add('rinsp-sell-high');
            }
        } else {
            relayButton.setAttribute('value', buyButton.getAttribute('value'));
        }
        relayButton.addEventListener('click', evt => {
            evt.stopPropagation();
            if (price >= 99999) {
                return false;
            } else if (price > 5) {
                if (!confirm('高额出售 确认购买?')) {
                    return false;
                }
            }
            if (userConfig.buyRefreshFree) {
                relayButton.classList.add('rinsp-sell-buying');
                try {
                    const handled = executeBackgroundBuy(tpc, triggerRefresh);
                    if (handled) {
                        return false;
                    }
                } catch (err) {
                    console.error('exception when handling buy', err);
                }
            }
            // if failed, fallback to execute original buy action
            buyButton.click();
            return false;
        });
    });
}

function executeBackgroundBuy(tpc, triggerRefresh) {
    const pm = getPostMetadata(tpc);
    if (pm == null) {
        return false;
    }
    const {tid, uid, pid} = pm;
    const contentHref = `${document.location.origin}/read.php?tid-${tid}-uid-${uid}.html`; // one page should be enough to find the bought post
    const sellframes = [];
    tpc.querySelectorAll('.quote.jumbotron + blockquote').forEach(el => {
        el.innerHTML = '正在购买 ...';
        sellframes.push(el);
    });
    tpc.querySelectorAll('.quote.jumbotron').forEach(el => {
        el.innerHTML = '';
    });

    async function run() {
        let feedbackPage;
        while (true) {
            feedbackPage = await fetch(`${document.location.origin}/job.php?action=buytopic&tid=${tid}&pid=${pid}&verify=${verifyhash()}`, {
                method: 'GET',
                mode: 'same-origin',
                credentials: 'same-origin',
                headers: {
                    'Content-Type': 'application/x-www-form-urlencoded'
                }
            })
            .then(resp => resp.text());
            if (feedbackPage.indexOf('刷新不要快于') === -1) {
                break;
            }
            await sleep(1000);
        }
        if (feedbackPage.match(/>\s*操作完成\s*</)) {
            let html;
            while (true) {
                await sleep(1000);
                html = await fetch(contentHref, {
                    method: 'GET',
                    mode: 'same-origin',
                    credentials: 'same-origin',
                    headers: {
                        'Content-Type': 'application/x-www-form-urlencoded'
                    }
                })
                .then(resp => resp.text());
                if (html.indexOf('刷新不要快于') === -1) {
                    break;
                }
            }
            const doc = new DOMParser().parseFromString(html, 'text/html');
            let purchasedContent = doc.querySelector('#read_' + pid);
            if (purchasedContent) {
                const oldContent = document.querySelector('#read_' + pid);
                oldContent.parentNode.replaceChild(purchasedContent, oldContent);
            } else {
                throw Error('content outdated');
            }
        } else {
            const title = feedbackPage.match(/<title>([^-<]+)/);
            if (title) {
                const reason = title[1].trim();
                sellframes.forEach(el => {
                    el.classList.add('rinsp-buy-failed');
                    el.textContent = reason;
                });
            }
        }
        triggerRefresh();
    }
    run().catch(e => {
        console.info('buy-refresh-free ERR', e);
        window.location.reload();
    });
    return true;
}

function autoCheckReply() {
    let cb = document.querySelector('form[action="post.php?"] input[name="atc_newrp"]');
    if (cb) {
        cb.checked = true;
    }
}

async function addUserPinArea(pinnedUsersConfigAccess, hideInactivePinnedUsers, showResourceSpotsFloating) {
    document.addEventListener('click', evt => {
        if (evt.target && evt.target.classList.contains('rinsp-button-placeholder-pinuser')) {
            const uid = evt.target.dataset.uid * 1;
            const nickName = evt.target.dataset.nickname;
            const img = evt.target.dataset.img;
            if (pinnedUsers.has(uid)) {
                unwatchUser(uid);
            } else {
                watchUser(uid, nickName, img);
            }
        }
    });

    const pinMain = addElem(document.body, 'div', 'rinsp-pinuser-main');
    let spotsMain = null;
    if (showResourceSpotsFloating) {
        spotsMain = addElem(document.body, 'div', 'rinsp-spots-main');
    }
    let spotsBar = null;
    const threadHeaderAnchor = document.querySelector('#main > .h2 > .fr.w');
    if (threadHeaderAnchor) {
        const targetControlBar = threadHeaderAnchor.parentNode;
        targetControlBar.classList.add('rinsp-thread-header');
        spotsBar = newElem('div', 'rinsp-spots-bar');
        targetControlBar.insertBefore(spotsBar, targetControlBar.childNodes[0]);
    }

    let updateListeners = [];
    const pinnedUsers = new Map([[0, null]]); // 0 entry marks it out of date
    const localWatchUsers = new Map();
    const locationHighlights = [];

    async function update(notifyListeners) {
        if (pinnedUsers.has(0)) {
            const pinnedUsersConfig = await pinnedUsersConfigAccess.read();
            pinnedUsers.clear();
            Object.values(pinnedUsersConfig.users).forEach(record => {
                pinnedUsers.set(record.uid, record);
            });
        }
        pinMain.innerHTML = '';
        if (spotsMain) {
            spotsMain.innerHTML = '';
        }
        if (spotsBar) {
            spotsBar.innerHTML = '';
        }
        function createSpotButton(spot) {
            let locButton = newElem('a', null, { href: spot.href||'javascript:', target: '_blank' });
            if (spot.floor != null) {
                const flootLabel = spot.floor === 0 ? 'GF' : '' + spot.floor + 'F';
                addElem(locButton, 'span', 'rinsp-spot-floor').textContent = flootLabel;
            }
            const pair = spot.type.split(':', 2);
            addElem(locButton, 'span', 'rinsp-spot-icon').textContent = pair[0];
            addElem(locButton, 'span', 'rinsp-spot-label').textContent = pair[1];
            locButton.addEventListener('click', evt => {
                evt.preventDefault();
                let alignment = spot.alignment||'center';
                if (alignment === 'center') {
                    let diff = spot.elem.getBoundingClientRect().height - window.screen.availHeight;
                    if (diff > window.screen.availHeight / 3) { // if target is larger than 2/3 of the screen, align top
                        alignment = 'start';
                    }
                }
                if (alignment === 'start') {
                    window.scrollTo({
                        behavior: 'smooth',
                        top: spot.elem.getBoundingClientRect().top - document.body.getBoundingClientRect().top - 50
                    });
                } else {
                    spot.elem.scrollIntoView({
                        behavior: 'smooth',
                        block: alignment
                    });
                }
            });
            return locButton;
        }
        if (locationHighlights.length > 0) {
            if (spotsMain) {
                const list = addElem(spotsMain, 'div', 'rinsp-highlight-spots');
                addElem(list, 'b').textContent = '焦点';
                const dl = addElem(list, 'dl');
                locationHighlights.forEach(spot => {
                    let locItem = addElem(dl, 'dt');
                    locItem.appendChild(createSpotButton(spot));
                });
            }
            if (spotsBar) {
                locationHighlights.forEach(spot => {
                    spotsBar.appendChild(createSpotButton(spot));
                });
            }
        }
        const timeComparator = comparator('time');
        const pinnedRecords = Array.from(pinnedUsers.values()).filter(record => !hideInactivePinnedUsers || record.locations && record.locations.length > 0).sort(timeComparator);
        if (pinnedRecords.length + localWatchUsers.size > 0) {
            const list = addElem(pinMain, 'div', 'rinsp-pinuser-list');
            if (pinnedRecords.length > 0) {
                addElem(list, 'b').textContent = '焦点人物';
                const listitems = addElem(list, 'div');
                pinnedRecords.forEach(record => {
                    renderUserRecord(listitems, record);
                });
            }
            if (localWatchUsers.size > 0) {
                addElem(list, 'b').textContent = '活跃人物';
                const listitems = addElem(list, 'div');
                Array.from(localWatchUsers.values())
                    .sort((x, y) => {
                        const xc = (x.locations||[]).length;
                        const yc = (y.locations||[]).length;
                        if (xc > yc)
                            return -1;
                        if (xc < yc)
                            return 1;
                        return timeComparator(x, y);
                    })
                    .forEach(record => {
                        renderUserRecord(listitems, record);
                    });
            }
        }
        if (notifyListeners) {
            updateListeners.forEach(listener => {
                listener();
            });
        }

        function renderUserRecord(parent, record) {
            const item = addElem(parent, 'div', 'rinsp-pinuser-item');
            const face = addElem(item, 'div', 'rinsp-pinuser-item-face');
            const locs = addElem(item, 'dl', 'rinsp-pinuser-item-locs');

            const img = addElem(face, 'img', null, { src: record.img });
            const isDefaultImg = record.img == null || record.img.startsWith('images/face/');
            const label = addElem(face, 'div');
            addElem(label, 'a', null, { href: `u.php?action-show-uid-${record.uid}.html`, target: '_blank' }).textContent = record.nickName||`#${record.uid}`;
            const unpin = addElem(label, 'div', 'rinsp-pinuser-unpin-icon');
            unpin.addEventListener('click', () => {
                unwatchUser(record.uid);
            });
            const locCount = record.locations ? record.locations.length : 0;
            if (locCount > 0) {
                const locButtons = [];
                for (let loc of record.locations) {
                    let locItem = addElem(locs, 'dt', loc.style ? 'rinsp-pinuser-loc-' + loc.style : null);
                    let locButton = addElem(locItem, 'a', null, { href: 'javascript:' });
                    locButton.textContent = loc.label;
                    locButton.addEventListener('click', loc.action);
                    locButtons.push(locButton);
                }
                let lastIndex = -1;
                img.addEventListener('click', () => {
                    lastIndex += 1;
                    if (lastIndex >= locButtons.length) {
                        lastIndex = 0;
                    }
                    locButtons[lastIndex].click();
                });
                if (isDefaultImg) {
                    const estHeight = Math.max(locCount * 14, 32);
                    if (estHeight < 100) {
                        img.setAttribute('style', `max-height:${estHeight}px; object-fit:contain;background:#FFF`);
                    }
                }
            } else {
                item.classList.add('rinsp-pinuser-item-absent');
            }
        }

    }

    async function unwatchUser(uid) {
        await pinnedUsersConfigAccess.update(function(updatingPinnedUsersConfig) {
            delete updatingPinnedUsersConfig.users['#' + uid];
            // clean up corrupt data ...
            delete updatingPinnedUsersConfig.users['#NaN'];
            delete updatingPinnedUsersConfig.users['#null'];
            return updatingPinnedUsersConfig;
        });
        pinnedUsers.set(0, null);
        await update(true);
    }

    async function watchUserLocally(uid, nickName, img) {
        if (pinnedUsers.has(uid)) {
            return;
        }
        localWatchUsers.set(uid, {
            uid, nickName, img, time: Date.now(), locations: []
        });
    }

    async function watchUser(uid, nickName, img) {
        await pinnedUsersConfigAccess.update(function(updatingPinnedUsersConfig) {
            updatingPinnedUsersConfig.users['#' + uid*1] = {
                uid, nickName, img, time: Date.now(), locations: []
            };
            return updatingPinnedUsersConfig;
        });
        pinnedUsers.set(0, null);
        await update(true);
    }

    function guessInterestingContent(text, spotsFilter) {
        const textLower = text.toLowerCase();
        if (spotsFilter.shares) {
            if (textLower.indexOf('pan.baidu.com/') != -1) {
                return '📂:分享 (度盘)';
            }
            if (textLower.indexOf('pan.quark.cn/') != -1) {
                return '📂:分享 (夸克)';
            }
            if (textLower.indexOf('/mypikpak.com/s/') != -1) {
                return '📂:分享 (PP)';
            }
            if (textLower.indexOf('://mega.nz/') != -1) {
                return '📂:分享 (MEGA)';
            }
            if (textLower.indexOf('/drive.google.com/') != -1) {
                return '📂:分享 (谷歌)';
            }
            if (textLower.indexOf('/1drv.ms/f/s!') != -1 ||
                textLower.match(/:\/\/[a-zA-Z-]+\.sharepoint\.com\/:/) ||
                textLower.indexOf('-my.sharepoint.com/:') != -1) {
                return '📂:分享 (OD)';
            }
            if (textLower.indexOf('.lanzoub.com') != -1 ||
                textLower.indexOf('/share.weiyun.com/') != -1 ||
                textLower.indexOf('.aliyundrive.com') != -1 ||
                textLower.indexOf('.123pan.com/') != -1 ||
                textLower.indexOf('pan.xunlei.com/') != -1 ||
                textLower.indexOf('cowtransfer.com/s/') != -1 ||
                textLower.indexOf('/files.catbox.moe/') != -1 ||
                textLower.indexOf('/litter.catbox.moe/') != -1 ||
                textLower.indexOf('/pixeldrain.com/l/') != -1 ||
                textLower.indexOf('/pixeldrain.com/u/') != -1 ||
                textLower.indexOf('/gofile.io/d/') != -1 ||
                textLower.indexOf('/uploadhaven.com/download/') != -1 ||
                textLower.indexOf('/racaty.com/') != -1 ||
                textLower.indexOf('.swisstransfer.com/d/') != -1 ||
                textLower.indexOf('.gigafile.nu/') != -1 ||
                textLower.indexOf('/sakuradrive.com/s/') != -1 ||
                textLower.indexOf('/dogpan.com/s/') != -1) {
                return '📂:分享';
            }
            if (textLower.match(/https?:\/\/([a-z0-9]+\.)?[a-z0-9]+\.(com|net|io)\/s\//)) {
                return '📂:分享';
            }
            if (textLower.match(/(?:[^a-zA-Z0-9]|^)(?:bafy[a-z0-9]{55}|Qm[A-Za-z0-9]{44})(?:[^a-zA-Z0-9]|$)/)) {
                return '🧲:ipfs';
            }
            if (textLower.indexOf('magnet:?') != -1) {
                return '🧲:磁链';
            }
            if (textLower.indexOf('ed2k://') != -1) {
                return '🧲:ed2k';
            }
            if (textLower.indexOf('thunder://') != -1) {
                return '🧲:迅雷';
            }
            if (textLower.indexOf('/jmj.cc/s/') != -1 ||
                textLower.indexOf('.feimaoyun.com/') != -1 ||
                textLower.indexOf('/rosefile.net/') != -1 ||
                textLower.indexOf('.xun-niu.com/file-') != -1 ||
                textLower.indexOf('.xunniu-pan.com/file-') != -1) {
                return '💲:网赚盘';
            }
        }
    }

    async function offerPostContent(post, spotsFilter, likePatternMatcher) {
        if (spotsFilter.likes && likePatternMatcher) {
            scanTextualContent(childText => {
                const matches = likePatternMatcher.match(childText.toLowerCase());
                if (matches.size > 0) {
                    return '💚:' + Array.from(matches)[0];
                } else {
                    return null;
                }
            });
        }

        if (spotsFilter.sells) {
            const sellArea = post.contentElem.querySelector('.quote.jumbotron > .s3 + .btn-danger');
            if (sellArea) {
                const jumbotron = sellArea.closest('.jumbotron');
                let type = '💰:出售框';
                if (jumbotron.classList.contains('rinsp-sell-99999')) {
                    type = '🔒:密享';
                }
                locationHighlights.push({
                    floor: post.floor,
                    type: type,
                    elem: jumbotron
                });
                return;
            }
        }

        const found = scanTextualContent(childText => guessInterestingContent(childText, spotsFilter));
        if (found) {
            return;
        }

        let links = post.contentElem.querySelectorAll('a[href]:not(.rinsp-subject-floor-link)');
        for (let link of links) {
            let hit = guessInterestingContent(link.getAttribute('href'), spotsFilter);
            if (hit) {
                locationHighlights.push({
                    floor: post.floor,
                    type: hit,
                    elem: link
                });
                return;
            }
        }

        if (spotsFilter.images) {
            for (let img of post.contentElem.closest('.tpc_content').querySelectorAll('img[src],img[data-rinsp-defer-src]')) {
                let src = getImgSrc(img);
                if (src.startsWith('images/')) {
                    continue; // skip default emotions
                }
                if (!img.classList.contains('rinsp-img-loading') && img.naturalWidth > 0 && img.naturalHeight > 0 && img.naturalWidth < 64 && img.naturalHeight < 64) {
                    continue; // skip tiny images
                }
                locationHighlights.push({
                    floor: post.floor,
                    type: '🖼️:图片',
                    elem: img,
                    alignment: 'start'
                });
                return;
            }
            for (let link of links) {
                let href = link.getAttribute('href');
                if (href.match(/\.(png|jpg|jpeg|gif|avif|webp)$/)) {
                    locationHighlights.push({
                        floor: post.floor,
                        type: '🖼️:图片',
                        elem: link,
                        alignment: 'start'
                    });
                    return;
                }
            }
        }


        if (spotsFilter.attachments) {
            const attFile = post.rootElem.querySelector('a[href^="job.php?action-download-"]');
            if (attFile) {
                locationHighlights.push({
                    floor: post.floor,
                    type: '📄:附件',
                    elem: attFile
                });
                return;
            }
        }

        if (spotsFilter.sells) {
            const sellArea = post.contentElem.querySelector('h6.quote.jumbotron + blockquote.jumbotron');
            if (sellArea) {
                let type;
                if (sellArea.textContent.startsWith('若发现会员采用欺骗的方法获取财富')) {
                    type = '💰:出售框';
                } else {
                    type = '✔️:已购买';
                }

                locationHighlights.push({
                    floor: post.floor,
                    type: type,
                    elem: sellArea
                });
                return;
            }
        }

        if (spotsFilter.links) {
            for (let link of links) {
                if (link.querySelector('img') == null || link.textContent.length > 0) {
                    locationHighlights.push({
                        floor: post.floor,
                        type: '🔗:链接',
                        elem: link
                    });
                    return;
                }
            }
        }

        function scanTextualContent(handler) {
            let lastNode = null;
            for (let childNode of post.contentElem.childNodes) {
                let childText;
                if (childNode.nodeType === 3) {
                    childText = childNode.textContent.trim();
                } else if (childNode.nodeType === 1) {
                    childText = childNode.textContent.trim();
                    lastNode = childNode;
                }
                if (childText) {
                    let hit = handler(childText, spotsFilter);
                    if (hit) {
                        if (lastNode) {
                            locationHighlights.push({
                                floor: post.floor,
                                type: hit,
                                elem: lastNode
                            });
                        } else {
                            locationHighlights.push({
                                floor: post.floor,
                                type: hit,
                                elem: post.rootElem,
                                alignment: 'start'
                            });
                        }
                        return true;
                    }
                }
            }
            return false;
        }

    }

    await update();
    return {
        addUpdateListener(listener) {
            updateListeners.push(listener);
        },
        fireUpdateListeners() {
            updateListeners.forEach(listener => {
                listener();
            });
        },
        getWatchCount() {
            return pinnedUsers.size + localWatchUsers.size;
        },
        isPinned(uid) {
            return pinnedUsers.has(uid);
        },
        getPinned(uid) {
            const record = pinnedUsers.get(uid);
            if (record) {
                return {
                    uid: uid,
                    nickName: record.nickName,
                    img: record.img
                };
            } else {
                return null;
            }
        },
        isWatching(uid) {
            return pinnedUsers.has(uid) || localWatchUsers.has(uid);
        },
        clearState() {
            for (let record of pinnedUsers.values()) {
                record.locations = [];
            }
            localWatchUsers.clear();
            locationHighlights.length = 0;
        },
        addLocation(uid, label, locClass, action) {
            let record = pinnedUsers.get(uid);
            if (record == null) {
                record = localWatchUsers.get(uid);
            }
            if (record) {
                record.locations.push({ label, style: locClass, action });
            }
        },
        unwatchUser,
        watchUser,
        watchUserLocally,
        offerPostContent,
        addLocationHighlight(floor, type, elem, href) {
            locationHighlights.push({
                floor,
                type,
                elem,
                href
            });
        },
        update
    };
}

const TITLE_SPLIT_PATTERN = /[\u3000-\u303F()/:,\s\[\]\{\}\(\)\\\/<>\u00A0➕\+]+|(?<![妙秒喵])[轉传]|原档|原檔|分享|下载|下載|长期|長期|直[链连連]接?|(?<![妙秒喵])[链连連]接?|[网網云雲][盘盤]|[盘盤云雲]|有效期?|(?<![0-9a-z])\d+(?:\.\d{1,2})?(?:[GMKT](?:i?B)?)|(?<![0-9a-z])\d{1,3}(?:[日天]|个?月)/i;
const SHARE_TYPES = [
    {
        name: '度盘',
        pattern: /(?:百?度(?:[标標][准準])?[长長短]?[盘盤云雲链连連]|百?度娘|毒[盘盤云雲链连連]|[妙秒喵][传傳链连連])+|^(?:bd|百度)$/i,
    },
    {
        name: 'OD',
        pattern: /(?:onedrive|one drive|微软盘)|^(?:OD|微软)$/i,
    },
    {
        name: 'Google盘',
        pattern: /(?:GoogleDrive|Google Drive)|(?:GD|Google|谷歌)$/i,
    },
    {
        name: 'MEGA',
        pattern: /(?:mega|mg)[盘盤云雲]|^(?:mega|mg)$/i,
    },
    {
        name: '樱盒',
        pattern: /(?:樱花?[盒盘盤云雲])|^(?:樱花)$/i,
    },
    {
        name: '微云',
        pattern: /(?:微[云雲][盘盤]?)/,
    },
    {
        name: '移动云',
        pattern: /(?:移动[云雲][盘盤]?)/,
    },
    {
        name: '夸克',
        pattern: /(?:夸克|Quack|夸[盘盤云雲])/,
    },
    {
        name: '阿里云',
        pattern: /(?:阿里|aliyun|aliyundrive)[网網]?[盘盤云雲]?|ali[网網]?[盘盤云雲]|^(?:阿里|aliyun|ali)$/i,
    },
    {
        name: 'mediafire',
        pattern: /^(?:mediafire)$/i,
    },
    {
        name: 'ipfs',
        pattern: /^(?:ipfs)$/i,
    },
    {
        name: 'BT',
        pattern: /^(?:磁[力链鏈]|bt|bittorrent|torrent)$/i,
    },
    {
        name: 'TERABOX',
        pattern: /(?:terabox)/i,
    },
    {
        name: 'catbox',
        pattern: /(?:catbox|[猫喵][猫喵]?盒)/i,
    },
    {
        name: 'PikPak',
        pattern: /(?:pikpak|p[网網]?[盘盤云雲])|^(?:pp)$/i,
    },
    {
        name: 'pixeldrain',
        pattern: /(?:pixeldrain)|^(?:pd)$/i,
    },
    {
        name: 'gofile',
        pattern: /(?:gofile)/i,
    },
    {
        name: '115',
        pattern: /^(?:ed2k|115ed2k|115)$/i,
    },
    {
        name: 'RF',
        pattern: /(?:rosefile)|^(?:RF)$/i,
    },
    {
        name: '飛猫',
        pattern: /^(?:飛[猫喵]|FM)$/i,
    },
    {
        name: '迅牛',
        pattern: /^(?:迅牛|XN)$/i,
    },
    {
        name: '不明',
        pattern: /^(?:多空|多[盘盤])$/,
    },
];
const SHARE_TYPES_MIXED_SPLIT_PATTERNS = /(bd|od|gd|mg)/i;
const SHARE_UNKNOWN_NAME = '不明';

async function addShareTypeFilterArea(target, sharetypeFilterConfigAccess) {
    let updateListeners = [];
    const controlBlock = addElem(target, 'div', 'rinsp-sharetype-filter');
    const states = new Map();
    const sharetypeFilterConfig = await sharetypeFilterConfigAccess.read();
    SHARE_TYPES.map(shareTypeDef => shareTypeDef.name).forEach(shareType => {
        const toggler = addElem(controlBlock, 'div', 'rinsp-sharetype-item');
        toggler.textContent = shareType;
        const model = {
            count: 0,
            toggler,
            disabled: sharetypeFilterConfig.hides.indexOf(shareType) !== -1
        };
        states.set(shareType, model);
        toggler.addEventListener('click', async () => {
            model.disabled = !model.disabled;
            sharetypeFilterConfigAccess.update(function(updatingSharetypeFilterConfigAccess) {
                const set = new Set(updatingSharetypeFilterConfigAccess.hides||[]);
                if (model.disabled) {
                    set.add(shareType);
                } else {
                    set.delete(shareType);
                }
                updatingSharetypeFilterConfigAccess.hides = Array.from(set);
                return updatingSharetypeFilterConfigAccess;
            });
            update(true);
        });
    });
    const clearButton = addElem(controlBlock, 'img', 'rinsp-sharetype-filter-clear', { src: '/images/close.gif'});
    clearButton.addEventListener('click', async () => {
        states.forEach(model => {
            model.disabled = false;
        });
        sharetypeFilterConfigAccess.update(function(updatingSharetypeFilterConfigAccess) {
            updatingSharetypeFilterConfigAccess.hides = [];
            return updatingSharetypeFilterConfigAccess;
        });
        update(true);
    });

    async function update(notifyListeners) {
        let disabledCount = 0;
        states.forEach(state => {
            state.toggler.setAttribute('count', state.count);
            state.toggler.setAttribute('disabled', state.disabled ? '1' : '0');
            if (state.disabled) {
                disabledCount++;
            }
        });
        if (notifyListeners) {
            updateListeners.forEach(listener => {
                listener();
            });
        }
        clearButton.setAttribute('disabled', disabledCount === 0 ? '1' : '0');
    }

    await update();
    return {
        addUpdateListener(listener) {
            updateListeners.push(listener);
        },
        clearState() {
            states.forEach(entry => entry.count = 0);
        },
        addAndAcceptThread(tid, title) {
            const shares = [];
            const parts = title.split(TITLE_SPLIT_PATTERN);
            parts.push(title);
            nextpart:
            for (let part of parts) {
                for (let shareTypeDef of SHARE_TYPES) {
                    if (part.match(shareTypeDef.pattern)) {
                        shares.push(shareTypeDef.name);
                        continue nextpart;
                    }
                }
                if (title.length > part.length) {
                    const splits = part.split(SHARE_TYPES_MIXED_SPLIT_PATTERNS);
                    if (splits.length > 1) {
                        for (let i = 0; i < splits.length; i += 2) {
                            if (splits[i]) { // any additional content between known share names invalidates the whole part
                                continue nextpart;
                            }
                        }
                        nextsubpart:
                        for (let i = 1; i < splits.length; i += 2) {
                            for (let shareTypeDef of SHARE_TYPES) {
                                if (splits[i].match(shareTypeDef.pattern)) {
                                    shares.push(shareTypeDef.name);
                                    continue nextsubpart;
                                }
                            }
                        }
                    }
                }
            }
            if (shares.length === 0) {
                shares.push(SHARE_UNKNOWN_NAME);
            }

            let accepted = false;
            shares.forEach(shareType => {
                const entry = states.get(shareType);
                entry.count++;
                if (!entry.disabled) {
                    accepted = true;
                }
            });
            return accepted;
        },
        update
    };
}

function readEnhanceUserInfoMap(myUserId, userConfig) {
    let map = new Map();
    document.querySelectorAll('.r_two .user-info > .user-infoWrap > .c + .f12').forEach(function(uidCell) {
        let uid = uidCell.textContent.trim() * 1;
        const img = uidCell.closest('.r_two').querySelector(`.user-pic a[href="u.php?action-show-uid-${uid}.html"] > img`);
        let imgSrc = null;
        if (img) {
            imgSrc = getImgSrc(img);
        }
        let infoWrap = uidCell.closest('.user-infoWrap');
        let infoText = infoWrap.textContent;
        let nickName = ((infoText.match(/\s昵称: *([^\r\n]+)在线时间/)||[])[1]||'').trim()||'';
        let unnamed = !!infoText.match(/昵称:\s*在线时间/);

        let m = infoText.match(/\s发帖: (\d+)\s+HP: (-?\d+) 点\s+SP币: (-?\d+) G\s+.*\s+在线时间: (\d+)\(小时\)\s+注册时间: (\d{4}-\d{2}-\d{2})\s*最后登录: (\d{4}-\d{2}-\d{2})\s/);
        let data = {
            post: m[1] * 1,
            hp: m[2] * 1,
            sp: m[3] * 1,
            online: m[4] * 1,
            register: m[5],
            login: m[6],
            img: imgSrc,
            nickName,
            unnamed
        };
        map.set(uid, data);

        if (userConfig.useCustomUserInfoPopup && infoWrap.querySelector('.listread > .rinsp-user-popup-action-list') == null) {
            const actionList = infoWrap.querySelector('.listread > ul');
            const actionItems = actionList.querySelectorAll('li');
            const newActionList = addElem(actionList.parentElement, 'ul', 'rinsp-user-popup-action-list');

            // read profile
            actionItems[0].querySelector('a').setAttribute('target', '_blank');
            newActionList.appendChild(actionItems[0]);

            if (myUserId !== uid) {
                // send mail
                const mailToItem = addElem(newActionList, 'li', 'rinsp-user-popup-action-mailto', { title: '发短消息' });
                addElem(mailToItem, 'a', null, { target: '_blank', href: `message.php?action-write-touid-${uid}.html` }).innerHTML = '<div></div>';
            }

            // show topics
            const showTopicItem = addElem(newActionList, 'li', 'rinsp-user-popup-action-topics', { title: '主题列表' });
            addElem(showTopicItem, 'a', null, { target: '_blank', href: `u.php?action-topic-uid-${uid}.html` }).innerHTML = '<div></div>';
   
            if (myUserId !== uid) {
                // add friend (and add confirmation)
                const addFriendAction = actionItems[2].querySelector('a');
                addFriendAction.setAttribute('onclick', "if (confirm('加为好友 ?')) " + addFriendAction.getAttribute('onclick'));
                newActionList.appendChild(actionItems[2]);
            }

            // show topics
            const pinUserItem = addElem(newActionList, 'li', 'rinsp-user-popup-action-pinuser', { title: '设为焦点' });
            const pinButton = addElem(pinUserItem, 'a', null, { href: 'javascript:' });
            const dataNode = addElem(pinButton, 'div', 'rinsp-button-placeholder-pinuser');
            dataNode.dataset.uid = uid;
            dataNode.dataset.nickname = nickName;
            const img = document.querySelector(`.user-pic a[href="u.php?action-show-uid-${uid}.html"] > img`);
            if (img) {
                dataNode.dataset.img = getImgSrc(img);
            }
    
            actionList.remove();
        }

    });
    return map;
}

function getCurrentPageFid() {
    const item = document.querySelector('#breadcrumbs .crumbs-item.current > strong > a[href^="thread.php?fid-"]') || Array.from(document.querySelectorAll('#breadcrumbs .crumbs-item[href^="thread.php?fid-"]')).pop() || null;
    if (item) {
        return Number.parseInt(item.getAttribute('href').substring(15));
    }
    return null;
}

function showAutoSpTaskStatus(myUserId, userConfig, mainConfigAccess) {

    const headCell = Array.from(document.querySelectorAll('td > .t > table td.h[colspan="4"]')).filter(el => el.textContent === '社区论坛任务')[0];
    if (headCell == null) {
        return;
    }
    const block = headCell.closest('.t').closest('td');

    const autoSpOptionBlock = addElem(block, 'div', null, { style: 'padding: 0.5em' });

    const labelElem = addElem(autoSpOptionBlock, 'label');
    const checkbox = addElem(labelElem, 'input', null, { type: 'checkbox' });
    checkbox.checked = !!userConfig.autoSpTasks;
    addElem(labelElem, 'span', null, { style: 'vertical-align: middle; margin-right: 0.5em' }).textContent = '自动签到 (社区论坛任务)';
    checkbox.addEventListener('change', () => {
        mainConfigAccess
            .update(updatingUserConfig => {
                updatingUserConfig.autoSpTasks = checkbox.checked;
                return updatingUserConfig;
            })
            .catch(ex => alert('' + ex))
            .then(() => document.location.reload());
    });

    if (userConfig.autoSpTasks) {
        const taskRecordAccess = initConfigAccess(myUserId, 'autosptask', {});
        const autoSpTasksSummary = addElem(autoSpOptionBlock, 'p');
        taskRecordAccess.read().then(taskRecord => {
            if (taskRecord.lastCompleteSp > 0) {
                const hr = (Date.now() - taskRecord.lastComplete) / 3600000;
                autoSpTasksSummary.textContent = `💡 ${getAgeString(hr * 60)} 已领取 ${taskRecord.lastCompleteSp||'?'} SP | 记录中总共领取 ${taskRecord.totalSp||'?'} SP`;
            }
        });
    }
}

function autoCompleteSpTasks(myUserId) {
    const taskMenuButton = document.querySelector('#guide a[href="plugin.php?H_name-tasks.html"]');
    const taskRecordAccess = initConfigAccess(myUserId, 'autosptask', {});
    async function execute() {
        const now = Date.now();
        let taskRecord = await taskRecordAccess.read();
        if ((Math.max(taskRecord.lastChecked, taskRecord.lastComplete, taskRecord.runningUntil) - now) / 3600000 > 23) {
            // if any time record is ahead of time for more than 23 hour, reset
            taskRecord = await taskRecordAccess.update(newRecord => {
                delete newRecord.lastChecked;
                delete newRecord.lastComplete;
                delete newRecord.lastCompleteSp;
                delete newRecord.runningUntil;
                return newRecord;
            });

        }

        if (taskRecord.lastComplete && now > taskRecord.lastComplete) {
            const hr = (now - taskRecord.lastComplete) / 3600000;
            if (taskMenuButton && taskRecord.lastCompleteSp > 0) {
                taskMenuButton.setAttribute('title', `${getAgeString(hr * 60)} 已领取 ${taskRecord.lastCompleteSp} SP`);
            }
            if (hr < 11) {
                // min. check interval from last completed task is 11 hour
                return;
            }
        }
        if (taskRecord.lastChecked && now > taskRecord.lastChecked && (now - taskRecord.lastChecked) < 3600000) {
            // min. check interval is 1 hour
            return;
        }

        if (taskRecord.runningUntil > now) {
            return;
        }

        await taskRecordAccess.update(newRecord => {
            newRecord.runningUntil = Date.now() + 10000; // pause other pages for 10 seconds
            return newRecord;
        });

        const newTaskDoc = await fetchGetPage(`${document.location.origin}/plugin.php?H_name-tasks.html`);
        const newTasks = await scanAndRunAllTasks(newTaskDoc, 'a[onclick^="startjob(\'"][title=按这申请此任务]', 'job');
        await taskRecordAccess.update(newRecord => {
            newRecord.runningUntil = Date.now() + 10000;
            return newRecord;
        });
        if (newTasks.length > 0) {
            await sleep(1500);
        }
        const currentTaskDoc = await fetchGetPage(`${document.location.origin}/plugin.php?H_name-tasks-actions-newtasks.html.html`);
        const completedTasks = await scanAndRunAllTasks(currentTaskDoc, 'a[onclick^="startjob(\'"][title=领取此奖励]', 'job2');
        const sp = completedTasks.reduce((acc, tk) => acc + (tk.sp||0), 0);
        await taskRecordAccess.update(newRecord => {
            const now = Date.now();
            newRecord.totalSp = (newRecord.totalSp || 0) + sp;
            newRecord.lastChecked = now;
            // if just complete a task
            if (completedTasks.length > 0) {
                newRecord.lastComplete = now;
                newRecord.lastCompleteSp = sp;
            }
            delete newRecord.runningUntil;
            return newRecord;
        });
        if (sp > 0) {
            if (taskMenuButton) {
                taskMenuButton.setAttribute('rinsp-sp-get', '' + sp);
            }
        }
        setTimeout(execute, 3600000); // 1 hour
    }
    
    async function scanAndRunAllTasks(doc, selector, actionType) {
        let completed = [];
        for (let taskButton of doc.querySelectorAll(selector)) {
            const rewardMatch = taskButton.closest('td').previousElementSibling.textContent.match(/奖励 : SP币 (\d+) G/);
            const sp = rewardMatch ? rewardMatch[1] * 1 : 0;
            const jobId = Number.parseInt(taskButton.getAttribute('onclick').substring(10));
            const taskURL = `${document.location.origin}/plugin.php?H_name=tasks&action=ajax&actions=${actionType}&cid=${jobId}&nowtime=${Date.now()}&verify=${verifyhash()}`;
            await sleep(100);
            const resp = await fetch(taskURL, {
                method: 'GET',
                mode: 'same-origin',
                credentials: 'same-origin',
                headers: {
                    'Content-Type': 'text/html'
                },
                cache: 'no-cache'
            });
            if (resp.ok) {
                const xml = await resp.text();
                if (xml.indexOf('success\t')) {
                    // success
                    completed.push({sp});
                }
            }
        }
        return completed;
    }
    setTimeout(execute, 0);
}

function addSearchShortcut() {
    function createSearchButton(fid) {
        const button = newElem('a', 'rinsp-fid-search-button', { href: `search.php#prefill("fid",${fid})` });
        return button;
    }
    document.querySelectorAll('#breadcrumbs').forEach(breadcrumbs => {
        // how comes the same id used by multiple elements ...

        const currentThreadLink = breadcrumbs.querySelector('.crumbs-item.current > strong > a[href^="thread.php?fid-"]');
        if (currentThreadLink) {
            const fid = Number.parseInt(currentThreadLink.getAttribute('href').substring(15));
            currentThreadLink.closest('.crumbs-item').prepend(createSearchButton(fid));
        } else {
            const deepestThreadLink = Array.from(breadcrumbs.querySelectorAll('.crumbs-item[href^="thread.php?fid-"]')).pop();
            if (deepestThreadLink) {
                const fid = Number.parseInt(deepestThreadLink.getAttribute('href').substring(15));
                deepestThreadLink.prepend(createSearchButton(fid));
            }
        }
    });
}

function setAreaScoped(target, fid) {
    target.classList.add('rinsp-area-scoped-item');
    target.classList.add(`rinsp-area-scoped-item-${fid}`);
}

const specialAreaNames = {
    171: 'CG资源 (网赚)',
    172: '实用动画 (网赚)',
    173: '实用漫画 (网赚)',
    174: '游戏资源 (网赚)',
};

const PIC_WALL_PREF_KEY = 'rinsp:pic-wall-fids';
const LAST_AREA_FILTER_MEMORY_KEY = 'rinsp:last-filter-fid';
const REQUEST_RATE_RECORD_KEY = 'rinsp:request-rate';

function defaultAreaFilterHandler(fid) {
    const targetClass = `rinsp-area-scoped-item-${fid}`;
    document.querySelectorAll('.rinsp-area-scoped-item').forEach(item => {
        if (fid === 0 || item.classList.contains(targetClass)) {
            item.classList.remove('rinsp-area-scoped-item-hidden');
        } else {
            item.classList.add('rinsp-area-scoped-item-hidden');
        }
    });
}

function addAreaFilter(baseAreaMap, userId, filterHandler) {
    return addSectionFilter(baseAreaMap, userId, filterHandler, {
        nameMappings: specialAreaNames,
        heading: '版块分类',
        showAllLabel: '全部版块'
    });
}

function addSectionFilter(baseAreaMap, userId, filterHandler, options) {
    const areaMap = new Map(baseAreaMap);
    let basePageKey = document.location.href.replace(/(-page-\d+)?\.html$/, '');
    if (basePageKey.indexOf('-uid-') === -1) {
        basePageKey = basePageKey + '-uid-' + userId;
    }
    const specialNameMappings = options.nameMappings||{};
    Object.keys(specialNameMappings).forEach(spFid => {
        if (areaMap.get(spFid * 1)) {
            areaMap.set(spFid * 1, specialNameMappings[spFid]);
        }
    });

    let defaultFid = 0;
    let defaultFilterItem = null;
    try {
        const lastState = JSON.parse(localStorage.getItem(LAST_AREA_FILTER_MEMORY_KEY) || 'null');
        if (lastState != null && lastState.key === basePageKey) {
            areaMap.set(lastState.fid, lastState.areaName);
            defaultFid = lastState.fid;
        }
    } catch (ex) {}

    const sidePanel = document.querySelector('#u-contentside');
    sidePanel.querySelectorAll('.rinsp-area-filter-panel').forEach(el => el.remove());
    const section = addElem(sidePanel, 'div', 'rinsp-area-filter-panel');
    const heading = addElem(section, 'h5', 'u-h5');
    addElem(heading, 'span').textContent = options.heading||'分类';
    const table = addElem(section, 'table', 'u-table', { width: "100%", cellspacing: "0", cellpadding: "0", border: "0" });
    const tbody = addElem(table, 'tbody');
    const filterItems = [];
    function addFilterItem(fid, areaName) {
        const tr = addElem(tbody, 'tr');
        const td = addElem(tr, 'td', 'fav');
        const a = addElem(td, 'a', null, { href: fid > 0 ? `thread.php?fid-${fid}.html` : 'javascript:' });
        const count = addElem(td, 'span', null);
        const filterItem = {
            fid, td, areaName,
            setCount(n) {
                if (n >= 0) {
                    count.textContent = ` (${n})`;
                } else {
                    count.textContent = '';
                }
            },
            apply() {
                filterItems.forEach(item => item.td.classList.remove('current'));
                td.classList.add('current');
                filterHandler(fid);
                preserveState();
            }
        };

        if (defaultFid > 0 && defaultFid === fid) {
            defaultFilterItem = filterItem;
        }
        a.addEventListener('click', evt => {
            evt.preventDefault();
            filterItem.apply();
        });
        a.textContent = areaName;
        filterItems.push(filterItem);
        return filterItem;
    }
    const defaultAllItem = addFilterItem(0, options.showAllLabel||'全部');
    Array.from(areaMap.keys()).sort(comparator()).forEach(fid => {
        addFilterItem(fid, areaMap.get(fid));
    });

    if (defaultFilterItem) {
        defaultFilterItem.td.classList.add('current');
    } else {
        defaultAllItem.td.classList.add('current');
    }

    document.addEventListener('visibilitychange', () => {
        if (!document.hidden) {
            preserveState();
        }
    });

    function getCurrentOption() {
        return filterItems.find(item => item.td.classList.contains('current'));
    }
    function preserveState() {
        const current = getCurrentOption();
        if (current && current.fid) {
            const data = { key: basePageKey, fid: current.fid, areaName: current.areaName };
            localStorage.setItem(LAST_AREA_FILTER_MEMORY_KEY, JSON.stringify(data));
        } else {
            localStorage.removeItem(LAST_AREA_FILTER_MEMORY_KEY);
        }
    }

    return {
        apply() {
            const current = getCurrentOption();
            if (current) {
                current.apply();
            }
        },
        getCurrentFid() {
            const current = getCurrentOption();
            return current ? current.fid : 0;
        },
        updateCounts(counts) {
            let total = 0;
            filterItems.forEach(filterItem => {
                if (filterItem.fid === 0) {
                    return;
                }
                const n = counts.get(filterItem.fid);
                filterItem.setCount(n);
                total += n;
            });
            defaultAllItem.setCount(total);
        }
    };
}

async function enhanceFrontPage(userConfig, mainConfigAccess) {
    const cell = document.querySelector('#main > .t > table > tbody > .tr2 > td:last-child > .b');
    if (cell == null || cell.textContent !== '提问求物') {
        return;
    }
    const tbody = cell.closest('tbody');
    const itemRow = tbody.querySelector('.tr3');
    const closeIcon = addElem(cell.parentNode, 'a', 'closeicon fr', { style: 'cursor: pointer' });
    const closeImg = addElem(closeIcon, 'img', null);
    let collapsed = !!userConfig.hideFrontPageRecentList;
    function update() {
        if (collapsed) {
            closeImg.setAttribute('src', 'images/colorImagination/index/cate_open.gif');
            itemRow.setAttribute('style', 'display: none');
        } else {
            closeImg.setAttribute('src', 'images/colorImagination/index/cate_fold.gif');
            itemRow.setAttribute('style', '');
        }
    }
    update();
    closeIcon.addEventListener('click', async () => {
        mainConfigAccess
            .update(function(updatingUserConfig) {
                collapsed = !collapsed;
                updatingUserConfig.hideFrontPageRecentList = collapsed;
                return updatingUserConfig;
            })
            .finally(() => update(collapsed));
    });

    let targets = {
        '最新讨论': '/thread.php?fid=9&page=1',
        '最新回复': '/thread.php?fid-9.html',
        '同人音声': '/thread.php?fid-128.html',
        '提问求物': '/thread.php?fid-48.html'
    };

    tbody.querySelectorAll('.tr2 > td > .b').forEach(el => {
        const label = el.textContent.trim();
        let target = targets[label];
        if (target) {
            el.textContent = '';
            addElem(el, 'a', null, { href: target }).textContent = label;
        }
    });
}

const cachedLastAccessRecords = new Map();
async function enrichThreadTitle(tid, tLink, myUserId, userConfig, threadHistoryAccess) {
    if (threadHistoryAccess == null) {
        return;
    }
    let prefix;
    if (tLink.textContent.endsWith(' ..')) {
        prefix = tLink.textContent.substring(0, tLink.textContent.length - 3).replace(/&?n?b?s?p?;?$/, '').trim();
    } else {
        prefix = tLink.textContent.trim();
    }
    function handleLastAccessRecord(lastAccessRecord, fromCache) {
        if (lastAccessRecord && lastAccessRecord.fid === QUESTION_AND_REQUEST_AREA_ID && userConfig.requestThreadUseHistoryData) {
            tLink.parentElement.querySelectorAll('.rinsp-thread-bounty-status').forEach(el => el.remove());
            const bountySummary = newElem('span', 'rinsp-thread-bounty-status');
            tLink.parentElement.insertBefore(bountySummary, tLink);
            const age = getAgeString((Date.now() - lastAccessRecord.time) / 60000);
            bountySummary.setAttribute('title', `记录时间: ${age}`);
            if (lastAccessRecord.uid === myUserId) {
                bountySummary.classList.add('rinsp-thread-bounty-status-own');
            }
            if (lastAccessRecord.bounty) {
                if (lastAccessRecord.bounty.ended > 0) {
                    if (lastAccessRecord.bounty.winner || lastAccessRecord.bounty.ended === 2) {
                        if (lastAccessRecord.bounty.winner === userConfig.myUserHashId) {
                            bountySummary.classList.add('rinsp-thread-bounty-status-won');
                        } else {
                            bountySummary.classList.add('rinsp-thread-bounty-status-ended');
                        }
                    } else {
                        bountySummary.classList.add('rinsp-thread-bounty-status-expired');
                    }
                } else {
                    bountySummary.classList.add('rinsp-thread-bounty-status-ongoing');
                }
                const baseBounty = lastAccessRecord.bounty.sp;
                const extraBounty = parseSpAmount(lastAccessRecord.title);
                if (extraBounty > baseBounty) {
                    addElem(bountySummary, 'span').textContent = `${baseBounty}+${extraBounty}`;
                } else {
                    addElem(bountySummary, 'span').textContent = `${baseBounty}`;
                }
    
            } else {
                bountySummary.classList.add('rinsp-thread-bounty-status-unknown');
            }
        }
    
        if (lastAccessRecord && lastAccessRecord.title.startsWith(prefix)) {
            tLink.textContent = lastAccessRecord.title;
            tLink.parentElement.querySelectorAll('.rinsp-thread-populate-button').forEach(el => el.remove());
            if (lastAccessRecord.initialTitle && lastAccessRecord.initialTitle !== lastAccessRecord.title) {
                if (userConfig.showInitialRememberedTitle) {
                    tLink.classList.add('rinsp-thread-title-replaced');
                    addElem(tLink, 'div', 'rinsp-thread-initial-title').textContent = lastAccessRecord.initialTitle;
                } else {
                    tLink.setAttribute('title', '原始标题: ' + lastAccessRecord.initialTitle);
                }
            }
    
        } else if (fromCache) {
            // skip modification
        } else if (tLink.textContent.endsWith(' ..') && tLink.parentElement.querySelector('.rinsp-thread-populate-button') == null) {
            const expander = newElem('a', 'rinsp-thread-populate-button', { href: 'javascript:'} );
            expander.textContent = '展开';
            tLink.parentElement.insertBefore(expander, tLink.nextElementSibling);
            expander.addEventListener('click', async () => {
                if (expander.getAttribute('href') == null) {
                    return;
                }
                expander.removeAttribute('href');
                expander.textContent = '⌛';
                try {
                    await populateThreadAccess(tid, myUserId, threadHistoryAccess);
                    const updatedAccessRecord = await threadHistoryAccess.recentAccessStore.get(tid);
                    if (updatedAccessRecord) {
                        tLink.textContent = updatedAccessRecord.title;
                    }
                    expander.remove();
                } catch (e) {
                    expander.textContent = '⚠️';
                }
            });
        }
    }
    let lastAccessRecord = cachedLastAccessRecords.get(tid);
    if (lastAccessRecord) {
        handleLastAccessRecord(lastAccessRecord, true); // purely to avoid flash-of-content when refreshed
    }
    lastAccessRecord = await threadHistoryAccess.recentAccessStore.get(tid);
    if (lastAccessRecord) {
        cachedLastAccessRecords.set(tid, lastAccessRecord);
    }
    handleLastAccessRecord(lastAccessRecord, false);
}

function isAlpha(c) {
    return c >= 'a' && c <= 'z';
}
function isNumeric(c) {
    return c >= '0' && c <= '9';
}

function createKeywordMatcherFactory(keywords) {
    if (keywords == null || keywords.length === 0) {
        return null;
    }
    const exp = keywords
        .sort(comparator('length', true)) // longest first for regex match to work correctly
        .filter(s => !!s)
        .map(s => {
            if (s.startsWith('ignored:')) {
                return null;
            }
            if (s.length > 8 && s.startsWith('regex:/') && s.endsWith('/')) {
                try {
                    const exp = s.substring(7, s.length - 1);
                    RegExp(exp);
                    return exp;
                } catch (e) {
                    console.info(`[rin+] invalid regex entry: ${s}`);
                    return null;
                }
            }
            const term = s.toLowerCase();
            let exp = escapeRegExp(term);
            if (isAlpha(term[0])) {
                exp = '(?<=[^a-z]|^)' + exp;
            } else if (isNumeric(term[0])) {
                exp = '(?<=[^0-9]|^)' + exp;
            }
            if (isAlpha(term[term.length - 1])) {
                exp = exp + '(?=[^a-z]|$)';
            } else if (isNumeric(term[term.length - 1])) {
                exp = exp + '(?=[^0-9]|$)';
            }
            return exp;
        })
        .filter(s => !!s)
        .join('|');
    try {
        new RegExp(exp, 'g');
    } catch (ex) {
        if (DEV_MODE) console.warn('invalid-regexp: ' + exp);
        return null;
    }
    return {
        isStaticKeyword(word) {
            return keywords.indexOf(word) != -1;
        },
        match(text) {
            const pattern = new RegExp(exp, 'g');
            const matches = new Set();
            while (true) {
                const match = pattern.exec(text);
                if (match == null) {
                    break;
                }
                matches.add(match[0]);
            }
            return matches;
        }
    };
}

async function openCustomThreadCategoryEditor(threadCategoryConfigAccess, callback) {
    openTextListEditor(THREAD_LIKE_POPUP_MENU_ID, {
        title: '自定分类词列表',
        description: `每行一个关键字
正则表达式高级设置 (懂的都懂 不懂可无视) 格式: regex:/正则表达式语法/
  例子: regex:/[东南西北]\+/`,
        async read() {
            const customConfig = await threadCategoryConfigAccess.read();
            return customConfig.keywords.sort().join('\n');
        },
        async save(textData) {
            return await threadCategoryConfigAccess.update(function(customConfig) {
                const newKeywords = textData.split('\n').map(itm=>itm.replace(/\s+/g, ' ').trim()).filter(itm=>!!itm);
                customConfig.keywords = newKeywords;
                return customConfig;
            });
        }
    }, callback);
}

function enhanceTopicPostListDisplay(userId, myUserId, userConfig, threadHistoryAccess, threadCategoryConfigAccess, hideVisitedThreadToggler) {
    
    const headerCell = document.querySelector('#u-contentmain .u-table tr:first-child > td');
    headerCell.textContent = '';
    headerCell.nextElementSibling.textContent = '';
    const editKeywordsButton = addElem(headerCell.nextElementSibling, 'a');
    editKeywordsButton.setAttribute('href', 'javascript:void(0)');
    editKeywordsButton.textContent = '自定分类';
    editKeywordsButton.addEventListener('click', () => {
        openCustomThreadCategoryEditor(threadCategoryConfigAccess, () => apply());
    });
    const toggleItemsArea = addElem(headerCell, 'span', 'rinsp-category-filter');
    const toggleItems = [];
    let activeKeyword = null;

    const threadsById = new Map();
    const threadsByKeywords = new Map();

    function refreshCategoryFilter() {
        toggleItems.forEach(toggleItem => {
            if (toggleItem.keyword === activeKeyword) {
                toggleItem.toggler.classList.add('rinsp-category-active');
            } else {
                toggleItem.toggler.classList.remove('rinsp-category-active');
            }
        });
        if (activeKeyword) {
            const activeTids = threadsByKeywords.get(activeKeyword);
            threadsById.forEach((row, tid) => {
                if (activeTids != null && activeTids.has(tid)) {
                    row.classList.remove('rinsp-category-hide');
                } else {
                    row.classList.add('rinsp-category-hide');
                }
            });
        } else {
            threadsById.forEach((row, tid) => {
                row.classList.remove('rinsp-category-hide');
            });
        }
    }

    async function apply() {

        let keywordPatternMatcher = null;
        let keywordCaseMappings = new Map();
        const threadCategoryConfig = await threadCategoryConfigAccess.read();
        if (threadCategoryConfig.keywords.length > 0) {
            let keywords = threadCategoryConfig.keywords.map(kw => {
                let lower = kw.toLowerCase();
                keywordCaseMappings.set(lower, kw);
                return lower;
            });
            keywordPatternMatcher = createKeywordMatcherFactory(keywords);
        }

        threadsById.clear();
        threadsByKeywords.clear();
        const areaMap = new Map();
        document.querySelectorAll('#u-contentmain th > a[href^="read.php?tid-"]:not(.rinsp-gf-link)').forEach(el => {
            const cell = el.closest('th');
            const row = cell.closest('tr');
            const tid = Number.parseInt(el.getAttribute('href').substring(13));
            threadsById.set(tid, row);

            if (cell.querySelector('.rinsp-gf-link') == null) {
                const gfLink = newElem('a', 'rinsp-gf-link', {
                    href: `read.php?tid-${tid}-uid-${userId}.html`,
                    target: '_blank'
                });
                gfLink.textContent = '只看楼主';
                cell.prepend(gfLink);
            }
    
            const fidLink = cell.querySelector('a[href^="thread.php?fid-"]');
            const fid = Number.parseInt(fidLink.getAttribute('href').substring(15));
            setAreaScoped(row, fid);
            areaMap.set(fid, fidLink.textContent.trim());
            cell.classList.remove('rinsp-thread-visited');
            enrichThreadTitle(tid, el, myUserId, userConfig, threadHistoryAccess);
            
            if (keywordPatternMatcher) {
                const threadTitle = el.textContent;
                const matches = keywordPatternMatcher.match(threadTitle.toLowerCase());
                for (let keyword of matches) {
                    let threadSet = threadsByKeywords.get(keyword);
                    if (threadSet == null) {
                        threadSet = new Set();
                        threadsByKeywords.set(keyword, threadSet);
                    }
                    threadSet.add(tid);
                }
            }

        });
        toggleItemsArea.textContent = '';
        toggleItems.length = 0;
        threadsByKeywords.forEach((threadSet, keyword) => {
            
            const toggler = addElem(toggleItemsArea, 'div', 'rinsp-category-toggle-item');
            toggler.setAttribute('count', '' + threadSet.size);
            toggler.textContent = keywordCaseMappings.get(keyword)||keyword;
            toggler.addEventListener('click', () => {
                if (activeKeyword === keyword) {
                    activeKeyword = null;
                } else {
                    activeKeyword = keyword;
                }
                refreshCategoryFilter();
            });
            toggleItems.push({
                toggler,
                keyword
            });
        });

        addAreaFilter(areaMap, myUserId, defaultAreaFilterHandler).apply();
        await updateHistoryCheck();

    }

    async function updateHistoryCheck() {
        if (threadHistoryAccess == null || hideVisitedThreadToggler == null)
            return;
        let visitedCount = 0;
        const tids = Array.from(threadsById.keys());
        const lastSeenRecords = await threadHistoryAccess.historyStore.getBatch(tids);
        lastSeenRecords.forEach((record, i) => {
            const row = threadsById.get(tids[i]);
            if (record != null) {
                row.classList.add('rinsp-thread-visited');
                visitedCount++;
            }
        });
        hideVisitedThreadToggler.setCount(visitedCount);
    }

    if (hideVisitedThreadToggler) {
        hideVisitedThreadToggler.restoreLastState(true);
        // if history based styling is active, need to recheck on page activation
        document.addEventListener('visibilitychange', async () => {
            if (!document.hidden) {
                await updateHistoryCheck();
            }
        });
    }

    // accounting for infinite scroll added by soul++
    const observer = createMutationObserver(async () => {
        await apply();
    });
    observer.init(document.querySelector('#u-contentmain .u-table'), { childList: true, subtree: false, attributes: false });
    observer.trigger();

    document.addEventListener('visibilitychange', () => {
        if (!document.hidden) {
            observer.trigger();
        }
    });
}

async function enhanceReplyPostListDisplay(userId, myUserId, userConfig, userPinArea, mainConfigAccess, threadHistoryAccess, hideVisitedThreadToggler) {
    const firstItem = document.querySelector('#u-contentmain .u-table tr > th > a[href^="job.php?action-topost-tid-"]');
    if (firstItem == null) {
        return;
    }
    let userReplyListFolded = userConfig.userReplyListFolded;
    let accessUpdaters = [];
    function registerEnrichThreadTitle(tid, elem) {
        if (threadHistoryAccess) {
            const action = () => enrichThreadTitle(tid, elem, myUserId, userConfig, threadHistoryAccess);
            accessUpdaters.push(action);
            action();
        }
    }
    const baseTable = firstItem.closest('.u-table:not(.rinsp-reply-fold-table)');
    const threadsById = new Map();
    async function apply() {
        accessUpdaters.length = 0;
        threadsById.clear();
        let table = baseTable;
        const areaMap = new Map();
        if (userReplyListFolded) {
            const dataItems = [];
            table.querySelectorAll('tr + tr').forEach(row => {
                const th = row.querySelector('th');
                const topostLink = th.querySelector('a[href^="job.php?action-topost-tid-"]');
                const threadLink = th.querySelector('a[href^="thread.php?fid-"]');
                const areaHref = threadLink.getAttribute('href');
                const date = th.querySelector('span.gray.f9').textContent.replace(/^\[|\]$/g, '');
                const postUidLink = row.querySelector('td a[href^="u.php?action-show-uid-"]');
                const topostHref = topostLink.getAttribute('href');
                const topostRef = getPostRef(topostHref);
                if (topostRef) {
                    dataItems.push({
                        postTitle: topostLink.textContent.trim(),
                        topostHref: topostHref,
                        tid: topostRef.tid,
                        pid: topostRef.pid,
                        areaName: threadLink.textContent.trim(),
                        areaHref: areaHref,
                        areaFid: Number.parseInt(areaHref.substring(15)),
                        date,
                        postUid: Number.parseInt(postUidLink.getAttribute('href').substring(22)),
                        postUhash: postUidLink.textContent.trim()
                    });
                }
            });

            const groupedItems = [];
            const map = new Map();
            dataItems.forEach(item => {
                let group = map.get(item.tid);
                if (group == null) {
                    group = [];
                    map.set(item.tid, group);
                    groupedItems.push(group);
                }
                group.push(item);
            });
    
            baseTable.parentElement.querySelectorAll('.rinsp-reply-fold-table').forEach(el => el.remove());
            const foldTable = newElem('table', 'u-table rinsp-reply-fold-table');
            const foldTbody = addElem(foldTable, 'tbody');
            const foldTr1 = addElem(foldTbody, 'tr');
            foldTr1.innerHTML = '<td><br></td><td><br></td>';
            
            groupedItems.forEach(group => {
                const row = addElem(foldTbody, 'tr');
                const th = addElem(row, 'th');
                const td = addElem(row, 'td');
                td.textContent = '作者:';
                const firstItem = group[0];
                setAreaScoped(row, firstItem.areaFid);
                areaMap.set(firstItem.areaFid, firstItem.areaName);
                threadsById.set(firstItem.tid, [row]);

                addElem(td, 'a', 'gray', { href: `u.php?action-show-uid-${firstItem.postUid}.html`, target: '_blank' }).textContent = firstItem.postUhash;

                if (group.length === 1) {
                    const tLink = addElem(th, 'a', null, { href: firstItem.topostHref, target: '_blank' });
                    tLink.textContent = firstItem.postTitle;
                    addElem(th, 'br');
                    registerEnrichThreadTitle(firstItem.tid, tLink);
                } else {
                    const foldedItem = addElem(th, 'div', 'rinsp-reply-fold-item');
                    const tLink = addElem(foldedItem, 'a', null, { href: firstItem.topostHref, target: '_blank' });
                    tLink.textContent = firstItem.postTitle;
                    const pages = addElem(foldedItem, 'div', 'rinsp-reply-fold-pages');
                    addElem(pages, 'span', 'rinsp-reply-fold-count').textContent = `${group.length}`;
                    let range = addElem(pages, 'div', 'rinsp-reply-fold-remains');
                    const splitIndex = group.length > 5 ? group.length - 3 : group.length;
                    group.slice(1).forEach((item, i) => {
                        addElem(range, 'a', null, { href: item.topostHref, target: '_blank' }).textContent = `${group.length - i - 1}`;
                        if (i === splitIndex) {
                            range.classList.add('rinsp-reply-fold-remains-start');
                            range = addElem(pages, 'div', 'rinsp-reply-fold-remains rinsp-reply-fold-remains-end');
                        }
                    });
                    registerEnrichThreadTitle(firstItem.tid, tLink);
                }

                const gfLink = newElem('a', 'rinsp-gf-link', {
                    href: `read.php?tid-${firstItem.tid}-uid-${userId}.html#${firstItem.pid}`,
                    target: '_blank'
                });
                gfLink.textContent = '只看回复';
                th.append(gfLink);
    
                addElem(th, 'a', 'gray', { href: firstItem.areaHref, target: '_blank' }).textContent = firstItem.areaName;
                const lastItem = group[group.length - 1];
                if (firstItem.date === lastItem.date) {
                    addElem(th, 'span', 'gray f9').textContent = ` [${firstItem.date}]`;
                } else {
                    addElem(th, 'span', 'gray f9').textContent = ` [${firstItem.date} ~ ${lastItem.date}]`;
                }
            });
            table.parentElement.insertBefore(foldTable, table);
            table = foldTable;
        } else {
            document.querySelectorAll('#u-contentmain .u-table.rinsp-reply-fold-table').forEach(el => el.remove());
            table.querySelectorAll('tr > th > a[href^="job.php?action-topost-tid-"]').forEach(el => {
                const cell = el.closest('th');
                const row = cell.closest('tr');
                const postRef = getPostRef(el.getAttribute('href'));
                if (!postRef) {
                    return;
                }
                registerEnrichThreadTitle(postRef.tid, el);
                if (cell.querySelector('.rinsp-gf-link') == null) {
                    const gfLink = newElem('a', 'rinsp-gf-link', {
                        href: `read.php?tid-${postRef.tid}-uid-${userId}.html#${postRef.pid}`,
                        target: '_blank'
                    });
                    gfLink.textContent = '只看回复';
                    cell.prepend(gfLink);
                }
    
                const fidLink = cell.querySelector('a[href^="thread.php?fid-"]');
                const fid = Number.parseInt(fidLink.getAttribute('href').substring(15));
                setAreaScoped(row, fid);
                areaMap.set(fid, fidLink.textContent.trim());
                let rows = threadsById.get(postRef.tid);
                if (rows == null) {
                    rows = [];
                    threadsById.set(postRef.tid, rows);
                }
                rows.push(row);
                row.classList.remove('rinsp-thread-visited');
            });
        
        }

        addAreaFilter(areaMap, myUserId, defaultAreaFilterHandler).apply();
        await updateHistoryCheck();

        const controlContainer = table.querySelector('tr:first-child > td:first-child');
        if (controlContainer) {
            controlContainer.querySelectorAll('br, .rinsp-reply-list-controls').forEach(el => el.remove());
            const grouper = addElem(controlContainer, 'span', 'rinsp-reply-list-controls');
            grouper.appendChild(document.createTextNode('显示方式: '));
            const modeSelect = addElem(grouper, 'select');
            addElem(modeSelect, 'option', null, { value: '' }).textContent = '一般';
            addElem(modeSelect, 'option', null, { value: 'folding' }).textContent = '折叠式';
            modeSelect.value = userReplyListFolded ? 'folding' : '';
            modeSelect.value = modeSelect.value || '';
            modeSelect.addEventListener('change', () => {
                modeSelect.disabled = true;
                mainConfigAccess.update(function(updatingUserConfig) {
                    updatingUserConfig.userReplyListFolded = modeSelect.value === 'folding';
                    return updatingUserConfig;
                })
                .finally(function() {
                    userReplyListFolded = modeSelect.value === 'folding';
                    modeSelect.disabled = false;
                    apply();
                });
            });
        }
    }

    async function updateHistoryCheck() {
        if (threadHistoryAccess == null || hideVisitedThreadToggler == null)
            return;
        let visitedCount = 0;
        const tids = Array.from(threadsById.keys());
        const lastSeenRecords = await threadHistoryAccess.historyStore.getBatch(tids);
        lastSeenRecords.forEach((record, i) => {
            if (record != null) {
                const rows = threadsById.get(tids[i]);
                rows.forEach(row => row.classList.add('rinsp-thread-visited'));
                visitedCount++;
            }
        });
        hideVisitedThreadToggler.setCount(visitedCount);
    }

    if (hideVisitedThreadToggler) {
        hideVisitedThreadToggler.restoreLastState(true);
        // if history based styling is active, need to recheck on page activation
        document.addEventListener('visibilitychange', async () => {
            if (!document.hidden) {
                await updateHistoryCheck();
            }
        });
    }

    // accounting for infinite scroll added by soul++ (or me)
    const observer = createMutationObserver(async () => {
        await apply();
    });
    observer.init(baseTable, { childList: true, subtree: false, attributes: false });
    observer.trigger();

    document.addEventListener('visibilitychange', () => {
        if (!document.hidden) {
            accessUpdaters.forEach(accessUpdater => accessUpdater());
        }
    });

    await enhancePostListUserDisplay('post', userId, myUserId, userConfig, userPinArea);

    function getPostRef(postUrl) {
        const match = postUrl.match(/-tid-(\d+)-pid-(\d+)\.html$/);
        if (!match)
            return null;
        return {
            tid: match[1] * 1,
            pid: match[2] * 1,
        };
    }

}

function enhanceFavorPageDisplay(userId, myUserId, userConfig, threadHistoryAccess) {
    if (threadHistoryAccess) {
        document.querySelectorAll('#u-contentmain th > a[href^="read.php?tid-"]').forEach(el => {
            const tid = Number.parseInt(el.getAttribute('href').substring(13));
            enrichThreadTitle(tid, el, myUserId, userConfig, threadHistoryAccess);
        });
    }
}

function updateFavorRecords(favorThreadsCacheAccess) {
    const favors = readFavorList(document);
    favorThreadsCacheAccess.write({
        data: favors,
        time: Date.now()
    });
}

async function readFavorRecords(favorThreadsCacheAccess, fresh, noupdate) {
    let cached = fresh ? null : await favorThreadsCacheAccess.read();
    const now = Date.now();
    if (cached == null || cached.time > now || (now - cached.time) > 79200000) { // older than 22 hours
        const doc = await fetchGetPage(`${document.location.origin}/u.php?action-favor.html`);
        const favors = readFavorList(doc);
        const cacheItem = {
            data: favors,
            time: now
        };
        if (!noupdate) {
            await favorThreadsCacheAccess.write(cacheItem);
        }
        return favors;
    } else {
        return cached.data;
    }
}

async function enhancePostListUserDisplay(action, userId, myUserId, userConfig, userPinArea) {
    const mappings = userConfig.customUserHashIdMappings;

    function transformUserDisplay(el, postUid) {
        el.classList.add('rinsp-user-link');
        if (Number.isNaN(postUid)) {
            return;
        }

        const myself = userConfig.highlightMyself && myUserId === postUid;
        const owner = userId === postUid;
        const userRecord = mappings ? mappings['#' + postUid] : null;
        const pinnedUser = userPinArea.getPinned(postUid);
        if (userRecord || myself || owner || pinnedUser) {
            if (!el.dataset.rinspRestore) {
                el.dataset.rinspRestore = el.textContent;
            }
            el.innerHTML = '';
            if (myself) {
                addElem(el, 'span', 'rinsp-nickname-byme').textContent = MY_NAME_DISPLAY;
            } else if (owner) {
                addElem(el, 'span', 'rinsp-nickname-byowner').textContent = '楼主';
            } else if (pinnedUser) {
                addElem(el, 'span', 'rinsp-nickname-bypinned').textContent = pinnedUser.nickName;
                userPinArea.addLocation(postUid, '⚪', null, () => {
                    el.scrollIntoView({
                        behavior: 'smooth',
                        block: 'center'
                    });
                });
            } else {
                addElem(el, 'span', 'rinsp-nickname-byother').textContent = userRecord[2];
            }
        } else {
            const restore = el.dataset.rinspRestore;
            if (restore) {
                el.textContent = restore;
                delete el.dataset.rinspRestore;
            }
        }
    }

    async function update() {
        userPinArea.clearState();
        if (action === '' || action === 'feed') { // personal dashboard (好友近况)
            document.querySelectorAll('#minifeed .feed-list > dt > a.link1[href^="u.php?uid-"]').forEach(el => {
                const postUid = Number.parseInt(el.getAttribute('href').substring(10));
                transformUserDisplay(el, postUid);
            });
        } else if (action === 'favor') { // personal collection
            userPinArea.clearState();
            document.querySelectorAll('#u-content td > a[href^="u.php?uid-"]').forEach(el => {
                const postUid = Number.parseInt(el.getAttribute('href').substring(10));
                transformUserDisplay(el, postUid);
            });
        } else if (['topic', 'post', 'friend'].includes(action)) { // reply list, friend list
            const table = document.querySelector('#u-content .u-table');
            if (table) {
                table.querySelectorAll('td:nth-child(2) > a[href^="u.php?action-show-uid-"]').forEach(el => {
                    const postUid = Number.parseInt(el.getAttribute('href').substring(22));
                    transformUserDisplay(el, postUid);
                    const nextElem = el.nextElementSibling;
                    if (nextElem && nextElem.tagName === 'B' && nextElem.textContent === '在线') {
                        el.closest('tr').classList.add('rinsp-user-state-online');
                    }
                });
            }
        }
        await userPinArea.update();
    }
    // accounting for infinite scroll
    let triggerUpdate;
    const scope = document.querySelector('#u-contentmain');
    if (scope) {
        const observer = createMutationObserver(update);
        observer.init(scope, { childList: true, subtree: true, attributes: false });
        observer.trigger();
        triggerUpdate = () => observer.trigger();
    } else {
        await update();
        triggerUpdate = () => update();
    }
    userPinArea.addUpdateListener(() => {
        triggerUpdate();
    });

}

function createMutationObserver(callback) {
    let updateSchedule = null;
    let skipUpdate = 0;
    function onContentChange() {
        if (updateSchedule == null) {
            skipUpdate++;
            updateSchedule = setTimeout(function() {
                updateSchedule = null;
                callback()
                    .finally(() => skipUpdate--);
            }, 0);
        }
    }
    const observer = new MutationObserver(function(e) {
        if (DEBUG_MODE) console.info(skipUpdate > 0 ? 'skipped-mutation' : 'mutation');
        if (skipUpdate > 0) {
            return;
        }
        onContentChange();
    });
    return {
        init(watchElem, options) {
            observer.observe(watchElem, options);
        },
        trigger() {
            onContentChange();
        }
    };
}

function replaceTradeWithHistoryTab(myUserId, queryParams) {
    const tradeTab = document.querySelector(`#u_trade > a[href="u.php?action-trade.html"], #u_trade > a[href="u.php?action-trade-uid-${myUserId}.html"]`);
    if (tradeTab) {
        tradeTab.textContent = '浏览记录';
    }
}

function replaceTradeWithPunishHistoryTab(myUserId, queryParams) {
    const tradeTabLink = document.querySelector('#u_trade > a[href^="u.php?action-trade-uid-"]');
    if (tradeTabLink) {
        const tradeTab = tradeTabLink.closest('li');
        const uid = Number.parseInt(tradeTabLink.getAttribute('href').substring(23));
        if (uid === myUserId) {
            const adminHistoryTab = newElem('li');
            addElem(adminHistoryTab, 'a', null, { href: 'u.php?action-trade-view-admin.html' }).textContent = '管理记录';
            tradeTab.closest('ul').insertBefore(adminHistoryTab, tradeTab.nextElementSibling);
            if (queryParams.action === 'trade' && queryParams.view === 'admin') {
                tradeTab.classList.remove('current');
                adminHistoryTab.classList.add('current');
            }
        } else {
            tradeTabLink.textContent = '不良记录';
        }
    }
}

function comparator(prop, reversed) {
    const reverser = reversed ? -1 : 1;
    return (x, y) => {
        const xv = typeof prop === 'function' ? prop(x) : prop ? x[prop] : x;
        const yv = typeof prop === 'function' ? prop(y) : prop ? y[prop] : y;
        return xv > yv ? reverser : xv < yv ? -reverser : 0;
    };
}

function renderVisitHistoryPage(myUserId, userConfig, threadHistoryAccess) {
    const mainPane = document.querySelector('#u-content > #u-contentmain');
    const sidePane = document.querySelector('#u-content > #u-contentside');
    sidePane.innerHTML = '';
    if (threadHistoryAccess == null) {
        mainPane.innerHTML = '<div style="padding:16px 30px">💡 请先启用🗞️帖子记录功能</div>';
        return;
    }
    mainPane.innerHTML = '<div style="padding:16px 30px"><img src="images/loading.gif" align="absbottom"> 加载中 ... (请耐心等待)</div>';

    async function init() {
        const records = await threadHistoryAccess.recentAccessStore.list();
        if (records.length === 0) {
            mainPane.innerHTML = '<div style="padding:16px 30px">💡 没有历史记录</div>';
            return;
        }
        mainPane.innerHTML = '';
        const sortByVisitTime = comparator('time', true);
        const sortByPostTime = comparator('tid', true);
        
        const now = Date.now();
        const areaMap = new Map();
        records.forEach(record => areaMap.set(record.fid, record.area));
        const areaFilter = addAreaFilter(areaMap, myUserId, () => update(false));

        const table = addElem(mainPane, 'table', 'u-table rinsp-thread-history-table');
        const thead = addElem(table, 'thead');
        const tr1 = addElem(thead, 'tr');
        const filterCell = addElem(tr1, 'td', null, { colspan: '2' });
        const filterBox1 = addElem(filterCell, 'div', 'rinsp-quick-filter-box');
        addElem(filterBox1, 'div').textContent = '🔍︎搜索';
        const filterInput = addElem(filterBox1, 'input', null);

        addElem(filterBox1, 'div', null, { style: 'flex: 0 1 1em' });

        const controlBlock = addElem(filterBox1, 'div');
        controlBlock.appendChild(document.createTextNode('排序: '));
        const sortSelect = addElem(controlBlock, 'select');
        addElem(sortSelect, 'option', null, { value: 'last_visit' }).textContent = '访问时间';
        addElem(sortSelect, 'option', null, { value: 'thread_age' }).textContent = '发帖时间';
        sortSelect.value = 'last_visit';
        sortSelect.addEventListener('change', () => update(false));

        controlBlock.appendChild(document.createTextNode('显示数: '));
        const limitSelect = addElem(controlBlock, 'select');
        addElem(limitSelect, 'option', null, { value: '100' }).textContent = '100';
        addElem(limitSelect, 'option', null, { value: '200' }).textContent = '200';
        addElem(limitSelect, 'option', null, { value: '500' }).textContent = '500';
        addElem(limitSelect, 'option', null, { value: '1000' }).textContent = '1000';
        limitSelect.value = '100';
        limitSelect.addEventListener('change', () => update(false));

        const filterBox2 = addElem(filterCell, 'div', 'rinsp-quick-filter-box');
        addElem(filterBox2, 'div').textContent = '';
        const boughtFilterCheckbox = addCheckbox(filterBox2, '已购买', 'rinsp-quick-filter-checkbox rinsp-quick-filter-bought-checkbox');
        const repliedFilterCheckbox = addCheckbox(filterBox2, '已回复', 'rinsp-quick-filter-checkbox rinsp-quick-filter-replied-checkbox');
        const notmineFilterCheckbox = addCheckbox(filterBox2, '隐藏我的主题', 'rinsp-quick-filter-checkbox rinsp-quick-filter-notmine-checkbox');
        addElem(filterBox2, 'div', null, { style: 'flex: 1' });

        const tbody = addElem(table, 'tbody');
        const tfoot = addElem(table, 'tfoot');
        const statusRow = addElem(tfoot, 'tr');
        const statusCell = addElem(statusRow, 'td', 'grey', { colspan: '2' });

        let lastFilterString = '';
        function update(filterChangeOnly) {
            const terms = filterInput.value.toLowerCase().trim().split(/\s+/g);
            const thisFilterString = terms.join(' ');
            if (filterChangeOnly && lastFilterString === thisFilterString)
                return;
            lastFilterString = thisFilterString;
            statusCell.textContent = '';

            const limit = limitSelect.value * 1;
            const sortMethod = sortSelect.value === 'thread_age' ? sortByPostTime : sortByVisitTime;
            const fid = areaFilter.getCurrentFid();
            let selectedRecords;
            // apply area filter, also clone the list for sorting
            if (fid > 0) {
                selectedRecords = records.filter(record => record.fid === fid);
            } else {
                selectedRecords = records.slice();
            }

            // apply bought filter
            if (boughtFilterCheckbox.checked) {
                selectedRecords = selectedRecords.filter(record => !!record.bought);
            }

            // apply replied filter
            if (repliedFilterCheckbox.checked) {
                selectedRecords = selectedRecords.filter(record => record.replied > 0);
            }

            // apply not self filter
            if (notmineFilterCheckbox.checked) {
                selectedRecords = selectedRecords.filter(record => record.uid !== myUserId);
            }

            // apply text filter
            if (terms.length === 1 && terms[0] === '') {
                table.classList.remove('rinsp-table-filtered');
                if (selectedRecords.length > limit) {
                    statusCell.textContent = `💡只显示前${limit}条记录 / 共${selectedRecords.length}条`;
                }
            } else {
                table.classList.add('rinsp-table-filtered');
                selectedRecords = selectedRecords.filter(record => match(record, terms));

                if (selectedRecords.length === 0) {
                    statusCell.textContent = '💡没有搜索结果';
                } else if (selectedRecords.length > limit) {
                    statusCell.textContent = `💡只显示前${limit}条搜索结果 / 共${selectedRecords.length}条`;
                }
            }

            selectedRecords.sort(sortMethod);
            tbody.innerHTML = '';

            selectedRecords.slice(0, limit).forEach(record => {
                const row = addElem(tbody, 'tr');
                const th = addElem(row, 'th');
                const td = addElem(row, 'td');
                setAreaScoped(row, record.fid);

                const status = addElem(th, 'span', 'rinsp-thread-status-icons');
                if (record.deleted) {
                    row.classList.add('rinsp-thread-deleted');
                }
                if (record.replied) {
                    row.classList.add('rinsp-thread-replied');
                    addElem(status, 'a', 'rinsp-thread-replied-icon', { title: '已回复', href: `read.php?tid-${record.tid}-uid-${myUserId}.html`, target: '_blank' }).textContent = '↩️';
                }
                if (record.bought) {
                    row.classList.add('rinsp-thread-bought');
                    addElem(status, 'span', 'rinsp-thread-bought-icon', { title: '已购买' }).textContent = '💰';
                }
                addElem(th, 'a', null, { href: `read.php?tid-${record.tid}.html`, target: '_blank' }).textContent = record.title;
                addElem(th, 'div');
                if (record.initialTitle && record.initialTitle !== record.title) {
                    addElem(th, 'div', 'rinsp-thread-initial-title').textContent = record.initialTitle;
                }
    
                addElem(th, 'a', 'gray', { href: `thread.php?fid-${record.fid}.html`, target: '_blank' }).textContent = record.area;
                
                addElem(th, 'span', 'gray f9').textContent = ' [ ' + getAgeString((now - record.time) / 60000) + ' ]';

                const uidBlock = addElem(th, 'span', 'rinsp-thread-record-uid');

                if (record.uid > 0) {
                    if (record.uid === myUserId) {
                        row.classList.add('rinsp-thread-byme');
                        const userLink = addElem(uidBlock, 'a', 'rinsp-owner-isme', { href: `u.php?uid-${record.uid}.html`, target: '_blank' });
                        userLink.textContent = MY_NAME_DISPLAY;
                    } else {
                        let customNameEntry = userConfig.customUserHashIdMappings['#' + record.uid];
                        if (customNameEntry) {
                            const userLink = addElem(uidBlock, 'a', 'rinsp-owner-isknown', { href: `u.php?uid-${record.uid}.html`, target: '_blank' });
                            userLink.textContent = customNameEntry[2];
    
                        }
                    }
                }
            });
        }

        function addCheckbox(target, label, styleClass) {
            const labelElem = addElem(target, 'label', styleClass);
            const checkbox = addElem(labelElem, 'input', null, { type: 'checkbox' });
            addElem(labelElem, 'span', null, { style: 'vertical-align: text-top' }).textContent = label;
            checkbox.addEventListener('change', () => update(false));
            return checkbox;
        }

        let enqueueTimer = null;
        filterInput.addEventListener('keyup', () => {
            if (enqueueTimer) clearTimeout(enqueueTimer);
            enqueueTimer = setTimeout(() => update(true), 200);
            
        });
        filterInput.addEventListener('change', () => {
            if (enqueueTimer) clearTimeout(enqueueTimer);
            update(true);
        });

        update(false);
    }
    
    function match(record, searchTerms) {
        const title = record.title.toLowerCase();
        const initialTitle = (record.initialTitle||'').toLowerCase();
        for (let searchTerm of searchTerms) {
            if (title.indexOf(searchTerm) === -1 && initialTitle.indexOf(searchTerm) === -1)
                return false;
        }
        return true;
    }

    init();
}

const today = new Date(new Date().toISOString().replace(/T.*$/, 'T00:00:00'));
function annotateUsers(posts, myUserId, userConfig, userMap, services) {
    posts.forEach(post => {
        if (post.rootElem.classList.contains('rinsp-post-annotated')) {
            return;
        }
        post.rootElem.classList.add('rinsp-post-annotated');
        let customNameEntry = userConfig.customUserHashIdMappings['#' + post.postUid];
        if (!post.defaultUserPic) {
            post.rootElem.classList.add('rinsp-post-custom-userpic');
            const img = post.userPicElem.querySelector('img');
            if ((img.getAttribute('height')||'150') * 1 > 150) {
                post.rootElem.classList.add('rinsp-post-userpic-tall');
            }
        }
        if (userConfig.selfBypassHideUserpic && post.postUid === myUserId) {
            // skip
        } else {
            if (!userConfig.customUserBypassHideUserpic || customNameEntry == null) {
                if (post.defaultUserPic) {
                    if (userConfig.hideDefaultUserpic) {
                        post.rootElem.classList.add('rinsp-userpic-replace');
                        post.rootElem.classList.add('rinsp-userpic-replace-default');
                    }
                } else {
                    if (userConfig.hideOtherUserpic) {
                        post.rootElem.classList.add('rinsp-userpic-replace');
                        post.rootElem.classList.add('rinsp-userpic-replace-custom');
                    }
                }
            }
        }
        const data = userMap.get(post.postUid); // basically impossible
        if (data == null) {
            return;
        }
        const userLink = post.userNameElem.closest('a');
        const bookmarkElem = newElem('span', 'rinsp-userbookmark');
        userLink.parentNode.insertBefore(bookmarkElem, userLink);
        const bookmarkActionElem = addElem(bookmarkElem, 'span', 'rinsp-userbookmark-action');
        if (customNameEntry != null) {
            post.rootElem.classList.add('rinsp-userframe-userknown');
            const renamed = customNameEntry[2] !== post.postUname;
            if (data.unnamed || renamed) {
                bookmarkElem.classList.add('rinsp-userbookmark-takeover');
                if (renamed) {
                    post.rootElem.classList.add('rinsp-userframe-userrenamed');
                }
                const customUserLink = addElem(bookmarkElem, 'a');
                customUserLink.setAttribute('href', `u.php?action-show-uid-${post.postUid}.html`);
                customUserLink.setAttribute('target', '_blank');
                customUserLink.textContent = customNameEntry[2];
            }
        }
        
        bookmarkActionElem.addEventListener('click', async () => {
            if (customNameEntry == null) {
                const userHashId = await fetchUserHashId(post.postUid);
                services.changeCustomUserMapping(post.postUid, userHashId, post.postUname)
                    .then(value => {
                        if (value != null) {
                            location.reload();
                        }
                    });
            } else {
                services.changeCustomUserMapping(customNameEntry[0], customNameEntry[1], customNameEntry[2])
                    .then(value => {
                        if (value != null) {
                            location.reload();
                        }
                    });
            }
        });
        let cell = post.userPicElem.closest('.r_two');
        if (data.unnamed) {
            cell.classList.add('rinsp-userframe-unnamed');
        }

        if (userConfig.showExtendedUserInfo && data != null) {
            if (userConfig.showExtendedUserInfo$HP && data.hp < 0) {
                let hpTag = newElem('div', 'rinsp-userframe-udata-hp');
                hpTag.textContent = data.hp + 'HP';
                cell.querySelector('.user-pic + div > a').appendChild(hpTag);
            }
            addElem(post.userPicElem, 'div', 'rinsp-userframe-pulldown');
            let infoBlock = addElem(post.userPicElem, 'div', 'rinsp-userframe-userinfo');

            const addItem = (key, value, cls) => {
                let item = addElem(infoBlock, 'dl', cls);
                let dt = addElem(item, 'dt');
                dt.textContent = key;
                let dd = addElem(item, 'dd');
                dd.textContent = value;
            };
            addItem('SP', data.sp.toLocaleString(), 'rinsp-userframe-udata-sp');
            addItem('在线', String(data.online) + '小时', 'rinsp-userframe-udata-online', { title: '在线时间' });

            addElem(infoBlock, 'div', 'rinsp-userframe-udata-spacer');

            let actionItem = addElem(infoBlock, 'div', 'rinsp-userframe-udata-punish-tags');
            if (services.punishRecordAccess) {
                services.punishRecordAccess
                    .getPunishRecord(post.postUid)
                    .then(record => {
                        actionItem.innerHTML = '';
                        if (record == null || record.logs.length === 0) {
                            return;
                        }
                        actionItem.appendChild(renderPunishTags(post.postUid, record));
                    });
            }


            let loginItem = addElem(infoBlock, 'div', 'rinsp-userframe-udata-login');
            loginItem.textContent = '最后登录 ';
            let loginItemValue = addElem(loginItem, 'span');
            let awayDays = (today - new Date(data.login + 'T00:00:00')) / 86400000;
            if (awayDays < 1) {
                loginItemValue.classList.add('rinsp-userframe-udata-login-today');
                loginItemValue.textContent = '今天';
            } else if (awayDays < 2) {
                loginItemValue.classList.add('rinsp-userframe-udata-login-yesterday');
                loginItemValue.textContent = '昨天';
            } else {
                loginItemValue.textContent = data.login;
            }

        }

        const nameLabel = cell.querySelector('.user-pic ~ div[align="center"]');
        if (nameLabel) {
            nameLabel.classList.add('rinsp-userframe-username');
            if (userConfig.stickyUserInfo) {
                const nameLabelSticky = addElem(post.userPicElem, 'div', 'rinsp-userframe-username-sticky');
                nameLabelSticky.appendChild(bookmarkElem.cloneNode(true));
                nameLabelSticky.appendChild(nameLabel.querySelector('.rinsp-userbookmark + a').cloneNode(true));
                nameLabelSticky.querySelector('.rinsp-userbookmark > .rinsp-userbookmark-action').addEventListener('click', () => {
                    bookmarkElem.querySelector('.rinsp-userbookmark-action').click();
                });
            }
        }

        if (userConfig.stickyUserInfo) {
            const dummyUserPic = newElem('div', 'rinsp-user-pic-dummy');
            post.userPicElem.parentNode.insertBefore(dummyUserPic, post.userPicElem.nextElementSibling);
        }
    });
}

async function fetchGetPage(url, docType, autoRetry) {
    const resp = await fetch(url, {
        method: 'GET',
        mode: 'same-origin',
        credentials: 'same-origin',
        cache: 'no-cache'
    });

    if (!resp.ok) {
        throw new Error('网络或登入错误');
    }
    const content = await resp.text();
    const parser = new DOMParser();
    const doc = parser.parseFromString(content, docType || 'text/html');
    if (autoRetry) {
        if (findErrorMessage(doc) === '论坛设置:刷新不要快于 1 秒') {
            await sleep(1100);
            return await fetchGetPage(url, docType, false);
        }
    }
    return doc;
}

async function fetchUserHashId(userId) {
    const doc = await fetchGetPage(`${document.location.origin}/u.php?action-show-uid-${userId}.html`);
    const hashIdElem = doc.querySelector('#main #u-wrap #u-content #u-top h1');
    const hashId = hashIdElem.textContent.trim();
    return hashId;
}

async function showUserNames(userConfig, userHashLookupStore, myUserId) {
    let mappings = userConfig.customUserHashIdMappings;
    if (mappings == null) {
        mappings = {};
    }

    function transformOwnerCells(elemList, alsoChangeName) {
        elemList.forEach(el => {
            if (el.classList.contains('rinsp-uid-inspected')) {
                return;
            }
            el.classList.add('rinsp-uid-inspected');
            const uid = Number.parseInt(el.getAttribute('href').substring(22));
            if (userConfig.highlightMyself && uid === myUserId) {
                el.classList.add('rinsp-owner-isme');
                if (alsoChangeName) {
                    el.textContent = MY_NAME_DISPLAY;
                }
            } else {
                const fullName = mappings['#' + uid];
                if (fullName != null) {
                    el.classList.add('rinsp-owner-isknown');
                    if (alsoChangeName) {
                        el.textContent = fullName[2];
                    }
                }
            }
        });
    }

    if (document.location.pathname === '/search.php') {
        transformOwnerCells(document.querySelectorAll('tr.tr3.tac > td.smalltxt.y-style > a[href^="u.php?action-show-uid-"]'), true);
    } else {
        transformOwnerCells(document.querySelectorAll('td[id^="td_"] ~ td.tal > a.bl[href^="u.php?action-show-uid-"]'), true);
    }

    async function transformReplyCells(itemList) {
        const unresolved = [];
        for (let item of itemList) {
            if (item.byElem.classList.contains('rinsp-hashid-inspected')) {
                continue;
            }

            item.byElem.classList.add('rinsp-hashid-inspected');
            let match = item.byElem.textContent.match(/^by: ([0-9a-f]{8})$/);
            if (match) {
                const uhash = match[1];
                let myself = userConfig.highlightMyself && userConfig.myUserHashId === uhash;
                let fullName = mappings['@' + uhash];
                if (fullName || myself) {
                    if (myself) {
                        item.byElem.textContent = 'by ' + MY_NAME_DISPLAY;
                        item.byElem.classList.add('rinsp-nickname-byme');
                    } else {
                        item.byElem.textContent = 'by ' + fullName[2];
                        item.byElem.classList.add('rinsp-nickname-byother');
                    }
                } else if (userConfig.replyShownAsByOp) {
                    if (item.replyCount === 0 || item.opHashId === uhash) {
                        item.byElem.textContent = 'by 楼主';
                        item.byElem.classList.add('rinsp-nickname-byop');
                        if (item.replyCount === 0) {
                            item.byElem.classList.add('rinsp-nickname-byop-only');
                        }
                    } else {
                        unresolved.push({
                            item, uhash, uhashNum: Number.parseInt(uhash, 16)
                        });
                    }
                }
            }
        }
        if (unresolved.length > 0) {
            const uids = await userHashLookupStore.getBatch(unresolved.map(entry => entry.uhashNum));
            unresolved.forEach((entry, i) => {
                if (uids[i] != null && entry.item.opUid === uids[i]) {
                    entry.item.byElem.textContent = 'by 楼主';
                    entry.item.byElem.classList.add('rinsp-nickname-byop');
                }
            });
        }
    }

    let itemList;
    if (document.location.pathname === '/search.php') {
        itemList = Array.from(document.querySelectorAll('tr.tr3.tac > td.smalltxt.y-style ~ td > a[href^="read.php?tid-"] + br'))
            .map(br => {
                let byElem;
                let next = br.nextSibling;
                if (next.nodeType === 3) {
                    byElem = document.createElement('span');
                    byElem.textContent = next.textContent;
                    br.parentNode.insertBefore(byElem, next);
                    next.remove();
                } else {
                    byElem = br.nextElementSibling;
                }
                const tr = byElem.closest('tr');
                const replyCount = tr.querySelector('td.smalltxt.y-style + .y-style').textContent * 1;
                const op = tr.querySelector('td.smalltxt.y-style > a[href^="u.php?action-show-uid-"]');
                const opUid = op ? Number.parseInt(op.getAttribute('href').substring(22)) : null;
                const opHashId = op ? op.textContent.trim() : null;
                return {
                    opUid,
                    opHashId,
                    replyCount,
                    byElem
                };
            });
    } else {
        itemList = Array.from(document.querySelectorAll('td[id^="td_"] ~ td.tal > a[href^="read.php?tid-"] + br + span.gray2'))
            .map(byElem => {
                const tr = byElem.closest('tr');
                const replyCount = tr.querySelector('td.tal.y-style.f10 > .s8').textContent * 1;
                const op = tr.querySelector('td.tal.y-style > a[href^="u.php?action-show-uid-"]');
                const opUid = op ? Number.parseInt(op.getAttribute('href').substring(22)) : null;
                return {
                    opUid,
                    opHashId: null,
                    replyCount,
                    byElem
                };
            });
    }
    await transformReplyCells(itemList);
}

function getScaledValue(watchItem, min, max) {
    return min + (max - min) * Math.pow(Math.min(OLD_ITEM_THRESHOLD_DAYS, Math.floor(getPostAge(watchItem))), OLD_ITEM_SCALING) / Math.pow(OLD_ITEM_THRESHOLD_DAYS, OLD_ITEM_SCALING);
}

function getPostAge(watchItem) {
    return (Date.now() - watchItem.timeOpened) / 86400000;
}

function getWatchExpiryDays(watchItem) {
    switch (watchItem.areaName) {
        case REQUEST_ZONE_NAME:
            return 45;
        case '茶馆':
            return 120;
        case '茶楼':
            return 60;
        default:
        return -1;
    }
}

function isWatchExpired(watchItem) {
    if (watchItem.timeOpened == null) { // impossible actually
        return false;
    } else {
        const expiryDays = getWatchExpiryDays(watchItem);
        if (expiryDays < 0) {
            return false;
        } else {
            return Date.now() - watchItem.timeOpened > expiryDays * 86400000;
        }
    }
}

function createInfiniteScrollHandler(typeId, infConfig, userConfig, mainConfigAccess) {
    const configKey = 'infiniteScroll$' + typeId;
    const enableEffect = {
        value: !!userConfig[configKey],
        listeners: [],
        setValue(newValue) {
            this.value = newValue;
            this.listeners.forEach(fn => fn());
            if (this.value) {
                install();
            } else {
                uninstall();
            }
        }
    };
    const cfg = Object.assign({
        initPaginator(doc) {
            const pagesones = Array.from(doc.querySelectorAll('.pages > ul > li.pagesone'));
            if (pagesones.length === 0) {
                return null;
            }
            pagesones.forEach(el => addInfSwitch(el));
            
            const pagesone = pagesones.pop();
            const otherPageBars = pagesones.map(el => el.closest('.pages'));
            const match = pagesone.textContent.match(/Pages:\s*(\d+)(?: *\u2013 *(\d+))?\/(\d+)/);
            let curPageStart = match[1] * 1;
            let curPageEnd = (match[2]||match[1]) * 1;
            const maxPage = match[3] * 1;
            const curPageLabel = pagesone.closest('ul').querySelector('li > b');
            const curPageElem = curPageLabel.parentNode;
            let morePageElems = [];
            let pageElem = curPageElem.nextElementSibling;
            while (!pageElem.classList.contains('pagesone')) {
                morePageElems.push(pageElem);
                pageElem = pageElem.nextElementSibling;
            }
            morePageElems.pop();
            return {
                getMorePageElems() {
                    return morePageElems;
                },
                getCurrentStart() {
                    return curPageStart;
                },
                getCurrentEnd() {
                    return curPageEnd;
                },
                getMaxPage() {
                    return maxPage;
                },
                getNextPageURL() {
                    return morePageElems.length > 0 ? morePageElems[0].querySelector('a[href]').href : null;
                },
                setCurrentEnd(paginator) {
                    curPageEnd = paginator.getCurrentEnd();
                    const cur = curPageStart === curPageEnd ? curPageStart : `${curPageStart} \u2013 ${curPageEnd}`;
                    const input = pagesone.querySelector('input');
                    doc.body.appendChild(input);
                    pagesone.textContent = `Pages: ${cur}/${maxPage}\u00A0 \u00A0 \u00A0Go `;
                    pagesone.appendChild(input);
                    curPageLabel.textContent = cur;
                    while (morePageElems.length > 0) {
                        morePageElems.pop().remove();
                    }
                    const temp = doc.createElement('div');
                    temp.innerHTML = paginator.getMorePageElems().map(el => el.outerHTML).join('');
                    morePageElems = Array.from(temp.children);
                    morePageElems.slice().reverse().forEach(el => {
                        curPageElem.parentNode.insertBefore(el, curPageElem.nextElementSibling);
                    });
                    temp.remove();
                    otherPageBars.forEach(el => {
                        const extraSwitches = Array.from(el.querySelectorAll('.pagesone ~ li')); // retain added switches
                        extraSwitches.forEach(ext => document.body.appendChild(ext));
                        el.innerHTML = pagesone.closest('.pages').innerHTML;
                        el.querySelectorAll('.pagesone ~ li').forEach(el => el.remove());
                        extraSwitches.forEach(ext => el.querySelector('.pagesone').parentNode.appendChild(ext));
                    });
                }
            };
            function addInfSwitch(pagesone) {
                const li = addElem(pagesone.closest('ul'), 'li');
                const infToggle = addElem(li, 'a', 'rinsp-infscroll-switch', { href: 'javascript:', title: '自动加载下一页' });
                infToggle.textContent = '↷';
                infToggle.addEventListener('click', async () => {
                    const newUserConfig = await mainConfigAccess.update(updatingUserConfig => {
                        updatingUserConfig[configKey] = !enableEffect.value;
                        return updatingUserConfig;
                    });
                    enableEffect.setValue(!!newUserConfig[configKey]);
                });
                function update() {
                    if (enableEffect.value) {
                        infToggle.classList.add('rinsp-active');
                    } else {
                        infToggle.classList.remove('rinsp-active');
                    }
                }
                enableEffect.listeners.push(update);
                update();
            }
        },
        getItemId(item) {
            return null;
        },
        getContentItems(doc) {
            return [];
        },
        appendContentItems(items, start, end) {

        },
        getDividerLabel(page) {
            return `第 ${page} 页`;
        }
    }, infConfig);
    const paginator = cfg.initPaginator(document);
    if (paginator == null) {
        return;
    }
    let prepareFlipTimer = null;
    let triggerFlipTimer = null;
    const endTrigger = addElem(document.body, 'div', 'rinsp-infscroll-page-endtrigger');
    addElem(endTrigger, 'div', 'rinsp-infscroll-loader-bar');

    const delayBeforeFlipPage = 500;
    const delayBeforeAllowFlipPage = 200;
    const stayBottomThreshold = 15;
    const triggerMargin = 5;

    let currentState = 0;
    function setInfScrollState(state) {
        currentState = state;
        function clearPrepare() {
            if (prepareFlipTimer) {
                clearTimeout(prepareFlipTimer);
                prepareFlipTimer = null;
            }
        }
        function clearTrigger() {
            if (triggerFlipTimer) {
                clearTimeout(triggerFlipTimer);
                triggerFlipTimer = null;
            }
        }
        if (state === 0) { // idle
            clearPrepare();
            clearTrigger();
            document.documentElement.classList.remove('rinsp-infscroll-armed');
            document.documentElement.classList.remove('rinsp-infscroll-firing');
        } else if (state === 1) { // armed
            document.documentElement.classList.add('rinsp-infscroll-armed');
            document.documentElement.classList.remove('rinsp-infscroll-firing');
            clearPrepare();
            clearTrigger();
        } else if (state === 2) { // firing
            document.documentElement.classList.add('rinsp-infscroll-armed');
            document.documentElement.classList.add('rinsp-infscroll-firing');
            clearPrepare();
        }
    }
    const scrollEventListener = () => {
        const bottomGap = document.body.scrollHeight - window.scrollY - window.innerHeight;
        if (bottomGap <= triggerMargin) {
            if (currentState === 1) {
                setInfScrollState(2);
                if (triggerFlipTimer == null) {
                    triggerFlipTimer = setTimeout(async () => {
                        setInfScrollState(0);
                        await insertNextPageSync();
                    }, delayBeforeFlipPage);
                }
            } else if (currentState === 0) {
                if (prepareFlipTimer == null) {
                    const nextPageURL = paginator.getNextPageURL();
                    if (nextPageURL != null) {
                        fetchNextPage(nextPageURL); // prefetch
                    }
                    prepareFlipTimer = setTimeout(() => {
                        setInfScrollState(1);
                    }, delayBeforeAllowFlipPage);
                }
            }
        } else if (bottomGap > stayBottomThreshold) {
            setInfScrollState(0);
        }
    };

    function install() {
        uninstall();
        if (paginator.getNextPageURL() == null) {
            return;
        }
        document.documentElement.classList.add('rinsp-infscroll-enabled');
        window.addEventListener('scroll', scrollEventListener);
    }
    function uninstall() {
        document.documentElement.classList.remove('rinsp-infscroll-enabled');
        setInfScrollState(0);
        window.removeEventListener('scroll', scrollEventListener);
    }

    let busy = false;
    let nextPageCache = { expiry: Date.now(), url: null, data: null };
    async function fetchNextPage(url) {
        if (nextPageCache.url === url && nextPageCache.expiry > Date.now()) {
            return nextPageCache.data;
        }
        const data = fetchGetPage(url, null, true);
        nextPageCache = {
            url,
            expiry: Date.now() + 10000, // 10 seconds
            data
        };
        return data;
    }

    async function insertNextPageSync() {
        if (!busy) {
            busy = true;
            await insertNextPage()
                .then(obj => {
                    if (obj && userConfig.infiniteScrollScrollToNewPage) {
                        obj.scrollToNewPage();
                    }
                })
                .finally(() => { busy = false; });
        }
    }
    async function insertNextPage() {
        const nextPageURL = paginator.getNextPageURL();
        if (nextPageURL == null) {
            return;
        }
        const oldScrollTop = document.documentElement.scrollTop;
        const divider = cfg.appendDivider();
        divider.classList.add('rinsp-infscroll-loading');
        divider.textContent = `第 ${paginator.getCurrentEnd() + 1} 页 - 加载中 ...`;
        const nextDoc = await fetchNextPage(nextPageURL);
        if (isFastLoadMode()) {
            deferImageLoading(nextDoc);
        }
        const currentIds = new Set(Array.from(cfg.getContentItems(document)).map(item => cfg.getItemId(item)).filter(item => item != null));
        const newItems = Array.from(cfg.getContentItems(nextDoc)).filter(item => {
            const itemId = cfg.getItemId(item);
            return itemId == null || !currentIds.has(itemId);
        });
        const newPaginator = cfg.initPaginator(nextDoc);
        if (newPaginator == null) {
            divider.textContent = '找不到更多记录';
            uninstall();
            return;
        }
        divider.textContent = cfg.getDividerLabel(newPaginator.getCurrentStart());
        divider.classList.remove('rinsp-infscroll-loading');
        cfg.appendContentItems(newItems);
        paginator.setCurrentEnd(newPaginator);
        if (paginator.getCurrentEnd() >= paginator.getMaxPage()) {
            uninstall();
        }
        if (userConfig.infiniteScrollReplaceURL) {
            window.history.replaceState(null, null, nextPageURL);
        }
        if (isFastLoadMode()) {
            resumeImageLoading(document);
        }
        try { unsafeWindow.peacemaker.rescan(); } catch (ex) {}
        document.documentElement.scrollTop = oldScrollTop;
        return {
            scrollToNewPage() {
                divider.scrollIntoView({
                    behavior: 'smooth',
                    block: 'start'
                });
            }
        };
    }

    if (enableEffect.value) {
        install();
    }
}

function setupInfiniteScroll_userTopics(userConfig, mainConfigAccess) {
    const table = document.querySelector('#u-contentmain > .u-table');

    createInfiniteScrollHandler('usertopics', {
        getContentItems(doc) {
            return doc.querySelectorAll('#u-contentmain > .u-table > tbody > tr ~ tr');
        },
        appendDivider() {
            return addElem(addElem(addElem(table, 'tbody', 'rinsp-infscroll-divider'), 'tr'), 'td', null, { colspan: '2' });
        },
        appendContentItems(items) {
            const nextPageBody = addElem(table, 'tbody');
            items.forEach(item => nextPageBody.appendChild(item));
        },
        getItemId(item) {
            const a = item.querySelector('th > a[href^="read.php?tid-"]');
            return a ? a.getAttribute('href') : null;
        }
    }, userConfig, mainConfigAccess);

}

function setupInfiniteScroll_userReplies(userConfig, mainConfigAccess) {
    const table = document.querySelector('#u-contentmain > .u-table');

    createInfiniteScrollHandler('userposts', {
        getContentItems(doc) {
            return doc.querySelectorAll('#u-contentmain > .u-table > tbody > tr ~ tr');
        },
        appendDivider() {
            return addElem(addElem(addElem(table, 'tbody', 'rinsp-infscroll-divider'), 'tr'), 'td', null, { colspan: '2' });
        },
        appendContentItems(items) {
            const nextPageBody = addElem(table, 'tbody');
            items.forEach(item => nextPageBody.appendChild(item));
        },
        getItemId(item) {
            const a = item.querySelector('th > a[href^="job.php?action-topost-tid-"]');
            return a ? a.getAttribute('href') : null;
        }
    }, userConfig, mainConfigAccess);

}

function setupInfiniteScroll_thread(userConfig, mainConfigAccess) {
    const form = document.querySelector('form[name="delatc"]');

    createInfiniteScrollHandler('thread_posts', {
        getContentItems(doc) {
            const rows = Array.from(doc.querySelector('form[name="delatc"]').children);
            while (rows[0].tagName === 'INPUT') {
                rows.shift();
            }
            return rows;
        },
        appendDivider() {
            return addElem(form, 'div', 'rinsp-infscroll-divider');
        },
        appendContentItems(items) {
            items.forEach(item => {
                if (item.matches('.t5.t2[style="border-top:0"]')) {
                    item.removeAttribute('style');
                }
                form.appendChild(item);
            });
        },
        getItemId(item) {
            const idCell = item.querySelector('.t5.t2 tr > th[id^="td_"], div[id^="menu_read_"]');
            if (idCell) {
                return idCell.getAttribute('id');
            }
            if (item.matches('a[name]')) {
                return 'a:' + item.getAttribute('name');
            }
            return null;
        }
    }, userConfig, mainConfigAccess);

}

function setupInfiniteScroll_search(userConfig, mainConfigAccess) {
    const table = document.querySelector('#main > .t > table');

    createInfiniteScrollHandler('search', {
        getContentItems(doc) {
            return doc.querySelectorAll('#main > .t > table > tbody > .tr2, #main > .t > table > tbody > .tr3');
        },
        appendDivider() {
            return addElem(addElem(addElem(table, 'tbody', 'rinsp-infscroll-divider'), 'tr'), 'td', null, { colspan: '7' });
        },
        appendContentItems(items) {
            const endPaddingRow = table.querySelector('tbody > tr > td.f_one[colspan]');
            const nextPageBody = addElem(table, 'tbody');
            items.forEach(item => nextPageBody.appendChild(item));
            if (endPaddingRow) {
                nextPageBody.appendChild(endPaddingRow);
            }
        },
        getItemId(item) {
            const link = item.querySelector('.tr3 > th > a[href^="read.php?tid-"]');
            return link ? Number.parseInt(link.getAttribute('href').substring(13)) : null;
        }
    }, userConfig, mainConfigAccess);

}

function setupInfiniteScroll_threadList(userConfig, mainConfigAccess) {
    const table = document.querySelector('#ajaxtable');

    createInfiniteScrollHandler('threads', {
        getContentItems(doc) {
            return doc.querySelectorAll('#ajaxtable > tbody + tbody > .tr2, #ajaxtable > tbody + tbody > tr.t_one');
        },
        appendDivider() {
            return addElem(addElem(addElem(table, 'tbody', 'rinsp-infscroll-divider'), 'tr'), 'td', null, { colspan: '5' });
        },
        appendContentItems(items) {
            const endPaddingRow = table.querySelector('tbody > tr > td.f_one[colspan]');
            const nextPageBody = addElem(table, 'tbody');
            items.forEach(item => nextPageBody.appendChild(item));
            if (endPaddingRow) {
                nextPageBody.appendChild(endPaddingRow);
            }
        },
        getItemId(item) {
            const idCell = item.querySelector('.t_one > td[id^="td_"]');
            return idCell ? idCell.getAttribute('id') : null;
        }
    }, userConfig, mainConfigAccess);

}

function setupInfiniteScroll_threadPicWall(userConfig, mainConfigAccess) {
    const container = document.querySelector('#thread_img > .dcsns-content');

    createInfiniteScrollHandler('threads_picwall', {
        getContentItems(doc) {
            return doc.querySelectorAll('#thread_img > .dcsns-content > .stream > li');
        },
        appendDivider() {
            return addElem(container, 'div', 'rinsp-infscroll-divider');
        },
        appendContentItems(items) {
            const nextWall = addElem(container, 'ul', 'stream');
            items.forEach(item => nextWall.appendChild(item));
            try {
                unsafeWindow.jQuery('img.lazy', nextWall).lazyload({ skip_invisible : false });
            } catch (ignore) {}
        },
        getItemId(item) {
            const link = item.querySelector('.section-title > a[href]');
            return link ? link.getAttribute('href') : null;
        }
    }, userConfig, mainConfigAccess);

}

function setupInfiniteScroll_msgInbox(userConfig, mainConfigAccess) {
    const table = document.querySelector('#info_base .set-table2');

    createInfiniteScrollHandler('msg_inbox', {
        getContentItems(doc) {
            return doc.querySelectorAll('#info_base .set-table2 > tbody > tr');
        },
        appendDivider() {
            return addElem(addElem(addElem(table, 'tbody', 'rinsp-infscroll-divider'), 'tr'), 'td', null, { colspan: '5' });
        },
        appendContentItems(items) {
            const nextTbody = addElem(table, 'tbody');
            items.forEach(item => nextTbody.appendChild(item));
        },
        getItemId(item) {
            const link = item.querySelector('input[name="delid[]"]');
            return link ? link.value : null;
        }
    }, userConfig, mainConfigAccess);

}

function setupInfiniteScroll_msgSent(userConfig, mainConfigAccess) {
    const table = document.querySelector('#info_base .set-table2');

    createInfiniteScrollHandler('msg_sent', {
        getContentItems(doc) {
            return doc.querySelectorAll('#info_base .set-table2 > tbody > tr');
        },
        appendDivider() {
            return addElem(addElem(addElem(table, 'tbody', 'rinsp-infscroll-divider'), 'tr'), 'td', null, { colspan: '7' });
        },
        appendContentItems(items) {
            const nextTbody = addElem(table, 'tbody');
            items.forEach(item => nextTbody.appendChild(item));
        },
        getItemId(item) {
            const link = item.querySelector('input[name="delid[]"]');
            return link ? link.value : null;
        }
    }, userConfig, mainConfigAccess);

}

function setupInfiniteScroll_msgChatLog(userConfig, mainConfigAccess) {
    const table = document.querySelector('#info_base .set-table2');

    createInfiniteScrollHandler('msg_chatlog', {
        getContentItems(doc) {
            return doc.querySelectorAll('#info_base .set-table2 > tbody > tr');
        },
        appendDivider() {
            return addElem(addElem(addElem(table, 'tbody', 'rinsp-infscroll-divider'), 'tr'), 'td', null, { colspan: '7' });
        },
        appendContentItems(items) {
            const nextTbody = addElem(table, 'tbody');
            items.forEach(item => nextTbody.appendChild(item));
        },
        getItemId(item) {
            const link = item.querySelector('input[name="delid[]"]');
            return link ? link.value : null;
        }
    }, userConfig, mainConfigAccess);

}

function submitOriginalForm(form) {
    submitPostForm(form.getAttribute('action'), new FormData(form), form.getAttribute('target')||null);
}

function submitPostForm(action, formData, target) {
    const newForm = addElem(document.body, 'form', null, {
        action: action,
        method: 'post',
        target: target||''
    });
    formData.forEach((value, key) => {
        addElem(newForm, 'input', null, { type: 'hidden', name: key, value: value });
    });
    newForm.submit();
}

//============= admin =============//

function renderPunishTags(uid, record) {
    const tags = newElem('a', 'rinsp-profile-punish-tags', { href: `u.php?action-trade-uid-${uid}.html`, target: '_blank' });
    let bloodCount = 0;
    let banCount = 0;
    let delCount = 0;
    let otherCount = 0;
    record.logs.forEach(log => {
        switch (log.type) {
            case PUNISH_TYPE_HP:
                bloodCount++;
                break;
            case PUNISH_TYPE_BAN:
                banCount++;
                break;
            case PUNISH_TYPE_DELETE_THREAD:
            case PUNISH_TYPE_DELETE_REPLY:
            case PUNISH_TYPE_SHIELD:
                delCount++;
                break;
            case PUNISH_TYPE_UNSHIELD:
                // skip
                break;
            default:
                otherCount++;
        }
    });
    const putCountTag = (count, tag) => {
        if (count > 0) {
            addElem(tags, 'span', 'rinsp-profile-punish-tag rinsp-profile-punish-tag-' + tag).textContent = `${count}`;
        }
    };
    putCountTag(bloodCount, 'blood');
    putCountTag(banCount, 'ban');
    putCountTag(delCount, 'del');
    putCountTag(otherCount, 'misc');
    return tags;
}

function initAdminFunctions(myUserId, userConfig, userPunishRecordStore) {
    const NO_PING_NOTIFICATION = 'no-ping-mail';
    const extraPageParams = {};
    if (document.location.hash) {
        const pairs = document.location.hash.substring(1).split('-');
        while (pairs.length > 0) {
            const key = pairs.shift();
            const value = pairs.shift();
            extraPageParams[key] = value;
        }
    }

    const adminTemplateConfigAccess = initConfigAccess(myUserId, ADMIN_TEMPLATE_CONFIG_KEY, DEFAULT_ADMIN_TEMPLATEUSER_FILTER_CONFIG);

    async function recordPunishment(uid, type, summary, reason, data) {
        let punishRecord = await userPunishRecordStore.get(uid);
        if (punishRecord == null) {
            punishRecord = {
                uid,
                logs: []
            };
        }
        punishRecord.logs.push({
            time: Date.now(),
            type,
            summary,
            reason,
            data
        });
        await userPunishRecordStore.put(uid, punishRecord);
    }

    async function openReasonTemplateEditor(callback) {
        openTextListEditor(SCORING_REASON_TEMPLATE_POPUP_MENU_ID, {
            title: '常用评分或操作理由列表',
            async read() {
                const adminTemplate = await adminTemplateConfigAccess.read();
                const text = adminTemplate.pingReasons.join('\n');
                if (text.length === 0) {
                    return DEFAULT_ADMIN_TEMPLATEUSER_FILTER_CONFIG.pingReasons.slice().join('\n');
                }
                return text;
            },
            async save(textData) {
                return await adminTemplateConfigAccess.update(function(adminTemplate) {
                    adminTemplate.pingReasons = textData.split('\n').map(itm=>itm.trim()).filter(itm=>!!itm);
                    return adminTemplate;
                });
            }
        }, callback);
    }
    
    async function enableQuickPingFunction() {
        let gfPost = getPosts(document, 1)[0];
        if (gfPost == null || gfPost.floor !== 0) {
            return;
        }
        const postTitleElem = document.querySelector('#subject_tpc');
        if (postTitleElem == null) {
            return;
        }
        const postTitle = postTitleElem.textContent.trim();
        const postTime = gfPost.postTime;
        const postUid = gfPost.postUid;

        const headtopicButton = document.querySelector('.fl > #headtopic');
        if (headtopicButton) { // scoring allowed
            const markElem = document.querySelector('#mark_tpc');
            if (markElem) {
                const pingStatus = newElem('a', 'rinsp-quickping-status');
                const score = (markElem.textContent.match(/SP币:([+-]\d+)\b/)||[])[1];
                if (score) {
                    pingStatus.textContent = '已评分: SP' + score;
                } else {
                    pingStatus.textContent = '已评分';
                }
                headtopicButton.parentNode.prepend(pingStatus);
            } else {
                const currentPostLink = document.querySelector('#breadcrumbs > .crumbs-item.current > strong > a[href^="read.php?tid-"]');
                if (currentPostLink == null)
                    return;
                const tid = Number.parseInt(currentPostLink.getAttribute('href').substring(13));
                const fidLink = currentPostLink.closest('.crumbs-item').previousElementSibling.getAttribute('href');
                const fid = Number.parseInt(fidLink.substring(15)); // e.g. "thread.php?fid-4.html"
                const defaultRating = getDefaultRating(postTitle);

                let instantPingButton = null;
                if (defaultRating.baseTotalScore > 0 && defaultRating.baseTotalScore < 99999) {
                    if (!defaultRating.ownBought && !defaultRating.ownTranslate) {
                        const notif = !userConfig.adminNoScoreNotifByDefault;
                        const score = Math.min(Math.max(1, Math.floor(defaultRating.baseTotalScore)), 99999);
                        instantPingButton = newElem('a', 'rinsp-instantping-button');
                        instantPingButton.setAttribute('href', 'javascript:void(0)');
                        instantPingButton.textContent = notif ? `一键SP+${score}` : `一键SP+${score} (不通知)`;
                        instantPingButton.addEventListener('click', () => {
                            executeScoreThenReload(null, fid, tid, score, 0, {}, notif, '');
                            return false;
                        });
                        headtopicButton.parentNode.prepend(instantPingButton);
                        headtopicButton.parentNode.prepend(document.createTextNode(' | '));
                    }
                }
                
                const quickPingButton = newElem('a', 'rinsp-quickping-button');
                quickPingButton.setAttribute('href', 'javascript:void(0)');
                quickPingButton.textContent = '快捷评分';
                quickPingButton.addEventListener('click', () => {
                    let dayAge = (Date.now() - postTime) / 86400000;
                    openScoringDialog(tid, fid, postTitle, dayAge, defaultRating);
                    return false;
                });
                headtopicButton.parentNode.prepend(quickPingButton);

                if (!userConfig.adminHideMarkBadFormatButton) {
                    headtopicButton.parentNode.append(document.createTextNode(' | '));

                    const badFormatButton = newElem('a', 'rinsp-noping-button');
                    badFormatButton.setAttribute('href', 'javascript:void(0)');
                    badFormatButton.textContent = '格式不正';
                    badFormatButton.addEventListener('click', () => {
                        executeScoreThenReload(null, fid, tid, -1, 0, {}, true, '请修正格式问题');
                        return false;
                    });
                    headtopicButton.parentNode.append(badFormatButton);
                }
                
                if (!userConfig.adminHideMarkUnscoreButton) {
                    headtopicButton.parentNode.append(document.createTextNode(' | '));
                    const noPingButton = newElem('a', 'rinsp-noping-button');
                    noPingButton.setAttribute('href', 'javascript:void(0)');
                    noPingButton.textContent = '标记不评分';
                    noPingButton.addEventListener('click', () => {
                        executeScoreThenReload(null, fid, tid, 1, 0, {}, false, '');
                        return false;
                    });
                    headtopicButton.parentNode.append(noPingButton);
                }

                const customNameEntry = userConfig.customUserHashIdMappings['#' + postUid];
                if (customNameEntry && customNameEntry[2].startsWith(NEVER_SCORE_USER_PREFIX)) {
                    quickPingButton.classList.add('rinsp-quickping-grey');
                    if (instantPingButton) {
                        instantPingButton.classList.add('rinsp-instantping-grey');
                    }
                }

            }
        }
    }

    async function enhanceReasonSelector() {
        const reasonField = document.querySelector('form[action] textarea[name="atc_content"]');
        if (reasonField == null) {
            return;
        }
        const reasonSelect = reasonField.nextElementSibling;
        if (reasonSelect == null || reasonSelect.tagName !== 'SELECT' || !!reasonSelect.getAttribute('name')) {
            return;
        }
        const replacement = addReasonSelector(reasonField);
        replacement.setAttribute('style', 'display:inline-block;vertical-align:top');
        reasonField.parentNode.insertBefore(replacement, reasonSelect);
        reasonSelect.remove();
    }

    function autoFillShowPingPage() {
        const postTitle = (document.querySelector('form[name="ping"] .tr3 > th > a[href^="read.php?tid-"]')||{}).textContent||'';
        const defaultRating = getDefaultRating(postTitle);
        const addPointField = document.querySelector('form[name="ping"] input[name="addpoint"]');
        if (defaultRating.baseTotalScore > 0) {
            addPointField.value = String(Math.max(1, Math.floor(defaultRating.baseTotalScore)));
        } else {
            addPointField.value = '';
            setTimeout(() => addPointField.focus());
        }
    }

    const OPTION_SEPARATOR_VALUE = ' NONE ';
    const OPTION_CLEAR_ACTION_VALUE = ' CLEAR ';

    async function updateReasonSelect(reasonSelect, reasonField) {
        function addOption(value, label) {
            let option = addElem(reasonSelect, 'option');
            option.setAttribute('value', value);
            option.textContent = label;
        }
        reasonSelect.innerHTML = '';
        addOption('', '自定义');
        addOption(OPTION_CLEAR_ACTION_VALUE, '清除');
        addOption(OPTION_SEPARATOR_VALUE, '-------');
        const adminTemplate = await adminTemplateConfigAccess.read();
        adminTemplate.pingReasons.forEach(item => {
            item = item.trim();
            if (item) {
                if (item.match(/---+/)) {
                    addOption(OPTION_SEPARATOR_VALUE, '-------');
                } else {
                    addOption(item.replace(/\|/g, '\n'), item.substring(0, 10));
                }
            }
        });
        reasonSelect.value = reasonField.value.trim();
        reasonSelect.value = reasonSelect.value||'';
    }

    function addReasonSelector(reasonField) {
        const reasonTemplateElem = newElem('div');
        const reasonSelect = addElem(reasonTemplateElem, 'select');
        reasonSelect.setAttribute('name', '');
        reasonSelect.setAttribute('size', '6');
        reasonSelect.addEventListener('change', () => {
            if (reasonSelect.value === '' || reasonSelect.value === OPTION_SEPARATOR_VALUE) {
                reasonSelect.value = '';
            } else if (reasonSelect.value === OPTION_CLEAR_ACTION_VALUE) {
                reasonSelect.value = '';
                reasonField.value = '';
            } else {
                reasonField.value = reasonSelect.value;
            }
        });
        reasonField.addEventListener('change', () => {
            reasonSelect.value = reasonField.value.trim();
            reasonSelect.value = reasonSelect.value||'';
        });
        addElem(reasonTemplateElem, 'div');
        const editTemplateButton = addElem(reasonTemplateElem, 'a');
        editTemplateButton.setAttribute('href', 'javascript:void(0)');
        editTemplateButton.textContent = '📝编辑常用列表';
        editTemplateButton.addEventListener('click', () => {
            openReasonTemplateEditor(() => updateReasonSelect(reasonSelect, reasonField));
        });

        updateReasonSelect(reasonSelect, reasonField);
        return reasonTemplateElem;
    }

    async function openScoringDialog(tid, fid, postTitle, postAgeInDays, defaultRating) {
        const preset = DEFAULT_SCORING_PRESETS[`fid=${fid}`] || DEFAULT_SCORING_PRESETS['fid=*'];
        const anchor = null;
        const modelMask = addModalMask();
        const popupMenu = createPopupMenu(SCORING_DIALOG_POPUP_MENU_ID, anchor);
    
        popupMenu.renderContent(async function(borElem) {

            addElem(borElem, 'div', 'rinsp-quickping-subject').textContent = postTitle;
            const form = addElem(borElem, 'dl', 'rinsp-quickping-form');
            addElem(form, 'dt').textContent = '发帖日期';
            addElem(form, 'dd').textContent = postAgeInDays <= 1 ? '一天内' : postAgeInDays <= 2 ? '昨天' : `${Math.floor(postAgeInDays)} 天前`;
            addElem(form, 'div');

            addElem(form, 'dt').textContent = '资源大小';
            const sizeCell = addElem(form, 'dd');
            addElem(form, 'div');
            addElem(form, 'dt').textContent = '基础评分';
            const baseScoreCell = addElem(form, 'dd');
            const baseScoreInputField = addElem(baseScoreCell, 'input');
            baseScoreInputField.setAttribute('size', '12');

            addElem(baseScoreCell, 'span').textContent = ' \u00A0 \u00A0 可加乘部分: ';

            const multiBaseScoreInputField = addElem(baseScoreCell, 'input');
            multiBaseScoreInputField.setAttribute('type', 'text');
            multiBaseScoreInputField.setAttribute('size', '10');
            multiBaseScoreInputField.setAttribute('placeholder', '(全部)');


            baseScoreInputField.setAttribute('type', 'text');
            const resourcePackToggles = [];
            if (defaultRating.baseTotalScore) {
                baseScoreInputField.value = defaultRating.baseTotalScore;
            } else {
                setTimeout(() => baseScoreInputField.focus());
            }
            if (defaultRating.resourceSizeTexts.length === 0) {
                sizeCell.textContent = '不明';
            } else if (defaultRating.resourceSizeTexts.length === 1) {
                sizeCell.textContent = defaultRating.resourceSizeTexts[0];
            } else {
                const computeAutofillScore = () => {
                    let totalSizeMB = 0;
                    resourcePackToggles.forEach(toggle => {
                        if (toggle.checkbox.checked) {
                            totalSizeMB += toggle.sizeMB;
                        }
                    });
                    baseScoreInputField.value = computeBaseScoreText(totalSizeMB);
                    updatePreview();
                };
                const updateSizeCheckboxes = () => {
                    const sizeMB = Math.ceil(baseScoreInputField.value * 10);
                    if (sizeMB > 0) {
                        let remain = sizeMB;
                        resourcePackToggles.forEach(toggle => {
                            if (remain >= toggle.sizeMB) {
                                remain -= toggle.sizeMB;
                                toggle.checkbox.checked = true;
                            } else {
                                toggle.checkbox.checked = false;
                            }
                        });
                    } else {
                        resourcePackToggles.forEach(toggle => toggle.checkbox.checked = false);
                    }
                };
                defaultRating.resourceSizeTexts.forEach((sizeText, i) => {
                    if (i > 0) {
                        addElem(sizeCell, 'span').textContent = ' ';
                    }
                    const item = addElem(sizeCell, 'label');
                    const checkbox = addElem(item, 'input', null, { type: 'checkbox' });
                    addElem(item, 'span', null, { style: 'vertical-align:bottom' }).textContent = sizeText;
                    resourcePackToggles.push({
                        checkbox,
                        sizeMB: defaultRating.resourceSizeMBs[i]
                    });
                    checkbox.addEventListener('change', computeAutofillScore);
                });
                resourcePackToggles.sort(comparator(toggle => toggle.sizeMB, true));
                baseScoreInputField.addEventListener('change', updateSizeCheckboxes);
                baseScoreInputField.addEventListener('keyup', updateSizeCheckboxes);
            }

            addElem(form, 'div');
            addElem(form, 'dt').textContent = '加亮置顶';
            const attCell = addElem(form, 'dd');

            const ownBoughtLabel = addElem(attCell, 'label', 'rinsp-quickping-attr-ownbought');
            const ownBoughtField = addElem(ownBoughtLabel, 'input');
            ownBoughtLabel.appendChild(document.createTextNode('自购 / 原创首发'));
            ownBoughtField.setAttribute('type', 'checkbox');
            if (defaultRating.ownBought) {
                ownBoughtField.checked = true;
            }

            const ownTranslateLabel = addElem(attCell, 'label', 'rinsp-quickping-attr-owntranslate');
            const ownTranslateField = addElem(ownTranslateLabel, 'input');
            ownTranslateLabel.appendChild(document.createTextNode('红字 (例: 个人汉化游戏)'));
            ownTranslateField.setAttribute('type', 'checkbox');

            const compilationLabel = addElem(attCell, 'label', 'rinsp-quickping-attr-compilation');
            const compilationField = addElem(compilationLabel, 'input');
            compilationLabel.appendChild(document.createTextNode('优秀合集'));
            compilationField.setAttribute('type', 'checkbox');

            const extraScores = [];
            preset.extraScoreAdjustments.forEach(extra => {
                const extraOffsetLabel = addElem(attCell, 'label', 'rinsp-quickping-attr-extra');
                const extraOffsetField = addElem(extraOffsetLabel, 'input');
                extraOffsetLabel.appendChild(document.createTextNode(extra.label));
                extraOffsetField.setAttribute('type', 'checkbox');
                extraScores.push({
                    amount: extra.amount,
                    reason: extra.reason||'',
                    checkbox: extraOffsetField
                });
            });


            addElem(form, 'div');
            addElem(form, 'dt').textContent = '评分加乘';
            const multiScoreCell = addElem(form, 'dd');
            const multiScoreInputField = addElem(multiScoreCell, 'input');
            multiScoreInputField.setAttribute('type', 'text');
            multiScoreInputField.setAttribute('size', '12');
            
            addElem(form, 'div');
            addElem(form, 'dt').textContent = '置顶';
            const headtopCell = addElem(form, 'dd');
            const headtopInputField = addElem(headtopCell, 'input');
            headtopInputField.setAttribute('type', 'text');
            headtopInputField.setAttribute('size', '12');

            addElem(form, 'div');
            addElem(form, 'dt').textContent = '其他';
            const notifyCell = addElem(form, 'dd');
            const notifyLabel = addElem(notifyCell, 'label');
            const notifyField = addElem(notifyLabel, 'input');
            notifyLabel.appendChild(document.createTextNode('评分消息通知'));
            notifyField.setAttribute('type', 'checkbox');
            addElem(notifyCell, 'small').textContent = ' (加亮及置顶消息一律自动通知)';

            // NO_PING_NOTIFICATION
            notifyField.checked = !userConfig.adminNoScoreNotifByDefault;

            addElem(form, 'div');
            addElem(form, 'dt').textContent = '评分理由';
            const reasonCell = addElem(form, 'dd', 'rinsp-quickping-reason-cell');
            const reasonField = addElem(reasonCell, 'textarea');
            const templateColumn = addElem(reasonCell, 'div');
            templateColumn.appendChild(addReasonSelector(reasonField));

            addElem(form, 'div');
            addElem(form, 'dt').textContent = '最终效果';
            const previewSection = addElem(form, 'div', 'rinsp-quickping-preview-section');
            const previewCell = addElem(previewSection, 'div');

            baseScoreInputField.addEventListener('change', () => updatePreview());
            baseScoreInputField.addEventListener('keyup', () => updatePreview());
            multiBaseScoreInputField.addEventListener('change', () => updatePreview());
            multiBaseScoreInputField.addEventListener('keyup', () => updatePreview());
            multiScoreInputField.addEventListener('change', () => updatePreview());
            multiScoreInputField.addEventListener('keyup', () => updatePreview());
            headtopInputField.addEventListener('change', () => updatePreview());
            headtopInputField.addEventListener('keyup', () => updatePreview());
            ownBoughtField.addEventListener('change', () => updatePreview());
            ownTranslateField.addEventListener('change', () => updatePreview());
            compilationField.addEventListener('change', () => updatePreview());

            extraScores.forEach(extra => {
                extra.checkbox.addEventListener('change', () => {
                    if (extra.reason) {
                        const currentReason = reasonField.value.trim();
                        if (extra.checkbox.checked) {
                            if (currentReason.length === 0) {
                                reasonField.value = extra.reason;
                            }
                        } else {
                            if (currentReason === extra.reason) {
                                reasonField.value = '';
                            }
                        }
                    }
                    
                    updatePreview();
                });
            });


            function updatePreview() {
                previewCell.textContent = '';
                previewCell.setAttribute('class', '');

                const defaultHeadtopDays = getDefaultHeadtopDays();
                if (defaultHeadtopDays == null) {
                    headtopInputField.setAttribute('placeholder', '自动 (无)');
                } else {
                    if (defaultHeadtopDays[0] > 0) {
                        headtopInputField.setAttribute('placeholder', `自动 (${defaultHeadtopDays[0]}天)`);
                    } else if (defaultHeadtopDays[1] > 0) {
                        headtopInputField.setAttribute('placeholder', '自动 (旧帖省略)');
                    } else {
                        headtopInputField.setAttribute('placeholder', '自动 (无)');
                    }
                }

                multiScoreInputField.setAttribute('placeholder', '自动 (无)');
                const finalScore = getFinalScore();
                const headtopDays = getFinalHeadtopDays();

                if (finalScore == null || headtopDays == null) {
                    previewCell.classList.add('rinsp-quickping-status-error');
                    if (!baseScoreInputField.value.trim()) {
                        previewCell.textContent = '';
                    } else {
                        previewCell.textContent = '输入错误 请更正';
                    }
                } else {
                    if (finalScore.total < 0) {
                        let text = addElem(previewCell, 'span');
                        text.appendChild(document.createTextNode('扣分 '));
                        addElem(previewCell, 'b', null, { style: 'color: red' }).textContent = `${finalScore.total} SP`;
                    } else if (finalScore.multiplier > 1) {
                        let text = addElem(previewCell, 'span');
                        text.appendChild(document.createTextNode('评分'));
                        if (finalScore.nonboostedScore > 0) {
                            text.appendChild(document.createTextNode(` +${finalScore.boostedScore} SP`));
                            addElem(text, 'small').textContent = `(x${finalScore.multiplier})`;
                            text.appendChild(document.createTextNode(` +${finalScore.nonboostedScore} SP`));
                        } else {
                            addElem(text, 'small').textContent = `(x${finalScore.multiplier})`;
                            text.appendChild(document.createTextNode(` +${finalScore.total} SP`));
                        }
                        multiScoreInputField.setAttribute('placeholder', `自动 (x${finalScore.multiplier})`);
                    } else {
                        addElem(previewCell, 'span').textContent = `评分 +${finalScore.total} SP`;
                        multiScoreInputField.setAttribute('placeholder', '自动 (无)');
                    }

                    if (headtopDays > 0) {
                        addElem(previewCell, 'span').textContent = ' | ';
                        addElem(previewCell, 'span').textContent = '置顶 ' + headtopDays + '天';
                    }
                    
                    const hlParams = getHighlightParams();
                    if (hlParams.color || hlParams.bold) {
                        addElem(previewCell, 'span').textContent = ' | ';
                        const hlElem = addElem(previewCell, hlParams.bold ? 'b' : 'span');
                        hlElem.textContent = '加亮';
                        if (hlParams.color) {
                            hlElem.setAttribute('style', 'color:' + hlParams.color);
                        }
                    }
                }
                
            }
            updatePreview();

            function getHighlightParams() {
                let color = compilationField.checked ? '#FF00FF' : ownBoughtField.checked ? '#0000FF' : ownTranslateField.checked ? '#FF0000' : null;
                let bold = compilationField.checked || ownBoughtField.checked || ownTranslateField.checked;
                return { color, bold };
            }

            function getDefaultScoreMultiplier() {
                return ownBoughtField.checked || ownTranslateField.checked ? 10 : 1;
            }

            function getDefaultHeadtopDays() {
                const baseDays = compilationField.checked ? preset.headtopDaysCompilation : ownBoughtField.checked || ownTranslateField.checked ? preset.headtopDaysBought : 0;
                if (postAgeInDays >= baseDays) {
                    return [0, baseDays];
                }
                return [baseDays, baseDays];
            }

            function getFinalHeadtopDays() {
                if (headtopInputField.value.trim().length === 0) {
                    return getDefaultHeadtopDays()[0];
                } else {
                    let days = headtopInputField.value.trim() * 1;
                    if (!Number.isInteger(days) || days < 0)
                        return null; // invalid
                    return days;
                }
            }

            function getFinalScore() {
                let scoreMultiplier;
                if (multiScoreInputField.value.trim().length === 0) {
                    scoreMultiplier = getDefaultScoreMultiplier();
                } else {
                    scoreMultiplier = multiScoreInputField.value.trim() * 1;
                    if (!Number.isInteger(scoreMultiplier) || scoreMultiplier < 1)
                        return null; // invalid
                }
                let baseScore = baseScoreInputField.value * 1;
                if (Number.isNaN(baseScore)) {
                    baseScore = getDefaultRating(baseScoreInputField.value).baseTotalScore;
                }

                let boostableBaseScore = (multiBaseScoreInputField.value.trim()||'9999.9') * 1;
                if (Number.isNaN(boostableBaseScore)) {
                    boostableBaseScore = getDefaultRating(multiBaseScoreInputField.value).baseTotalScore;
                }

                if (Number.isNaN(baseScore) || baseScore == 0 || boostableBaseScore === 0) {
                    return null;
                }
                let scoreOffset = 0;
                for (let extraScore of extraScores) {
                    if (extraScore.checkbox.checked) {
                        scoreOffset += extraScore.amount;
                    }
                }
                if (baseScore < 0) {
                    if (scoreMultiplier === 1) {
                        const rawScore = Math.max(-99999, Math.min(-1, Math.floor(baseScore + scoreOffset)));
                        return {
                            total: rawScore,
                            multiplier: 1,
                            boostedScore: 0,
                            nonboostedScore: rawScore
                        };
                    } else {
                        return null;
                    }
                }
                if (scoreMultiplier > 1) {
                    let boostedScore = Math.floor(Math.min(baseScore, boostableBaseScore) * scoreMultiplier);
                    let nonboostedScore = Math.floor(Math.max(0, baseScore - boostableBaseScore)) + scoreOffset;
                    if (boostedScore + nonboostedScore === 0) {
                        nonboostedScore = 1;
                    }
                    return {
                        total: boostedScore + nonboostedScore,
                        multiplier: scoreMultiplier,
                        boostedScore: boostedScore,
                        nonboostedScore: nonboostedScore
                    };
                } else {
                    const rawScore = Math.max(1, Math.floor(baseScore * scoreMultiplier)) + scoreOffset;
                    return {
                        total: rawScore,
                        multiplier: scoreMultiplier,
                        boostedScore: 0,
                        nonboostedScore: rawScore
                    };
                }
            }

            const footer = addElem(borElem, 'ul', null, {
                style: 'text-align:center;padding:4px 0;'
            });
            const submitButton = addElem(footer, 'input', 'btn', { type: 'button', value: '执行评分' });
            submitButton.addEventListener('click', async function() {
                const finalHeadtopDays = getFinalHeadtopDays();
                const finalScore = getFinalScore();
                if (finalScore == null || finalHeadtopDays == null) {
                    alert('输入错误 请更正');
                } else {
                    modelMask.remove();
                    closePopupMenu(SCORING_DIALOG_POPUP_MENU_ID);
                    const notif = notifyField.checked ? true : NO_PING_NOTIFICATION;
                    executeScoreThenReload(anchor, fid, tid, finalScore.total, finalHeadtopDays, getHighlightParams(), notif, reasonField.value.trim());
                }

            });

            footer.appendChild(document.createTextNode(' '));

            const closeButton = addElem(footer, 'input', 'btn', { type: 'button', value: '关闭' });
            closeButton.addEventListener('click', function() {
                modelMask.remove();
                closePopupMenu(SCORING_DIALOG_POPUP_MENU_ID);
            });
        });
    }

    const RETRY_SIGNAL = 'RETRY_SIGNAL';
    async function readResponse(resp) {
        const respText = await resp.text();
        if (respText.indexOf('刷新不要快于') !== -1) {
            return Promise.resolve(RETRY_SIGNAL);
        } else if (!respText.match(/(操作完成|发帖完毕点击进入主题列表)/)) {
            const respDoc = new DOMParser().parseFromString(respText, 'text/html');
            if (((respDoc.querySelector('#main .t table .h')||{}).textContent||'').trim().endsWith('提示信息')) {
                return Promise.reject(((respDoc.querySelector('#main .t table .f_one center')||{}).textContent||'').trim());
            } else {
                return Promise.reject('不明错误');
            }
        } else {
            return Promise.resolve(true);
        }
    }

    let lastRequestTime = 0;

    async function attemptRequest() {
        let interval = Date.now() - lastRequestTime;
        if (interval < MIN_REQUEST_INTERVAL) {
            await sleep(MIN_REQUEST_INTERVAL - interval);
        }
        lastRequestTime = Date.now();
    }

    async function executeWithRetry(action) {
        let returnCode = await action();
        while (returnCode === RETRY_SIGNAL) {
            returnCode = await action();
        }
        return returnCode;
    }

    function encodeData(data) {
        return new URLSearchParams(data).toString();
    }

    async function executeScoreThenReload(anchor, fid, tid, addPoint, headtopDays, highlightParams, sendNotif, reason, pid) {
        const action = async () => {
            await executeScoreFunction(fid, tid, addPoint, headtopDays, highlightParams, sendNotif, reason, pid);
            return '✔️评分操作已完成';
        };
        runWithProgressPopup(action, '评分操作中...', anchor, 1500)
            .catch((ex) => alert(String(ex)))
            .finally(() => document.location.reload());
    }

    async function executeDeletePost(fid, tid, reason) {
        const url = `${document.location.origin}/mawhole.php`;
        await attemptRequest();
        await fetch(url, {
            method: 'POST',
            mode: 'same-origin',
            credentials: 'same-origin',
            headers: {
                'Content-Type': 'application/x-www-form-urlencoded'
            },
            body: encodeData({
                'verify': verifyhash(),
                'action': 'del',
                'fid': fid,
                'tidarray[]': tid,
                'step': 2,
                'ifdel': 1,
                'ifmsg': 1,
                'atc_content': reason||'',
            })
        })
        .then(async resp => {
            await readResponse(resp);
            return true;
        });
    }

    async function executeDeleteReply(fid, tid, pid, reason) {
        const url = `${document.location.origin}/masingle.php?action=delatc`;
        await attemptRequest();
        await fetch(url, {
            method: 'POST',
            mode: 'same-origin',
            credentials: 'same-origin',
            headers: {
                'Content-Type': 'application/x-www-form-urlencoded'
            },
            body: encodeData({
                'verify': verifyhash(),
                'fid': fid,
                'tid': tid,
                'step': 3,
                'selid[]': pid,
                'ifdel': 1,
                'ifmsg': 1,
                'atc_content': reason||'',
            })
        })
        .then(async resp => {
            await readResponse(resp);
            return true;
        });
    }

    async function executeScoreFunction(fid, tid, addPoint, headtopDays, highlightParams, sendNotif, reason, pid) {

        async function executePing(sp) {
            const url = `${document.location.origin}/operate.php?action=showping`;
            await attemptRequest();
            await fetch(url, {
                method: 'POST',
                mode: 'same-origin',
                credentials: 'same-origin',
                headers: {
                    'Content-Type': 'application/x-www-form-urlencoded'
                },
                body: encodeData({
                    'verify': verifyhash(),
                    'selid[]': pid||'tpc',
                    'cid': 'money',
                    'addpoint': sp,
                    'step': 1,
                    'ifmsg': sendNotif && sendNotif !== NO_PING_NOTIFICATION ? 1 : 0,
                    'atc_content': reason||'',
                    'page': 1,
                    'tid': tid
                })
            })
            .then(async resp => {
                await readResponse(resp);
                return true;
            });
        }

        async function executeHighlight() {
            if (!highlightParams.color && !highlightParams.bold)
                return Promise.resolve(true);
            const url = `${document.location.origin}/mawhole.php`;
            await attemptRequest();
            await fetch(url, {
                method: 'POST',
                mode: 'same-origin',
                credentials: 'same-origin',
                headers: {
                    'Content-Type': 'application/x-www-form-urlencoded'
                },
                body: encodeData({
                    'verify': verifyhash(),
                    'action': 'edit',
                    'fid': fid,
                    'tidarray[]': tid,
                    'title1': highlightParams.color||'',
                    'title2': highlightParams.bold ? 1 : '',
                    'title3': '',
                    'title4': '',
                    'ifmsg': sendNotif ? 1 : 0,
                    'atc_content': '',
                    'timelimit': '',
                    'nextto': '',
                    'step': 2
                })
            })
            .then(async resp => {
                await readResponse(resp);
                return null;
            });
        }

        async function executeHeadtopic() {
            if (headtopDays <= 0)
                return Promise.resolve(true);
            const url = `${document.location.origin}/mawhole.php`;
            await attemptRequest();
            await fetch(url, {
                method: 'POST',
                mode: 'same-origin',
                credentials: 'same-origin',
                headers: {
                    'Content-Type': 'application/x-www-form-urlencoded'
                },
                body: encodeData({
                    'verify': verifyhash(),
                    'action': 'headtopic',
                    'fid': fid,
                    'tidarray[]': tid,
                    'topped': 1,
                    'timelimit': headtopDays,
                    'ifmsg': sendNotif ? 1 : 0,
                    'atc_content': '',
                    'nextto': '',
                    'step': 2
                })
            })
            .then(async resp => {
                await readResponse(resp);
                return null;
            });
        }

        const extraBatches = Math.floor(addPoint / 99999);
        if (extraBatches > 0) {
            const lastLeftover = addPoint % 99999;
            await executeWithRetry(async () => executePing(lastLeftover));
            const ping99999 = async () => executePing(99999);
            for (let i = 0; i < extraBatches; i++) {
                await executeWithRetry(ping99999);
            }
        } else {
            await executeWithRetry(async () => executePing(addPoint));
        }

        await executeWithRetry(executeHighlight);
        await executeWithRetry(executeHeadtopic);

    }

    function enhanceShieldActionPage() {
        const uid = extraPageParams['uid'];
        if (uid == null) {
            return;
        }
        const form = document.querySelector('form[action="masingle.php?action=shield"]');
        if (form == null) {
            return;
        }

        const userNameCell = form.querySelector('table > tbody > tr + tr.tr3 > td.tar + td');
        const userName = userNameCell.textContent.trim();
        userNameCell.innerHTML = '';
        addElem(userNameCell, 'a', null, { href: `u.php?action-show-uid-${uid}.html`, target: '_blank' }).textContent = userName;
        if (uid !== myUserId) {
            userNameCell.append(document.createTextNode(' / 🚩记录已启用'));
        }
        
        async function handleSubmit() {
            const formData = new FormData(form);
            const step = formData.get('step') * 1;
            const fid = formData.get('fid') * 1;
            const tid = formData.get('tid') * 1;
            const pid = formData.get('pid') * 1 || 0; // tpc = 0
            const reason = formData.get('atc_content');
            if (uid !== myUserId) {
                await recordPunishment(uid, step === 5 ? PUNISH_TYPE_UNSHIELD : PUNISH_TYPE_SHIELD, '', reason, {
                    fid,
                    tid,
                    pid
                });
            }
            submitOriginalForm(form);
        }

        form.addEventListener('submit', evt => {
            evt.preventDefault();
            handleSubmit();
            return false;
        });
    }

    // ban user is missing the reason preset selector
    function enhanceBanUserActionPage() {
        const form = document.querySelector('form[action="masingle.php?action=banuser"]');
        if (form == null) {
            return;
        }
        const reasonField = form.querySelector('td > textarea[name="atc_content"]');
        if (reasonField == null) {
            return;
        }
        const userNameCell = form.querySelector('table > tbody > tr + tr.tr3 > td.tar + td');
        const userName = userNameCell.textContent.trim();
        const uid = extraPageParams['uid'];
        if (uid) {
            userNameCell.innerHTML = '';
            addElem(userNameCell, 'a', null, { href: `u.php?action-show-uid-${uid}.html`, target: '_blank' }).textContent = userName;
            userNameCell.append(document.createTextNode(' / 🚩记录已启用'));
        }
        
        const mainAreaName = form.querySelector('input[type="radio"][name="range"][value="0"]').nextSibling.textContent.trim();
        const initFormData = new FormData(form);
        const reasonSelector = addReasonSelector(reasonField);
        reasonField.setAttribute('style', 'width: 250px; height: 135px;');
        reasonSelector.setAttribute('style', 'display:inline-block;vertical-align:top');
        reasonField.parentElement.appendChild(reasonSelector);
        const delRow = newElem('tr', 'tr3');
        addElem(delRow, 'td', 'tar').textContent = '追加操作:';
        const extraOptionCell = addElem(delRow, 'td');
        const delCheckBox = addElem(extraOptionCell, 'input', null, { type: 'checkbox' });
        extraOptionCell.append(document.createTextNode(` 删除${initFormData.get('pid') === 'tpc' ? '主题 (注意:如帖上有其他违规回复,请先返回处理)' : '回复'}`));
        const reasonRow = reasonField.closest('tr');
        reasonRow.parentElement.insertBefore(delRow, reasonRow.previousElementSibling);
        form.addEventListener('submit', evt => {
            evt.preventDefault();
            const formData = new FormData(form);
            handleBanAndDelete(formData, delCheckBox.checked);
            return false;
        });

        async function handleBanAndDelete(banFormData, alsoDelete) {
            const pid = banFormData.get('pid');
            const fid = banFormData.get('fid');
            const tid = banFormData.get('tid');
            const page = banFormData.get('page');
            const limit = banFormData.get('type') * 1 === 1 ? banFormData.get('limit') : -1;

            let actionSummary = '禁言';
            if (limit > 0) {
                actionSummary += ` (${limit}天)`;
            } else {
                actionSummary += ' (永久)';
            }
            if (banFormData.get('range') * 1 === 0) {
                actionSummary += ` / 版块 (${mainAreaName})`;
            }
            const reason = banFormData.get('atc_content');
            if (uid) {
                await recordPunishment(uid, PUNISH_TYPE_BAN, actionSummary, reason, {
                    fid,
                    tid,
                    pid,
                    banDur: limit,
                    areaName: mainAreaName,
                    delete: alsoDelete
                });
            }

            async function executeDeleteViolationMaterial() {
                if (alsoDelete) {
                    if (pid === 'tpc') {
                        return await executeDeletePost(fid, tid, reason);
                    } else {
                        return await executeDeleteReply(fid, tid, pid, reason);
                    }
                } else {
                    return true;
                }
            }
            function finish() {
                if (!alsoDelete) {
                    document.location.href = `${document.location.origin}/read.php?tid=${tid}&page=${page||1}#${pid}`;
                } else if (pid === 'tpc') {
                    document.location.href = `${document.location.origin}/u.php?action=show&username=${encodeURIComponent(userName)}`;
                } else {
                    document.location.href = `${document.location.origin}/read.php?tid=${tid}&page=${page||1}`;
                }
            }

            addModalMask();
            const popupMenu = createPopupMenu(LOADING_DIALOG_POPUP_MENU_ID, null);
            popupMenu.renderContent(async function(borElem) {
                const operationLabel = alsoDelete ? '禁言及删除' : '禁言';
                // not happy about copy-n-paste ...
                borElem.innerHTML = `<div style="padding:16px 30px"><img src="images/loading.gif" align="absbottom">${operationLabel}操作中...</div>`;
    
                executeWithRetry(executeBan)
                    .then(() => executeWithRetry(executeDeleteViolationMaterial))
                    .then(() => {
                        borElem.innerHTML = `<div style="padding:16px 30px;font-size:1.5em">✔️${operationLabel}操作已完成</div>`;
                        setTimeout(() => {
                            closePopupMenu(LOADING_DIALOG_POPUP_MENU_ID);
                            finish();
                        }, 1500);
                    })
                    .catch(ex => {
                        alert(String(ex));
                        finish();
                    });
            });

            async function executeBan() {
                const url = `${document.location.origin}/masingle.php?action=banuser`;
                await attemptRequest();
                await fetch(url, {
                    method: 'POST',
                    mode: 'same-origin',
                    credentials: 'same-origin',
                    headers: {
                        'Content-Type': 'application/x-www-form-urlencoded'
                    },
                    body: encodeData(banFormData)
                })
                .then(async resp => {
                    await readResponse(resp);
                    return true;
                });
            }
        }

    }

    function addUserList(target, uids) {
        target.append(document.createTextNode(' / 对象: '));
        uids.forEach(uid => {
            target.append(document.createTextNode(' '));
            addElem(target, 'a', null, {
                href: 'u.php?action-show-uid-' + uid + '.html',
                target: '_blank'
            }).textContent = '#' + uid;
        });
    }

    function enhanceMawholeActionPage() {
        const form = document.querySelector('form[action="mawhole.php?"]');
        if (form == null) {
            return;
        }
        const actionField = form.querySelector('input[type="hidden"][name="action"]');
        if (actionField == null) {
            return;
        }
        if (actionField.value === 'del') {
            enhanceMawholeDelActionPage(form);
        }
    }

    function enhanceMawholeDelActionPage(form) {
        const uids = (extraPageParams.uids||'').split(':').map(x=>x*1).filter(x=>x>0);
        const tids = (extraPageParams.tids||'').split(':').map(x=>x*1).filter(x=>x>0);
        if (uids.length === 0 || uids.length !== tids.length) {
            return;
        }
        const ifdelCell = form.querySelector('input[type="radio"][name="ifdel"]').closest('th');
        if (uids.filter(uid => uid !== myUserId).length > 0) {
            addElem(ifdelCell, 'span', 'rinsp-record-delatc-msg').textContent = ' / 🚩记录已启用';
        }

        const mappings = new Map();

        tids.forEach((tid, i) => mappings.set(tid, uids[i]));

        async function handleSubmit() {
            const formData = new FormData(form);
            if (formData.get('ifdel') === '1') {
                const fid = formData.get('fid') * 1;
                const tids = formData.getAll('tidarray[]').map(x=>x*1);
                const reason = formData.get('atc_content');
                for (let tid of tids) {
                    const uid = mappings.get(tid);
                    if (uid && uid !== myUserId) {
                        const threadTitle = form.querySelector(`th > a[href="read.php?tid-${tid}.html"]`).textContent;
                        await recordPunishment(uid, PUNISH_TYPE_DELETE_THREAD, '', reason, {
                            fid,
                            tid,
                            title: threadTitle
                        });
                    }
                }
            }
            submitOriginalForm(form);
        }

        form.addEventListener('submit', evt => {
            evt.preventDefault();
            handleSubmit();
            return false;
        });
    }

    function enhanceDeleteReplyActionPage() {
        const form = document.querySelector('form[action="masingle.php?action=delatc"]');
        if (form == null) {
            return;
        }
        const countDisplayCell = form.querySelector('table > tbody > tr + tr.tr3 > td.tar + td');
        const ifdelCell = form.querySelector('input[type="radio"][name="ifdel"]').closest('td');
        const uids = (extraPageParams.uids||'').split(':').map(x=>x*1).filter(x=>x>0);
        if (uids.length > 0) {
            addUserList(countDisplayCell, uids);
            if (uids.filter(uid => uid !== myUserId).length > 0) {
                addElem(ifdelCell, 'span', 'rinsp-record-delatc-msg').textContent = ' / 🚩记录已启用';
            }
        }

        async function handleSubmit() {
            const formData = new FormData(form);
            if (formData.get('ifdel') === '1') {
                const tid = formData.get('tid') * 1;
                const fid = formData.get('fid') * 1;
                const reason = formData.get('atc_content');
                for (let uid of uids) {
                    if (uid !== myUserId) {
                        await recordPunishment(uid, PUNISH_TYPE_DELETE_REPLY, '', reason, {
                            tid,
                            fid
                        });
                    }
                }
            }
            submitOriginalForm(form);
        }

        form.addEventListener('submit', evt => {
            evt.preventDefault();
            handleSubmit();
            return false;
        });
    }

    function enhanceEndRewardActionPage() {
        const form = document.querySelector('form[action="job.php?action=endreward"]');
        if (form == null) {
            return;
        }
        const bounty = extraPageParams.bounty * 1;
        if (Number.isNaN(bounty)) {
            return;
        }
        const pts = extraPageParams.pts * 1 || 0;
        const opsp = extraPageParams.opsp * 1 || 0;

        const ifmsgRow = form.querySelector('input[name="ifmsg"]').closest('tr');
        const statusRow = newElem('tr', 'tr3');
        addElem(statusRow, 'th').textContent = '悬赏';
        addElem(statusRow, 'th').textContent = `最佳答案: ${bounty} SP` + (pts > 0 ? ` | 热心助人: ${pts}` : '') + ` | 楼主: ${opsp.toLocaleString('en-US')} SP`;
        ifmsgRow.parentNode.insertBefore(statusRow, ifmsgRow.previousElementSibling);

        const spRow = newElem('tr', 'tr3');
        addElem(spRow, 'th').textContent = 'SP 影响';
        const spCell = addElem(spRow, 'th');
        const spInput = addElem(spCell, 'input', null, { type: 'text', size: 4 });
        spCell.appendChild(document.createTextNode(' SP'));
        ifmsgRow.parentNode.insertBefore(spRow, ifmsgRow);

        const headerCell = form.querySelector('table tr > td.h');
        headerCell.removeAttribute('colspan');
        headerCell.parentNode.insertBefore(newElem('td', 'h'), headerCell);
        function addPresetButton(label, type, sp, ifmsg) {
            const presetButton = addElem(headerCell, 'a', 'rinsp-ping-preset-switch', { href: 'javascript:' });
            presetButton.addEventListener('click', () => {
                form.querySelector(`input[name="type"][value="${type}"]`).checked = true;
                spInput.value = sp == null ? '' : '' + sp;
                if (ifmsg) {
                    form.querySelector('input[name="ifmsg"][value="1"]').checked = true;
                } else {
                    form.querySelector('input[name="ifmsg"][value="0"]').checked = true;
                }
            });
            presetButton.textContent = label;
            headerCell.appendChild(document.createTextNode(' '));
            return presetButton;
        }
        addPresetButton('取消悬赏', 1, bounty * 2 + pts, true).setAttribute('style', 'color:#080');
        addPresetButton('返还押金', 2, bounty, true);
        addPresetButton('强行结案 (2倍)', 2, 0, true).setAttribute('style', 'color:#60C');
        addPresetButton('不结帖 (3倍)', 2, -bounty, true).setAttribute('style', 'color:#D00');

        form.addEventListener('submit', evt => {
            const sp = (spInput.value||'0') * 1;
            if (Number.isNaN(sp)) {
                evt.preventDefault();
                return false;
            }
            if (sp === 0) {
                return;
            }
            evt.preventDefault();

            const formData = new FormData(form);
            const tid = formData.get('tid');

            async function executeEndReward() {
                await attemptRequest();
                await fetch(form.action, {
                    method: 'POST',
                    mode: 'same-origin',
                    credentials: 'same-origin',
                    headers: {
                        'Content-Type': 'application/x-www-form-urlencoded'
                    },
                    body: encodeData(formData)
                })
                .then(async resp => {
                    await readResponse(resp);
                    return true;
                });
            }
            async function executeScore() {
                await attemptRequest();
                const reason = sp < 0 ? '不结帖' : '';
                await executeScoreFunction(48, tid, sp, 0, {}, true, reason);
            }
            function finish() {
                document.location.href = `${document.location.origin}/read.php?tid-${tid}.html`;
            }

            addModalMask();
            executeWithRetry(executeEndReward)
                .then(() => executeScore())
                .then(() => {
                    finish();
                })
                .catch(ex => {
                    alert(String(ex));
                    finish();
                });

            return false;
        });

    }

    function enhancePingActionPage() {
        const form = document.querySelector('form[action="operate.php?action=showping"]');
        if (form == null) {
            return;
        }
        const countDisplayCell = form.querySelector('table > tbody > tr + tr.tr3 + tr.tr3 > td.tar + th');
        const pingStepCell = form.querySelector('input[name="step"]').closest('th');
        const cidInput = form.querySelector('select[name="cid"]');
        const addpointInput = form.querySelector('input[name="addpoint"]');
        let logCheckbox = null;
        let onUpdate = () => {};
        function addPresetButton(label, cid, addpoint, log, ifmsg, reason) {
            const target = form.querySelector('table tr > th.h + th.h');
            const presetButton = addElem(target, 'a', 'rinsp-ping-preset-switch', { href: 'javascript:' });
            presetButton.addEventListener('click', () => {
                cidInput.value = cid;
                addpointInput.value = '' + addpoint;
                if (logCheckbox) {
                    logCheckbox.checked = log;
                }
                form.querySelector('input[name="step"][value="1"]').checked = true;
                if (ifmsg) {
                    form.querySelector('input[name="ifmsg"][value="1"]').checked = true;
                } else {
                    form.querySelector('input[name="ifmsg"][value="0"]').checked = true;
                }
                form.querySelector('textarea[name="atc_content"]').value = reason;
                onUpdate();
            });
            presetButton.textContent = label;
            target.appendChild(document.createTextNode(' '));
            
        }
        addPresetButton('清除', 'money', 0, true, true, '');
        addPresetButton('done', 'money', 1, true, true, 'done');
        addPresetButton('+5', 'money', 5, true, true, '');
        addPresetButton('+50', 'money', 50, true, true, '');

        const uids = (extraPageParams.uids||'').split(':').map(x=>x*1).filter(x=>x>0);
        if (uids.filter(uid => uid !== myUserId).length > 0) {
            addUserList(countDisplayCell, uids);
            const recordOption = addElem(pingStepCell, 'span', 'rinsp-record-showping-option rinsp-hide');
            recordOption.append(document.createTextNode(' / '));
            logCheckbox = addElem(recordOption, 'input', null, { type: 'checkbox', style: 'vertical-align: text-bottom' });
            logCheckbox.checked = true;
            recordOption.append(document.createTextNode('🚩启用扣分记录'));
            onUpdate = () => {
                if (addpointInput.value.startsWith('-')) {
                    recordOption.classList.remove('rinsp-hide');
                } else {
                    recordOption.classList.add('rinsp-hide');
                }
            };
            addpointInput.addEventListener('keyup', evt => onUpdate());
        }
        const soldSp = extraPageParams.sold * 1;
        if (soldSp > 0) {
            if (extraPageParams.badsell === 'opsell') {
                addPresetButton('楼主出售', 'money', -soldSp, true, true, '禁止楼主出售');
            } else {
                addPresetButton('违例出售', 'money', -soldSp, true, true, '');
            }
        }
        const ansSp = extraPageParams.answer * 1;
        if (ansSp > 0) {
            addPresetButton('小号自收', 'money', -ansSp, true, true, '禁止小号自收');
        }

        async function handleSubmit() {
            if (logCheckbox && logCheckbox.checked) {
                const formData = new FormData(form);
                const addpoint = formData.get('addpoint') * 1;
                const step = formData.get('step') * 1;

                let punishType = null;
                let actionSummary = '';
                if (step === 1 && addpoint < 0) {
                    const cid = formData.get('cid');
                    if (cid === 'money') {
                        punishType = PUNISH_TYPE_SP;
                        actionSummary = 'SP' + addpoint;
                    } else if (cid === 'rvrc') {
                        punishType = PUNISH_TYPE_HP;
                        actionSummary = 'HP' + addpoint;
                    }
                }
                if (punishType) {
                    const tid = formData.get('tid') * 1;
                    const reason = formData.get('atc_content');
                    for (let uid of uids) {
                        if (uid !== myUserId) { // actually impossible at this point
                            await recordPunishment(uid, punishType, actionSummary, reason, {
                                tid,
                                amount: addpoint
                            });
                        }
                    }
                }
            }
            submitOriginalForm(form);
        }

        form.addEventListener('submit', evt => {
            evt.preventDefault();
            handleSubmit();
            return false;
        });

    }

    function showBatchPingWindow(items) {
        const modelMask = addModalMask();
        const popupMenu = createPopupMenu(SCORING_DIALOG_POPUP_MENU_ID, null, false);
        popupMenu.renderContent(async function(borElem) {
            const eForm = addElem(borElem, 'div', 'rinsp-batchsel-form rinsp-batchping-form');
            const tableElem = addElem(eForm, 'table', null, {
                width: '850',
                cellspacing: '0',
                cellpadding: '0',
                style: 'table-layout:fixed'
            });

            const colgroup = addElem(tableElem, 'colgroup');
            addElem(colgroup, 'col', null, { style: 'width: 30px' });
            addElem(colgroup, 'col');
            addElem(colgroup, 'col', null, { style: 'width: 120px' });
            addElem(colgroup, 'col', null, { style: 'width: 40px' });
            addElem(colgroup, 'col', null, { style: 'width: 70px' });
            addElem(colgroup, 'col', null, { style: 'width: 90px' });
            addElem(colgroup, 'col', null, { style: 'width: 80px' });
            addElem(colgroup, 'col', null, { style: 'width: 20px' }); // allow for scrollbar
            const tbodyElem = addElem(tableElem, 'tbody');

            const trElem1 = addElem(tbodyElem, 'tr');

            const thElem1_1 = addElem(trElem1, 'th', 'h', { colspan: '7' });
            addElem(trElem1, 'th', 'h');

            const frElem1 = addElem(thElem1_1, 'span', 'fr', {
                style: 'margin-top:2px;cursor:pointer'
            });
            frElem1.addEventListener('click', function() {
                modelMask.remove();
                closePopupMenu(SCORING_DIALOG_POPUP_MENU_ID);
            });

            addElem(frElem1, 'img', null, {
                src: 'images/close.gif'
            });
            thElem1_1.appendChild(document.createTextNode('批量评分'));

            const trElem2 = addElem(tbodyElem, 'tr', 'tr2 tac');
            addElem(trElem2, 'td');
            addElem(trElem2, 'td').textContent = '帖子标题';
            addElem(trElem2, 'td').textContent = '发布者';
            addElem(trElem2, 'td').textContent = '回复';
            addElem(trElem2, 'td').textContent = '资源大小';
            addElem(trElem2, 'td').textContent = '评分';
            addElem(trElem2, 'td');

            const pingableItems = [];
            items.forEach(function(item) {
                const itemTr = addElem(tbodyElem, 'tr', 'tr3 tac rinsp-batchsel-form-item');
                const checkCell = addElem(itemTr, 'td');
                const selectCheckbox = addElem(checkCell, 'input', null, { type: 'checkbox' });
                selectCheckbox.addEventListener('change', () => sync());
                const titleCell = addElem(itemTr, 'td', 'tal');
                let postUrl = `read.php?tid-${item.tid}.html`;
                addElem(titleCell, 'a', null, {
                    href: postUrl,
                    target: '_blank'
                }).textContent = item.title;

                const posterCell = addElem(itemTr, 'td', 'rinsp-batchsel-form-op-cell');
                if (item.opKnownName) {
                    posterCell.classList.add('rinsp-batchsel-form-op-known');
                    posterCell.textContent = item.opKnownName;
                } else {
                    posterCell.textContent = item.opName;
                }

                addElem(itemTr, 'td').textContent = '' + item.replyCount;

                addElem(itemTr, 'td').textContent = item.defaultScoreRating.resourceSizeTexts.join(' + ');

                const scoreCell = addElem(itemTr, 'td', 'rinsp-batchping-form-ping-cell');
                const scoreControls = addElem(scoreCell, 'div');
                const scoreInput = addElem(scoreControls, 'input', 'rinsp-batchping-form-sp-input', { type: 'text', size: '5' });
                addElem(scoreControls, 'span').textContent = ' SP';
                const unscoreCell = addElem(itemTr, 'td');
                const markUnscoredCheckbox = addElem(unscoreCell, 'input', null, { type: 'checkbox' });
                markUnscoredCheckbox.addEventListener('change', () => sync());
                addElem(unscoreCell, 'span').textContent = '不评分';

                const defaultUnscored = item.opKnownName && item.opKnownName.startsWith(NEVER_SCORE_USER_PREFIX);
                const defaultChecked = item.opKnownName && item.opKnownName.startsWith(TRUSTED_SCORE_USER_PREFIX);
                const unavailable = item.defaultScoreRating.baseTotalScore <= 0 || item.defaultScoreRating.ownBought || item.defaultScoreRating.ownTranslate;
                if (defaultUnscored) {
                    markUnscoredCheckbox.checked = true;
                }
                if (defaultChecked || defaultUnscored) {
                    selectCheckbox.checked = true;
                }
                if (unavailable) {
                    itemTr.classList.add('rinsp-batchsel-form-item-unavailable');
                } else {
                    scoreInput.value = '' + Math.min(99999, Math.max(1, Math.floor(item.defaultScoreRating.baseTotalScore)));
                }
                pingableItems.push({
                    item: item,
                    selected: () => selectCheckbox.checked,
                    score() {
                        if (markUnscoredCheckbox.checked) {
                            return -1;
                        }
                        const score = Math.floor(scoreInput.value * 1);
                        if (score > 0 && score <= 99999) {
                            return score;
                        }
                        return null;
                    },
                });
                
                sync();
                function sync() {
                    itemTr.classList.remove('rinsp-batchsel-form-item-checked');
                    itemTr.classList.remove('rinsp-batchsel-form-item-unchecked');
                    itemTr.classList.remove('rinsp-batchping-form-item-unscore');
                    if (selectCheckbox.checked) {
                        itemTr.classList.add('rinsp-batchsel-form-item-checked');
                        if (markUnscoredCheckbox.checked) {
                            itemTr.classList.add('rinsp-batchping-form-item-unscore');
                        }
                    } else {
                        itemTr.classList.add('rinsp-batchsel-form-item-unchecked');
                    }
                }
            });


            const footer = addElem(borElem, 'ul', null, {
                style: 'text-align:center;padding:4px 0;'
            });
            const submitButton = addElem(footer, 'input', 'btn', { type: 'button', value: '执行评分' });
            submitButton.addEventListener('click', () => {
                const pingItems = [];
                pingableItems.forEach(pItem => {
                    if (pItem.selected()) {
                        const score = pItem.score();
                        if (score != null) {
                            pingItems.push({
                                item: pItem.item,
                                score
                            });
                        }
                    }
                });
                if (pingItems.length === 0) {
                    showMessagePopup('没有项目', submitButton, 3000);
                    return;
                }
                if (confirm(`执行评分: ${pingItems.length} 个项目 ?`)) {
                    executeBatchPing(pingItems);
                }
            });

            footer.appendChild(document.createTextNode(' '));

            const closeButton = addElem(footer, 'input', 'btn', { type: 'button', value: '关闭' });
            closeButton.addEventListener('click', function() {
                modelMask.remove();
                closePopupMenu(SCORING_DIALOG_POPUP_MENU_ID);
            });

            function executeBatchPing(pingItems) {
                modelMask.remove();
                closePopupMenu(SCORING_DIALOG_POPUP_MENU_ID);
                runWithProgressPopup(action, '评分操作中...', null, 1500)
                    .catch((ex) => alert(String(ex)))
                    .finally(() => window.location.reload());

                async function action(progressDisplay) {
                    const itemList = addElem(progressDisplay, 'ul', 'rinsp-batchping-progress-list');
                    const displayItems = [];
                    for (let pingItem of pingItems) {
                        const li = addElem(itemList, 'li');
                        li.textContent = pingItem.item.title;
                        displayItems.push(li);
                    }
                    await sleep(1000);
                    for (let pingItem of pingItems) {
                        if (pingItem.score > 0) {
                            await executeScoreFunction(pingItem.item.fid, pingItem.item.tid, pingItem.score, 0, {}, true, '');
                        } else {
                            await executeScoreFunction(pingItem.item.fid, pingItem.item.tid, 1, 0, {}, false, '');
                        }
                        displayItems.shift().remove();
                    }
                    await sleep(500);
                    itemList.remove();
                    return '✔️已完成评分';
                }
            }
        });        
    }

    function addDeleteFormMetadata() {
        const delButton = document.querySelector('a#del[href^="mawhole.php?action-del-"]');
        if (delButton == null) {
            return null;
        }
        let gfPost = getPosts(document, 1)[0];
        if (gfPost == null || gfPost.floor !== 0) {
            return;
        }
        const actionHref = delButton.getAttribute('href');
        delButton.setAttribute('href', `${actionHref}#uids-${gfPost.postUid}-tids-${gfPost.tid}`);
    }

    function addBulkDeleteFormMetadata() {
        const form = document.querySelector('form[name="mawhole"][action="mawhole.php"]');
        if (form == null) {
            return;
        }
        const submitButton = form.querySelector('input[type="button"][name="hello"][onclick]');
        if (submitButton == null) {
            return;
        }
        submitButton.removeAttribute('onclick');
        submitButton.addEventListener('click', evt => {
            const formData = new FormData(form);
            const tids = formData.getAll('tidarray[]');
            if (formData.get('action') !== 'del' || tids.length === 0) {
                form.submit();
                return;
            }
            const uids = [];
            tids.forEach(tid => {
                const tr = document.getElementById(`td_${tid}`).closest('tr');
                const userLink = tr.querySelector('a.bl[href^="u.php?action-show-uid-"]');
                uids.push(Number.parseInt(userLink.getAttribute('href').substring(22)));
            });
            
            submitPostForm('mawhole.php#uids-' + Array.from(uids).join(':') + '-tids-' + Array.from(tids).join(':'), formData);
        });
    }

    function addBatchPingFunction(fid) {
        const adminAreaAnchor = document.querySelector('form[name="mawhole"] > .t + .t5.tac > div > .btn[name="chkall"]');
        if (adminAreaAnchor == null) {
            return;
        }
        const pingAllButton = newElem('input', 'btn', { type: 'button', name: 'chkall', value: '批量评分', style: 'float: left; margin-left: 1em; margin-right: -100%' });
        adminAreaAnchor.closest('div').appendChild(pingAllButton);
        pingAllButton.addEventListener('click', () => {
            const threadList = readThreadList();
            const unscoredThreads = threadList.filter(thread => !thread.row.classList.contains('rinsp-thread-filter-scored') && !thread.row.classList.contains('rinsp-thread-filter-miscored'));
            if (unscoredThreads.length === 0) {
                showMessagePopup('💡没有未评分帖子', pingAllButton, 3000);
                return;
            }
            const items = unscoredThreads.map(thread => {
                const defaultScoreRating = getDefaultRating(thread.title);
                const customNameEntry = userConfig.customUserHashIdMappings['#' + thread.op];
                return {
                    fid,
                    tid: thread.tid,
                    title: thread.title,
                    opName: thread.opName,
                    opKnownName: customNameEntry && customNameEntry[2],
                    replyCount: thread.replyCount,
                    defaultScoreRating
                };
            });
            showBatchPingWindow(items);
        });
    }

    function showBatchSelectionWindow(items, action) {
        const modelMask = addModalMask();
        const popupMenu = createPopupMenu(SCORING_DIALOG_POPUP_MENU_ID, null, false);
        popupMenu.renderContent(async function(borElem) {
            const eForm = addElem(borElem, 'div', 'rinsp-batchsel-form rinsp-batchping-form');
            const tableElem = addElem(eForm, 'table', null, {
                width: '850',
                cellspacing: '0',
                cellpadding: '0',
                style: 'table-layout:fixed'
            });

            const colgroup = addElem(tableElem, 'colgroup');
            addElem(colgroup, 'col', null, { style: 'width: 30px' });
            addElem(colgroup, 'col');
            addElem(colgroup, 'col', null, { style: 'width: 120px' });
            addElem(colgroup, 'col', null, { style: 'width: 120px' });
            addElem(colgroup, 'col', null, { style: 'width: 70px' });
            addElem(colgroup, 'col', null, { style: 'width: 160px' });
            addElem(colgroup, 'col', null, { style: 'width: 20px' }); // allow for scrollbar
            const tbodyElem = addElem(tableElem, 'tbody');

            const trElem1 = addElem(tbodyElem, 'tr');

            const thElem1_1 = addElem(trElem1, 'th', 'h', { colspan: '6' });
            addElem(trElem1, 'th', 'h');

            const frElem1 = addElem(thElem1_1, 'span', 'fr', {
                style: 'margin-top:2px;cursor:pointer'
            });
            frElem1.addEventListener('click', function() {
                modelMask.remove();
                closePopupMenu(SCORING_DIALOG_POPUP_MENU_ID);
            });

            addElem(frElem1, 'img', null, {
                src: 'images/close.gif'
            });
            thElem1_1.appendChild(document.createTextNode('批量选择'));

            const trElem2 = addElem(tbodyElem, 'tr', 'tr2 tac');
            const selCell = addElem(trElem2, 'td');
            addElem(trElem2, 'td').textContent = '帖子标题';
            addElem(trElem2, 'td').textContent = '论坛';
            addElem(trElem2, 'td').textContent = '发布者';
            addElem(trElem2, 'td').textContent = '回复';
            addElem(trElem2, 'td').textContent = '';
            addElem(trElem2, 'td');

            const selAll = addElem(selCell, 'a', null, { href: 'javascript:void(0)' });
            selAll.textContent = '☑️';
            const selectableItems = [];
            items.forEach(function(item) {
                const itemTr = addElem(tbodyElem, 'tr', 'tr3 tac rinsp-batchsel-form-item');
                const checkCell = addElem(itemTr, 'td');
                const selectCheckbox = addElem(checkCell, 'input', null, { type: 'checkbox' });
                selectCheckbox.addEventListener('change', () => sync());
                const titleCell = addElem(itemTr, 'td', 'tal');
                addElem(titleCell, 'a', null, {
                    href: `read.php?tid-${item.tid}.html`,
                    target: '_blank'
                }).textContent = item.title;

                const areaCell = addElem(itemTr, 'td', 'tal');
                addElem(areaCell, 'a', null, {
                    href: `thread.php?fid-${item.fid}.html`,
                    target: '_blank'
                }).textContent = item.areaName;

                const posterCell = addElem(itemTr, 'td', 'rinsp-batchsel-form-op-cell');
                const customNameEntry = userConfig.customUserHashIdMappings['#' + item.op];
                const opKnownName = customNameEntry && customNameEntry[2];

                if (opKnownName) {
                    posterCell.classList.add('rinsp-batchsel-form-op-known');
                    posterCell.textContent = opKnownName;
                } else {
                    posterCell.textContent = item.opName;
                }

                addElem(itemTr, 'td').textContent = '' + item.replyCount;
                const badRecordCell = addElem(itemTr, 'td');
                userPunishRecordStore.get(item.op)
                    .then(record => {
                        if (record) {
                            badRecordCell.appendChild(renderPunishTags(item.op, record));
                        }
                    });

                addElem(itemTr, 'td');

                selectableItems.push({
                    item: item,
                    selected: () => selectCheckbox.checked,
                    setSelected(b) {
                        selectCheckbox.checked = b;
                    }
                });
                
                sync();
                function sync() {
                    if (selectCheckbox.checked) {
                        itemTr.classList.add('rinsp-batchsel-form-item-checked');
                        itemTr.classList.remove('rinsp-batchsel-form-item-unchecked');
                    } else {
                        itemTr.classList.remove('rinsp-batchsel-form-item-checked');
                        itemTr.classList.add('rinsp-batchsel-form-item-unchecked');
                    }
                }
            });

            selAll.addEventListener('click', () => {
                const toSelect = selectableItems.find(item => !item.selected());
                selectableItems.forEach(item => item.setSelected(toSelect));
            });


            const footer = addElem(borElem, 'ul', null, {
                style: 'text-align:center;padding:4px 0;'
            });
            const submitButton = addElem(footer, 'input', 'btn', { type: 'button', value: '决定选择' });
            submitButton.addEventListener('click', () => {
                const selectedItems = [];
                selectableItems.forEach(pItem => {
                    if (pItem.selected()) {
                        selectedItems.push(pItem.item);
                    }
                });
                if (selectedItems.length === 0) {
                    showMessagePopup('没有项目', submitButton, 3000);
                    return;
                }
                if (confirm(`决定选择: ${selectedItems.length} 个项目 ?`)) {
                    executeAction(selectedItems);
                }
            });

            footer.appendChild(document.createTextNode(' '));

            const closeButton = addElem(footer, 'input', 'btn', { type: 'button', value: '关闭' });
            closeButton.addEventListener('click', function() {
                modelMask.remove();
                closePopupMenu(SCORING_DIALOG_POPUP_MENU_ID);
            });

            function executeAction(selectedItems) {
                modelMask.remove();
                closePopupMenu(SCORING_DIALOG_POPUP_MENU_ID);
                action(selectedItems);
            }
        });        
    }


    function checkSomePosts(decideChecked, skipScroll) {
        let first = null;
        Array.from(document.querySelectorAll('input[type="checkbox"][name="selid[]"]'))
            .forEach(el => {
                const value = decideChecked(el);
                if (first == null && value) {
                    first = el;
                }
                el.checked = value;
            });
        if (first && !skipScroll) {
            first.scrollIntoView();
        }
    }

    function addBatchPostSelectionControl() {
        const targetControlBar = document.querySelector('#main > .h2 > .fr.w');
        if (targetControlBar.querySelector('.rinsp-reply-batch-sel')) {
            return;
        }

        function isUnscored(el) {
            return el.closest('table').querySelector('div[id^="mark_"]') == null;
        }
          
        function isBanned(el) {
            const span = el.closest('table').querySelector('.tpc_content div[id^="read_"] > span[style="color:black;background-color:#ffff66"]');
            return span && span.textContent.trim() === '用户被禁言,该主题自动屏蔽!';
        }

        function checkNotDuplicates() {
            checkDuplicates(true);
        }

        function checkDuplicates(invert) {
            const checkValue = !invert;
            const seen = new Set();
            let first = null;
            const checkboxes = Array.from(document.querySelectorAll('input[type="checkbox"][name="selid[]"]'));
            checkboxes.forEach(el => {
                const a = el.closest('table').querySelector('.user-pic a[href^="u.php?action-show-uid-"]');
                const uid = Number.parseInt(a.getAttribute('href').substring(22));
                if (seen.has(uid)) {
                    el.checked = checkValue;
                    if (first == null) first = el;
                } else {
                    el.checked = !checkValue;
                    seen.add(uid);
                }
            });
            if (first) {
                first.scrollIntoView();
            }
        }

        const actions = [];
        actions.push({
            label: '全选',
            action: () => checkSomePosts(()=>true)
        });
        actions.push({
            label: '未评分',
            action: () => checkSomePosts(isUnscored)
        });
        actions.push({
            label: '重复',
            action: () => checkDuplicates()
        });
        actions.push({
            label: '非重复',
            action: () => checkNotDuplicates()
        });
        actions.push({
            label: '已禁言',
            action: () => checkSomePosts(isBanned)
        });
        actions.push({
            label: '清除',
            class: 'rinsp-alert-menu-button',
            action: () => checkSomePosts(()=>false)
        });
        targetControlBar.append(document.createTextNode(' | '));
        const batchSel = addElem(targetControlBar, 'a', 'fn rinsp-reply-batch-sel');
        batchSel.setAttribute('href', 'javascript:void(0)');
        batchSel.textContent = '批量选择';
        batchSel.addEventListener('click', () => {
            setupPopupMenu({
                title: '批量选择',
                popupMenuId: 'BATCH_SEL_MENU',
                width: 150,
                anchor: batchSel,
                verticallyInverted: false,
                items: actions
            });
        });
    }

    function addBatchDeleteFunction() {
        const adminAreaAnchor = document.querySelector('#main > .fr');
        if (adminAreaAnchor == null) {
            return;
        }
        const delAllButton = newElem('input', 'btn', { type: 'button', value: '批量删除', style: 'float: left; margin-left: 1em; margin-right: -100%' });
        adminAreaAnchor.closest('#main').appendChild(delAllButton);
        delAllButton.addEventListener('click', () => {
            const threadList = readSearchThreadList();
            const threads = threadList.map(thread => {
                return {
                    fid: thread.areaId,
                    areaName: thread.areaName,
                    tid: thread.tid,
                    title: thread.title,
                    op: thread.op,
                    opName: thread.opName,
                    replyCount: thread.replyCount,
                };
            });
            showBatchSelectionWindow(threads, async selection => {
                const fidGroups = new Map();
                selection.forEach(item => {
                    let group = fidGroups.get(item.fid);
                    if (group == null) {
                        group = [];
                        fidGroups.set(item.fid, group);
                    }
                    group.push(item);
                });

                for (let [fid, group] of fidGroups) {
                    openBulkDeletePage(fid, group);
                    await sleep(1000);
                }
                function openBulkDeletePage(fid, items) {
                    const formData = new FormData();
                    formData.set('action', 'del');
                    formData.set('fid', fid);
                    const tids = [];
                    const uids = [];
                    items.forEach(item => {
                        formData.append('tidarray[]', item.tid);
                        tids.push(item.tid);
                        uids.push(item.op);
                    });
                    submitPostForm('mawhole.php#uids-' + Array.from(uids).join(':') + '-tids-' + Array.from(tids).join(':'), formData, '_blank');
                }
            });
        });
    }

    async function handleOutZoneRequestPunish(post) {
        if (!confirm('执行求物处分 -50SP + 删?')) {
            return;
        }

        const threadTitle = (document.querySelector('#breadcrumbs .crumbs-item.current > strong')||{}).textContent;
        const reason = '禁止茶馆求物、出处、梯子机场、直求主题';
        const action = async () => {
            await executeScoreFunction(post.areaId, post.tid, -50, 0, {}, true, reason);
            await executeDeletePost(post.areaId, post.tid, reason);
            await recordPunishment(post.postUid, PUNISH_TYPE_DELETE_THREAD, '', '-50SP ' + reason, {
                fid: post.areaId,
                tid: post.tid,
                title: threadTitle
            });
            return '✔️处分操作已完成';
        };
        runWithProgressPopup(action, '处分操作中...', null, 1500)
            .catch((ex) => alert(String(ex)))
            .finally(() => document.location.reload());
    }

    async function handleMisclassificationPunish(post) {
        if (!confirm('执行分类错误 SP-50?')) {
            return;
        }
        await executeScoreThenReload(null, post.areaId, post.tid, -50, 0, {}, true, '分类错误');
    }

    async function handleAwardBestAnswer(post, sp, multiplier) {
        const totalAmount = sp * multiplier;
        let suffix = multiplier > 1 ? ` (${multiplier}倍)` : '';
        if (!confirm(`选为最佳答案 ${totalAmount} SP${suffix}?`)) {
            return;
        }
        await executeScoreThenReload(null, post.areaId, post.tid, totalAmount, 0, {}, true, '最佳答案', post.postId);
    }

    function addPostAdminControl(posts, page, userMap) {

        function buildMultiAccountMap() {
            const colors = [ '#f9c6c9', '#bcd4e6', '#faedcb', '#c5dedd', '#eddcd2', '#dbcdf0' ];
        
            function nextColor() {
                const color = colors.shift();
                colors.push(color);
                return color;
            }
        
            const ipToUsers = new Map();
            const userToIps = new Map();
            function addMapping(map, key, value) {
                const values = map.get(key);
                if (values) {
                    values.add(value);
                } else {
                    map.set(key, new Set([value]));
                }
            }
        
            posts.forEach(post => {
                const banButton = post.rootElem.querySelector('.tipad a[id^="banuser_"]');
                if (banButton) {
                    const button = post.rootElem.querySelector('.rinsp-user-admin-button');
                    if (button) {
                        post.rootElem.classList.remove('rinsp-post-multi-uid');
                        button.parentElement.removeAttribute('style');
                        button.remove();
                    }
                    const ip = (banButton.closest('.fr').textContent.match(/IP:(\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}) \|/)||[])[1]||null;
                    if (ip) {
                        addMapping(ipToUsers, ip, post.postUid);
                        addMapping(userToIps, post.postUid, ip);
                    }
                }
            });
        
            let nextGid = 1;
            const userGrouped = new Map();
            ipToUsers.forEach((uids, ip) => {
                let group = userGrouped.get(ip);
                if (!group) {
                    group = {
                        gid: nextGid++,
                        uids: new Set()
                    };
                    userGrouped.set(ip, group);
                }
                for (let uid of uids) {
                    group.uids.add(uid);
                    const ips = userToIps.get(uid);
                    for (let ip2 of ips) {
                        if (ip2 !== ip) {
                            let group2 = userGrouped.get(ip2);
                            if (group2 && group2.gid !== group.gid) {
                                for (let uid2 of group2.uids) {
                                    group.uids.add(uid2); 
                                }
                                userGrouped.set(ip2, group);
                            }
                        }
                    }
                }
            });
        
            const multiAccountMap = new Map();
            Array.from(userGrouped.values())
                .filter(group => group.uids.size > 1)
                .forEach(group => {
                    if (!group.color) {
                        group.color = nextColor();
                    }
                    for (let uid of group.uids) {
                        multiAccountMap.set(uid, group);
                    }
                });
            return multiAccountMap;
        }

        function openActionPage(form, action) {
            const formData = new FormData(form);
            const postIds = new Set(formData.getAll('selid[]').map(pid => pid * 1));
            const postUids = new Set();
            posts.filter(post => postIds.has(post.postId)).forEach(post => {
                postUids.add(post.postUid);
            });
            submitPostForm(action + '#uids-' + Array.from(postUids).join(':'), formData);
        }

        const multiAccountMap = buildMultiAccountMap();
        let bountyStatus = null;
        let threadOwnerId = null;
        let bestAnswerMultiplier = 0;
        
        if (posts[0].areaId === 8 || posts[0].areaId === 9) { // ACG交流, 茶馆
            document.querySelectorAll('.rinsp-punishrequest-switch').forEach(el => el.remove());
            if (posts[0].floor === 0) {
                const showping = document.querySelector('li > #showping_tpc');
                let punishrequest = document.querySelector('li > #punishrequest_tpc');
                if (showping) {
                    const tpcPing = document.querySelector('#mark_tpc');
                    if (!punishrequest) {
                        if (tpcPing && tpcPing.textContent.match(/.*评分记录.*/) != null) {
                            // skip: already received penalty
                        } else {
                            const li = newElem('li', 'rinsp-punishrequest-switch');
                            showping.parentNode.parentNode.insertBefore(li, showping.parentNode.nextElementSibling);
                            punishrequest = addElem(li, 'a', null, { href: 'javascript:' });
                            punishrequest.textContent = '求物扣分';
                            punishrequest.addEventListener('click', () => handleOutZoneRequestPunish(posts[0]));
                        }
                    }
                }
            }
        } else if (posts[0].areaId === QUESTION_AND_REQUEST_AREA_ID) { // 求物区
            bountyStatus = getBountyStatus(document);
            document.querySelectorAll('.rinsp-punishclassify-switch').forEach(el => el.remove());
            if (posts[0].floor === 0) {
                threadOwnerId = posts[0].postUid;
                const showping = document.querySelector('li > #showping_tpc');
                let punishclassify = document.querySelector('li > #punishclassify_tpc');
                if (showping) {
                    const tpcPing = document.querySelector('#mark_tpc');
                    if (!punishclassify) {
                        if (tpcPing && tpcPing.textContent.match(/.*评分记录.*SP币:-50 .*分类.*/) != null) {
                            // skip: already received penalty
                        } else {
                            const li = newElem('li', 'rinsp-punishclassify-switch');
                            showping.parentNode.parentNode.insertBefore(li, showping.parentNode.nextElementSibling);
                            punishclassify = addElem(li, 'a', null, { href: 'javascript:' });
                            punishclassify.textContent = '分类错误';
                            punishclassify.addEventListener('click', () => handleMisclassificationPunish(posts[0]));
                        }
                    }
                    if (bountyStatus.ended === 2 && bountyStatus.winner == null) {
                        const pingSp = tpcPing ? (tpcPing.textContent.match(/.*评分记录.*SP币:\+?(-?\d+)/)||[null,0])[1] * 1 : 0;
                        if (pingSp === 0) {
                            bestAnswerMultiplier = 2;
                        } else if (bountyStatus.sp === pingSp) {
                            bestAnswerMultiplier = 1;
                        } else if (bountyStatus.sp === -pingSp) {
                            bestAnswerMultiplier = 3;
                        }
                    }
                    if (bestAnswerMultiplier > 0) {
                        posts.forEach(post => {
                            if (bestAnswerMultiplier === 0)
                                return;
                            const mark = post.rootElem.querySelector('div[id^="mark_"] li');
                            if (mark && mark.textContent.indexOf('最佳答案') !== -1) {
                                bestAnswerMultiplier = 0;
                            }
                        });
                    }
                }
            }
            const endrewardButton = document.querySelector('a[href^="job.php?action-endreward-tid-"]');
            if (endrewardButton) {
                const endrewardActionHref = endrewardButton.getAttribute('href');
                if (endrewardActionHref.indexOf('#') === -1) {
                    const threadOwnerInfo = userMap.get(threadOwnerId)||{ sp: 0 };
                    endrewardButton.setAttribute('href', endrewardActionHref + '#bounty-' + bountyStatus.sp + '-pts-' + bountyStatus.pts + '-opsp-' + threadOwnerInfo.sp);
                }
            }

        }
        posts.forEach(post => {
            const selCheckbox = post.rootElem.querySelector('.fr input[type="checkbox"][name="selid[]"]');
            if (selCheckbox) {
                const form = selCheckbox.closest('form[name="delatc"]');
                if (form) {
                    const deleteButton = selCheckbox.parentElement.querySelector('input[value="删除选定"][onclick]');
                    const pingButton = selCheckbox.parentElement.querySelector('input[value="评分选定"][onclick]');
                    if (deleteButton) {
                        deleteButton.removeAttribute('onclick');
                        deleteButton.addEventListener('click', () => {
                            openActionPage(form, 'masingle.php?action=delatc');
                        });
                    }
                    if (pingButton) {
                        pingButton.removeAttribute('onclick');
                        pingButton.addEventListener('click', () => {
                            openActionPage(form, `operate.php?action=showping&page=${page}`);
                        });
                    }
                    if (selCheckbox.parentElement.querySelector('.rinsp-selallid-button') == null) {
                        selCheckbox.parentElement.appendChild(document.createTextNode(' '));
                        const selAllButton = addElem(selCheckbox.parentElement, 'input', 'btn2 rinsp-selallid-button', { type: 'button', value: '全选' });
                        
                        selAllButton.addEventListener('click', () => {
                            const postIds = new Set(posts.filter(p => p.postUid === post.postUid).map(p => p.postId));
                            checkSomePosts(el => postIds.has(el.value * 1), true);
                        });
                    }
                }
            }

            const showpingButton = post.rootElem.querySelector('.tipad ul > li > a[id^="showping_"]');
            if (showpingButton) {
                const showpingActionHref = showpingButton.getAttribute('href');
                if (showpingActionHref.indexOf('#') === -1) {
                    let extraParams = [];
                    post.rootElem.querySelectorAll('.r_one .tpc_content .tips').forEach(tips => {
                        const match = tips.textContent.match(/最佳答案奖励:\s\(\+(\d+)\)\sSP币/);
                        if (match) {
                            extraParams.push('-answer-' + match[1] * 1);
                        }
                    });
                    let sellPrice = 0;
                    let boughtCount = 0;
                    post.contentElem.querySelectorAll('h6.quote.jumbotron > .s3.f12.fn').forEach(soldText => {
                        const match = soldText.textContent.match(/此帖售价 (\d+) SP币,已有 (\d+) 人购买/);
                        if (match) {
                            sellPrice = Math.max(sellPrice, match[1]*1);
                            boughtCount = match[2]*1;
                        }
                    });
                    if (boughtCount > 0 && bountyStatus != null) {
                        boughtCount--;
                    }
                    const total = sellPrice * boughtCount;
                    if (sellPrice > 0) {
                        if (bountyStatus != null && post.postUid === threadOwnerId) {
                            post.rootElem.classList.add('rinsp-post-illegal-sell');
                            extraParams.push('-badsell-opsell');
                        }
                        if (total > 0) {
                            extraParams.push(`-sold-${total}`);
                        }
                    }
                    if (post.postUid !== threadOwnerId) {
                        if (bountyStatus && bountyStatus.winner == null) {
                            extraParams.push('-bounty-' + bountyStatus.sp);
                        }
                        if (bestAnswerMultiplier > 0) {
                            const li = newElem('li', 'rinsp-markcorrect-switch');
                            showpingButton.parentNode.parentNode.insertBefore(li, showpingButton.parentNode.nextElementSibling);
                            const markcorrect = addElem(li, 'a', null, { href: 'javascript:' });
                            markcorrect.textContent = '最佳答案';
                            markcorrect.addEventListener('click', () => handleAwardBestAnswer(post, bountyStatus.sp, bestAnswerMultiplier));

                        }
                    }
                    showpingButton.setAttribute('href', showpingActionHref + '#uids-' + post.postUid + '-fid-' + post.areaId + extraParams.join(''));
                }
            }

            const shieldButton = post.rootElem.querySelector('.tipad a[id^="shield_"]');
            if (shieldButton) {
                const shieldActionHref = shieldButton.getAttribute('href');
                if (shieldActionHref.indexOf('#') === -1) {
                    shieldButton.setAttribute('href', shieldActionHref + '#uid-' + post.postUid);
                }
            }

            let adminBar;
            const banuserButton = post.rootElem.querySelector('.tipad a[id^="banuser_"]');
            if (banuserButton) {
                const banuserActionHref = banuserButton.getAttribute('href');
                if (banuserActionHref.indexOf('#') === -1) {
                    banuserButton.setAttribute('href', banuserActionHref + '#uid-' + post.postUid + '-fid-' + post.areaId);
                }
                adminBar = banuserButton.parentElement;
            }

            if (adminBar) {
                const group = multiAccountMap.get(post.postUid);
                const button = newElem('a', 'rinsp-user-admin-button', { href: 'javascript:'});
                button.textContent = '🔘';
                let addExtraMenuItems = () => {};
                if (group != null) {
                    adminBar.prepend(button);
                    adminBar.setAttribute('style', `outline: 2px solid ${group.color}; outline-offset: 2px`);
                    post.rootElem.classList.add('rinsp-post-multi-uid');
                    addExtraMenuItems = (items => {
                        if (document.body.classList.contains('rinsp-filter-multiip-focus-mode')) {
                            items.push({
                                label: '取消IP筛选',
                                action: () => {
                                    document.body.classList.remove('rinsp-filter-multiip-focus-mode');
                                    posts.forEach(p => {
                                        p.rootElem.classList.remove('rinsp-post-multi-uid-selected');
                                    });
                                }
                            });
                        } else {
                            items.push({
                                label: `只看相同IP用户 (${group.uids.size})`,
                                action: () => {
                                    document.body.classList.add('rinsp-filter-multiip-focus-mode');
                                    posts.forEach(p => {
                                        if (group.uids.has(p.postUid)) {
                                            p.rootElem.classList.add('rinsp-post-multi-uid-selected');
                                        }
                                    });
                                }
                            });
                        }
                    });
                }
                button.addEventListener('click', async () => {
                    const items = [];
                    addExtraMenuItems(items);
                    if (items.length === 0) {
                        return;
                    }
                    setupPopupMenu({
                        title: '请选择功能',
                        popupMenuId: 'MULTI_ACCOUNT_MENU',
                        width: 150,
                        anchor: button,
                        verticallyInverted: true,
                        items
                    });
                });
            }
        });
    }

    function renderAdminHistoryPage(myUserId, userConfig) {
        const services = {
            async getLogEntries() {
                const punishRecords = await userPunishRecordStore.list();
                const entries = [];
                punishRecords.forEach(record => {
                    record.logs.forEach(entry => {
                        entries.push(Object.assign({ uid: record.uid }, entry));
                    });
                });
                return entries;
            }
        };
        renderCommonPunishHistoryPage(myUserId, 0, services);
    }

    function renderPunishHistoryPage(userId, userConfig) {
        const services = {
            async getLogEntries() {
                let punishRecord = await userPunishRecordStore.get(userId);
                if (punishRecord == null) {
                    punishRecord = {
                        uid: userId,
                        logs: []
                    };
                }
                return punishRecord.logs.map(entry => {
                    return Object.assign({
                        uid: userId
                    }, entry);
                });
            }
        };
        renderCommonPunishHistoryPage(userId, userId, services);
    }

    function renderCommonPunishHistoryPage(pageScopeUid, subjectUid, services) {
        const mainPane = document.querySelector('#u-content > #u-contentmain');
        const sidePane = document.querySelector('#u-content > #u-contentside');
        sidePane.innerHTML = '';
        mainPane.innerHTML = '<div style="padding:16px 30px"><img src="images/loading.gif" align="absbottom">加载中...</div>';

        async function init() {
            mainPane.innerHTML = '';

            let entries = await services.getLogEntries();
            const sortByTime = comparator('time', true);
            entries.sort(sortByTime);
            
            const punishTypeToId = new Map();
            PUNISH_TYPES.forEach((type, i) => {
                punishTypeToId.set(type, i + 1);
            });

            const now = Date.now();
            const typeMap = new Map();
            entries = entries.map(entry => {
                let punishTypeId = punishTypeToId.get(entry.type);
                if (punishTypeId == null) {
                    punishTypeId = typeMap.size + 1;
                    punishTypeToId.set(punishTypeId, entry.type);
                }
                typeMap.set(punishTypeId, entry.type);
                return Object.assign({
                    typeId: punishTypeId
                }, entry);
            });
            const sectionFilter = addSectionFilter(typeMap, pageScopeUid, () => update(), {});
    
            const table = addElem(mainPane, 'table', 'u-table rinsp-punish-history-table');
            const thead = addElem(table, 'thead');
            const tr1 = addElem(thead, 'tr');
            const filterCell = addElem(tr1, 'td', null, { colspan: '3' });
            const filterBox1 = addElem(filterCell, 'div', 'rinsp-quick-filter-box');
            addElem(filterBox1, 'div').textContent = '🔍︎搜索';
            const filterInput = addElem(filterBox1, 'input', null);
            addElem(filterBox1, 'div', null, { style: 'flex: 0 1 4em' });
    
            if (subjectUid) {
                const addRecordButton = addElem(filterBox1, 'a', null, { href: 'javacript:'});
                addRecordButton.textContent = '➕添加记录';
                addRecordButton.addEventListener('click', async () => {
                    let comment = prompt('请添加备注');
                    if (comment) {
                        const newLogEntry = {
                            time: Date.now(),
                            type: PUNISH_TYPE_CUSTOM,
                            summary: comment,
                            reason: '',
                            data: {}
                        };
                        let updateRecord = await userPunishRecordStore.get(subjectUid);
                        if (updateRecord == null) {
                            updateRecord = {
                                uid: subjectUid,
                                logs: []
                            };
                        }
                        updateRecord.logs.push(newLogEntry);
                        await userPunishRecordStore.put(subjectUid, updateRecord);
                        entries.unshift(newLogEntry);
                        update();
                    }
                });
            }
            addElem(filterBox1, 'div', null, { style: 'flex: 0 1 1em' });
            const controlBlock = addElem(filterBox1, 'div');
    
            controlBlock.appendChild(document.createTextNode('显示数: '));
            const limitSelect = addElem(controlBlock, 'select');
            addElem(limitSelect, 'option', null, { value: '100' }).textContent = '100';
            addElem(limitSelect, 'option', null, { value: '200' }).textContent = '200';
            addElem(limitSelect, 'option', null, { value: '500' }).textContent = '500';
            addElem(limitSelect, 'option', null, { value: '1000' }).textContent = '1000';
            limitSelect.value = '100';
            limitSelect.addEventListener('change', () => update());
        
            const tbody = addElem(table, 'tbody');
            const tfoot = addElem(table, 'tfoot');
            const statusRow = addElem(tfoot, 'tr');
            const statusCell = addElem(statusRow, 'td', 'grey', { colspan: '2' });
    
            let lastFilterString = '';
            function update(filterChangeOnly) {
                const terms = filterInput.value.toLowerCase().trim().split(/\s+/g);
                const thisFilterString = terms.join(' ');
                if (filterChangeOnly && lastFilterString === thisFilterString)
                    return;
                lastFilterString = thisFilterString;
                statusCell.textContent = '';
                tbody.innerHTML = '';

                if (entries.length === 0) {
                    statusCell.textContent = '💡 没有记录';
                    return;
                }

                const countMap = new Map();
                entries.forEach(entry => {
                    countMap.set(entry.typeId, (countMap.get(entry.typeId)||0) + 1);
                });
                sectionFilter.updateCounts(countMap);
    
                const limit = limitSelect.value * 1;
                const punishTypeId = sectionFilter.getCurrentFid();
                let selectedEntries;
                // apply type filter, also clone the list for sorting
                if (punishTypeId > 0) {
                    selectedEntries = entries.filter(entry => entry.typeId === punishTypeId);
                } else {
                    selectedEntries = entries.slice();
                }
        
                // apply text filter
                if (terms.length === 1 && terms[0] === '') {
                    table.classList.remove('rinsp-table-filtered');
                    if (selectedEntries.length > limit) {
                        statusCell.textContent = `💡只显示前${limit}条记录 / 共${selectedEntries.length}条`;
                    }
                } else {
                    table.classList.add('rinsp-table-filtered');
                    selectedEntries = selectedEntries.filter(entry => match(entry, terms));

                    if (selectedEntries.length === 0) {
                        statusCell.textContent = '💡没有搜索结果';
                    } else if (selectedEntries.length > limit) {
                        statusCell.textContent = `💡只显示前${limit}条搜索结果 / 共${selectedEntries.length}条`;
                    }
                }

                if (selectedEntries.length > limit) {
                    statusCell.textContent = `💡只显示前${limit}条记录 / 共${selectedEntries.length}条`;
                }
    
                selectedEntries.slice(0, limit).forEach(entry => {
                    const row = addElem(tbody, 'tr');
                    switch (entry.type) {
                        case PUNISH_TYPE_HP:
                            row.classList.add('rinsp-profile-punish-tag-blood');
                            break;
                        case PUNISH_TYPE_BAN:
                            row.classList.add('rinsp-profile-punish-tag-ban');
                            break;
                        case PUNISH_TYPE_DELETE_THREAD:
                        case PUNISH_TYPE_DELETE_REPLY:
                        case PUNISH_TYPE_SHIELD:
                            row.classList.add('rinsp-profile-punish-tag-del');
                            break;
                        case PUNISH_TYPE_UNSHIELD:
                            row.classList.add('rinsp-profile-punish-tag-none');
                            break;
                        default:
                            row.classList.add('rinsp-profile-punish-tag-misc');
                    }
                    setAreaScoped(row, entry.typeId);

                    const th = addElem(row, 'th');
                    addElem(th, 'div').textContent = entry.type;
                    addElem(th, 'span', 'gray f9').textContent = ' [ ' + getAgeString((now - entry.time) / 60000) + ' ]';
                    
                    const descCell = addElem(row, 'td', 'rinsp-punish-history-summary-cell');
                    const details = addElem(descCell, 'details');
                    const summary = addElem(details, 'summary');
                    if (entry.summary) {
                        addElem(summary, 'span').textContent = entry.summary;
                        addElem(summary, 'div').textContent = entry.reason;
                    } else if (entry.reason) {
                        addElem(summary, 'span').textContent = entry.reason;
                    } else {
                        addElem(summary, 'span', 'gray').textContent = ' - ';
                    }
                    const fullContent = addElem(details, 'div', 'rinsp-punish-history-data');
                    const dataKeys = Object.keys(entry.data||{});
                    if (dataKeys.length === 0) {
                        addElem(fullContent, 'span', 'gray f9').textContent = '没有其他资讯';
                    } else {
                        dataKeys.forEach(key => {
                            const value = entry.data[key];
                            if (value != null) {
                                const pair = addElem(fullContent, 'div');
                                addElem(pair, 'label').textContent = key;
                                pair.appendChild(document.createTextNode(' '));
                                addElem(pair, 'span').textContent = value;
                            }
                        });
                    }

                    const delCell = addElem(row, 'td', 'rinsp-punish-history-del-cell');
                    const delButton = addElem(delCell, 'a');
                    delButton.textContent = '🗑️';
                    delButton.addEventListener('click', async () => {
                        if (confirm('删除这条记录?')) {
                            let updateRecord = await userPunishRecordStore.get(entry.uid);
                            if (updateRecord != null) {
                                updateRecord.logs = updateRecord.logs.filter(log => log.time != entry.time);
                                await userPunishRecordStore.put(entry.uid, updateRecord);
                            }
                            entries = entries.filter(r => r !== entry);
                            update();
                        }
                    });
                    if (!subjectUid) {
                        addElem(delCell, 'div');
                        const userLink = addElem(delCell, 'a', 'gray f9', {
                            href: `u.php?action-trade-uid-${entry.uid}.html`,
                            target: '_blank'
                        });
                        userLink.textContent = `#${entry.uid}`;
                    }
                });
                
            }

            let enqueueTimer = null;
            filterInput.addEventListener('keyup', () => {
                if (enqueueTimer) clearTimeout(enqueueTimer);
                enqueueTimer = setTimeout(() => update(true), 200);
                
            });
            filterInput.addEventListener('change', () => {
                if (enqueueTimer) clearTimeout(enqueueTimer);
                update(true);
            });
    
            update();

        }
        
        function match(entry, searchTerms) {
            foreach_term:
            for (let searchTerm of searchTerms) {
                if ((entry.summary||'').toLowerCase().indexOf(searchTerm) !== -1)
                    continue;
                if ((entry.reason||'').toLowerCase().indexOf(searchTerm) !== -1)
                    continue;
                if (entry.data) {
                    for (let value of Object.values(entry.data)) {
                        if (typeof value === 'string') {
                            if (value.toLowerCase().indexOf(searchTerm) !== -1)
                                continue foreach_term;
                        }
                    }
                }
                return false;
            }
            return true;
        }

        init();
    }

    function createPunishRecordAccess() {
        const punishRecordCache = new Map();
        return {
            async getPunishRecord(uid) {
                let record = punishRecordCache.get(uid);
                if (record == null) {
                    record = userPunishRecordStore.get(uid);
                    punishRecordCache.set(uid, record);
                }
                return await record;
            }
        };
    }

    async function findUserName(userId) {
        const doc = await fetchGetPage(`${document.location.origin}/sendemail.php?uid-${userId}.html`);
        const sendToField = doc.querySelector('input[name="sendtoname"]');
        if (sendToField) {
            return sendToField.value;
        }
        const userNameBold = doc.querySelector('#main .t .f_one center > b');
        if (userNameBold && userNameBold.parentElement.textContent.indexOf('不接受邮件') !== -1) {
            return userNameBold.textContent;
        }
        throw new Error('错误');
    }
    
    return {
        enableQuickPingFunction,
        autoFillShowPingPage,
        enhancePingActionPage,
        enhanceEndRewardActionPage,
        enhanceMawholeActionPage,
        enhanceDeleteReplyActionPage,
        enhanceReasonSelector,
        enhanceShieldActionPage,
        enhanceBanUserActionPage,
        addBatchPingFunction,
        addBatchDeleteFunction,
        addBatchPostSelectionControl,
        addDeleteFormMetadata,
        addBulkDeleteFormMetadata,
        addPostAdminControl,
        renderAdminHistoryPage,
        renderPunishHistoryPage,
        createPunishRecordAccess,
        findUserName
    };

}

function createLocalStorageProperty(name, defaultValue) {
    return {
        get() {
            let value = localStorage.getItem(name);
            if (value != null || defaultValue == null) {
                return value;
            }
            return defaultValue;
        },
        set(value) {
            if (value == null) {
                localStorage.removeItem(name);
            } else {
                localStorage.setItem(name, value);
            }
        }
    };
}

function applyDarkTheme() {
    document.documentElement.classList.forEach(cls => {
        if (cls.startsWith('rinsp-dark-theme-')) {
            document.documentElement.classList.remove(cls);
        }
    });
    if (darkModeEnabledProperty.get()) {
        document.documentElement.classList.add('rinsp-dark-mode-set');
        document.documentElement.classList.add(`rinsp-dark-theme-${darkModeThemeProperty.get()}`);
    } else {
        document.documentElement.classList.remove('rinsp-dark-mode-set');
    }
    if (isDarkMode()) {
        document.documentElement.classList.add('rinsp-dark-mode');
        return true;
    } else {
        document.documentElement.classList.remove('rinsp-dark-mode');
        return false;
    }
}

function isDarkMode() {
    if (darkModeEnabledProperty.get()) {
        if (darkModeFollowsSystemProperty.get()) {
            return window.matchMedia && window.matchMedia('(prefers-color-scheme: dark)').matches;
        } else {
            return true;
        }
    }
    return false;
}

const DOMAIN_REDIRECT_ENABLED_KEY = 'domain-redirect-enabled';
async function setDomainRedirectEnabled(active) {
    if (active) {
        await GM.setValue(DOMAIN_REDIRECT_ENABLED_KEY, document.location.origin);
    } else {
        await GM.deleteValue(DOMAIN_REDIRECT_ENABLED_KEY);
    }
}

async function getDomainRedirectTarget() {
    let value = await GM.getValue(DOMAIN_REDIRECT_ENABLED_KEY);
    if (value && value.match(/\*?https?:\/\/.*/)) {
        // fix corrupt "*undefined" value
        return value;
    } else {
        return null;
    }
}

function requiresDomainRedirect() {
    // exclude redirection for the following pages
    switch (document.location.pathname) {
        case '/login.php':
        case '/register.php':
        case '/admin.php':
            return false;
    }
    return document.querySelector('#user-login > a[href="register.php"]') != null;
}

async function attemptDomainRedirect() {
    let domainRedirectTarget = await getDomainRedirectTarget();
    if (domainRedirectTarget && domainRedirectTarget.startsWith('*')) {
        domainRedirectTarget = domainRedirectTarget.substring(1);
        if (document.location.origin === domainRedirectTarget) {
            return false;
        } else {
            GM.setValue(DOMAIN_REDIRECT_ENABLED_KEY, domainRedirectTarget);
        }
    }
    if (domainRedirectTarget) {
        await GM.setValue(DOMAIN_REDIRECT_ENABLED_KEY, '*' + domainRedirectTarget); // deactivate the target until is it reachable
        window.location.replace(`${domainRedirectTarget}${document.location.pathname}${document.location.search}${document.location.hash}`);
    }
    return true;
}

function setSafeMode(active) {
    if (active) {
        localStorage.setItem('rinsp-safe-mode', '1');
        document.documentElement.classList.add('rinsp-safe-mode');
    } else {
        localStorage.removeItem('rinsp-safe-mode');
        document.documentElement.classList.remove('rinsp-safe-mode');
    }
}

function isSafeMode() {
    return localStorage.getItem('rinsp-safe-mode') === '1';
}

function setSafeModeShowMyAvater(active) {
    if (active) {
        localStorage.setItem('rinsp-safe-mode-allow-myself', '1');
        document.documentElement.classList.add('rinsp-safe-mode-allow-myself');
    } else {
        localStorage.removeItem('rinsp-safe-mode-allow-myself');
        document.documentElement.classList.remove('rinsp-safe-mode-allow-myself');
    }
}

function isSafeModeShowMyAvater() {
    return localStorage.getItem('rinsp-safe-mode-allow-myself') === '1';
}

function setFastLoadMode(active) {
    if (active) {
        localStorage.setItem('rinsp-fastload-mode', '1');
    } else {
        localStorage.setItem('rinsp-fastload-mode', '0');
    }
}

function isFastLoadMode() {
    return localStorage.getItem('rinsp-fastload-mode') !== '0';
}

function setFastLoadLazyImageMode(active) {
    if (active) {
        localStorage.setItem('rinsp-fastload-mode-lazyimg', '1');
    } else {
        localStorage.removeItem('rinsp-fastload-mode-lazyimg');
    }
}

function isFastLoadLazyImageMode() {
    return localStorage.getItem('rinsp-fastload-mode-lazyimg') === '1';
}

//============= utils =============//

let verifyhashCache = null;
function verifyhash() {
    if (unsafeWindow.verifyhash) {
        return unsafeWindow.verifyhash;
    }
    if (verifyhashCache) {
        return verifyhashCache;
    }
    const hiddenField = document.querySelector('form[name="FORM"][action="post.php?"] input[type="hidden"][name="verify"][value]');
    if (hiddenField) {
        verifyhashCache = hiddenField.value;
        return hiddenField.value;
    }
    for (let scriptElement of document.querySelectorAll('head > script')) {
        const match = scriptElement.textContent.match(/;var verifyhash = '([A-Za-z0-9]{8})';/);
        if (match) {
            verifyhashCache = match[1];
            return match[1];
        }
    }
    alert('无法取得操作验证码 (运作环境错误,请用篡改猴及谷歌火狐等主流浏览器)');
}

// accounting for any deferred image loading, returning the original image src
function getImgSrc(img) {
    return img.getAttribute('data-rinsp-defer-src') || img.getAttribute('src');
}

function getByteLength(text) {
    if (textEncoder == null) return 0;
    return textEncoder.encode(text).length;
}

function truncateByByteLength(text, limit, ellipsis) {
    let diff = getByteLength(text) - limit;
    if (diff > 0) {
        if (ellipsis) {
            diff += getByteLength(ellipsis);
        }
        const chars = Array.from(text);
        while (diff > 0) {
            diff -= getByteLength(chars.pop());
        }
        if (ellipsis) {
            chars.push(ellipsis);
        }
        return chars.join('');
    } else {
        return text;
    }
}

function addGlobalStyle(parent, css) {
    const style = document.createElement('style');
    style.type = 'text/css';
    style.innerHTML = css;
    parent.appendChild(style);
}

function parseSpAmount(text) {
    const match = text.match(/([0-9一二两三四五六七八九十]+ ?[kKwW万千]?) *的? *(sp|SP)/);
    if (match) {
        match[1] = match[1].replace(/十([^一二三四五六七八九])/g, '10$1').replace(/十/g, '1').replace(/一/g, '1').replace(/[二两]/g, '2').replace(/三/g, '3').replace(/四/g, '4').replace(/五/g, '5').replace(/六/g, '6').replace(/七/g, '7').replace(/八/g, '8').replace(/九/g, '9');
        if (match[1].endsWith('w') || match[1].endsWith('W') || match[1].endsWith('万')) {
            return parseInt(match[1]) * 10000;
        } else if (match[1].endsWith('千') || match[1].endsWith('k') || match[1].endsWith('K')) {
            return parseInt(match[1]) * 1000;
        } else {
            const amt = match[1] * 1;
            return amt === 99999 ? 0 : amt;
        }
    } else {
        return 0;
    }
}

function getBountyStatus(doc) {
    const tpc = doc.querySelector('#td_tpc');
    if (tpc == null) {
        return null;
    }
    const bountyTimeElem = tpc.closest('table').querySelector('.tpc_content .tips .s3');
    if (bountyTimeElem == null) {
        return null;
    }
    let bountyAmount = 0;
    let helppointAmount = 0;
    let winnerHashId = null;
    tpc.closest('table').querySelectorAll('.tpc_content .tips .tac').forEach(elem => {
        const msg = elem.textContent.trim();
        const bountyMatch = msg.match(/最佳答案: (\d+)\s+SP币/);
        const helpPtsMatch = msg.match(/热心助人剩余点数: (\d+)\s+SP币/);
        const winnerMatch = msg.match(/最佳答案获得者: ([a-z0-9]{8})/);
        if (bountyMatch) {
            bountyAmount = bountyMatch[1] * 1;
        }
        if (helpPtsMatch) {
            helppointAmount = helpPtsMatch[1] * 1;
        }
        if (winnerMatch) {
            winnerHashId = winnerMatch[1];
        }
    });
    const statusMessage = bountyTimeElem.textContent.trim();
    
    let bountyUntil;
    if (statusMessage === '此帖悬赏结束') {
        bountyUntil = -1;
    } else if (statusMessage === '此帖悬赏中(剩余时间:已结束)...') {
        bountyUntil = Date.now();
    } else {
        const bountyHourLeft = (statusMessage.match(/此帖悬赏中\(剩余时间:(\d+)小时\)/)||[])[1] * 1;
        bountyUntil = Date.now() + (bountyHourLeft||0) * 3600000;
    }
    return {
        sp: bountyAmount,
        pts: helppointAmount,
        winner: winnerHashId,
        bountyUntil,
        ended: bountyUntil === -1 ? 2 : Date.now() > bountyUntil ? 1 : 0
    };
}

function findErrorMessage(doc) {
    let err = doc.querySelector('#main .t .f_one center');
    return err ? err.textContent.trim()||'不明错误' : null;
}

function getAgeString(ageMins) {
    if (ageMins >= 1440) {
        return Math.floor(ageMins / 1440) + '天前';
    } else if (ageMins >= 60) {
        return Math.floor(ageMins / 60) + '小时前';
    } else if (ageMins > 1) {
        return Math.floor(ageMins) + '分钟前';
    } else {
        return '刚刚';
    }
}

function addElem(parent, tag, styleClass, attrs) {
    const elem = newElem(tag, styleClass, attrs);
    parent.appendChild(elem);
    return elem;
}

function newElem(tag, styleClass, attrs) {
    const elem = document.createElement(tag);
    if (styleClass) {
        styleClass.split(' ').forEach(function(cls) {
            elem.classList.add(cls);
        });
    }
    if (attrs) {
        for (let name of Object.keys(attrs)) {
            elem.setAttribute(name, attrs[name]);
        }
    }
    return elem;
}

async function runWithLock(key, duration, job) {
    const prop = `rinsp-lock-${key}`;
    async function acquire() {
        while (true) {
            const randToken = (Math.random() * 10000000000).toFixed(0);
            const lockUntil = Date.now() + duration;
            localStorage.setItem(prop, `${randToken}-${lockUntil}`);
            await sleep(0);
            const acquired = (localStorage.getItem(prop)||'').split('-', 2);
            if (acquired[0] === randToken) {
                return true;
            }
            let waitTime = lockUntil - Date.now();
            if (waitTime > duration * 2) { // invalid lock, retry acquire
                localStorage.removeItem(prop);
            } else {
                await sleep(waitTime);
            }
        }
    }
    await acquire();
    return await Promise.resolve(job())
    .finally(() => {
            localStorage.removeItem(prop);
        });
}

// polyfill replacement
function Object_hasOwn(obj, prop) {
    return Object.prototype.hasOwnProperty.call(obj, prop);
}

function sleep(timeout) {
    return new Promise(function(resolve) {
        setTimeout(() => resolve(true), timeout);
    });
}

/*
    cyrb53 (c) 2018 bryc (github.com/bryc)
    License: Public domain. Attribution appreciated.
    A fast and simple 53-bit string hash function with decent collision resistance.
    Largely inspired by MurmurHash2/3, but with a focus on speed/simplicity.
*/
function cyrb53(str, seed) {
    let h1 = 0xdeadbeef ^ seed, h2 = 0x41c6ce57 ^ seed;
    for (let i = 0, ch; i < str.length; i++) {
        ch = str.charCodeAt(i);
        h1 = Math.imul(h1 ^ ch, 2654435761);
        h2 = Math.imul(h2 ^ ch, 1597334677);
    }
    h1  = Math.imul(h1 ^ (h1 >>> 16), 2246822507);
    h1 ^= Math.imul(h2 ^ (h2 >>> 13), 3266489909);
    h2  = Math.imul(h2 ^ (h2 >>> 16), 2246822507);
    h2 ^= Math.imul(h1 ^ (h1 >>> 13), 3266489909);
    return 4294967296 * (2097151 & h2) + (h1 >>> 0);
}

function checkDevMode() {
    return localStorage.getItem('RIN_PLUS_DEV_MODE') === '1';
}

function setCookie(name, value, hours) {
    var expires = '';
    if (hours) {
        var date = new Date();
        date.setTime(date.getTime() + (hours*60*60*1000));
        expires = '; expires=' + date.toUTCString();
    }
    document.cookie = name + '=' + (value || '')  + expires + '; path=/';
}

function console_info() {
    let out = document.querySelector('#rinsp-debug-console');
    if (out == null) {
        out = addElem(document.body, 'pre', null, { id: 'rinsp-debug-console', style: 'position:fixed;top:20px;left:20px;width:40em;max-height:30em;outline:5px solid #CCC;background:#FFF9;color:#000;overflow:auto;z-index:99999;' });
    }
    out.textContent += Array.from(arguments).join(', ') + '\n';
    out.scrollTop = 10000000;
}

/* ==== START: script initialisation ==== */

function readPinnedPicWallFids() {
    return (localStorage.getItem(PIC_WALL_PREF_KEY)||'').split(' ').map(fid=>fid*1).filter(fid=>!!fid);
}

function redirectToDesktopVersion() {
    if (document.location.pathname === '/simple/index.php') {
        const pageMatch = document.location.search.match(/\^?t(\d+)(?:_(\d+))?\.html$/);
        if (pageMatch) {
            setCookie('mobilever', 1, 720);
            window.location.replace(`${document.location.origin}/read.php?tid=${pageMatch[1]}&fpage=0&toread=&page=${pageMatch[2]||1}`);
            return true;
        }
    }
}

function attemptPicWallRedirect() {
    if (window.location.pathname === '/thread.php') {
        const fid = (window.location.search.match(/[&?]fid[=-](\d+)/)||[])[1] * 1;
        if (fid) {
            const fids = readPinnedPicWallFids();
            if (fids.includes(fid)) {
                window.location.replace(`${window.location.origin}/thread_new.php${window.location.search}${window.location.hash}`);
                return true;
            }
        }
    }
}

const AVATAR_LOADING_PLACEHOLDER = '';
const IMAGE_LOADING_PLACEHOLDER = '';
const IMAGE_LOADING_PLACEHOLDER_DARK = '';

function deferImageLoading(doc) {
    if (DEBUG_MODE) console.info('[STAGE] deferImageLoading');
    const dark = isDarkMode();
    const defaultImagePath = `${document.location.origin}/images/face/`;
    doc.querySelectorAll('.user-pic img[src], #u-portrait img.pic[src]:not([data-org-img])').forEach(img => {
        if (img.src.startsWith(defaultImagePath))
            return; // skip default avatar images
        img.dataset.rinspDeferSrc = img.src;
        img.src = AVATAR_LOADING_PLACEHOLDER;
    });
    doc.querySelectorAll('.gonggao #cate_thread img[src], .tpc_content img[src]:not([src^="images/"]), #info_base img[src]:not([src^="images/"])').forEach(img => {
        img.dataset.rinspDeferSrc = img.src;
        img.src = dark ? IMAGE_LOADING_PLACEHOLDER_DARK : IMAGE_LOADING_PLACEHOLDER;
    });
}

const RESUME_IMG_LOAD_DELAY = 0;
function resumeImageLoading(doc) {
    const lazy = isFastLoadLazyImageMode();
    setTimeout(() => {
        if (DEBUG_MODE) console.info('[STAGE] resumeImageLoading');
        doc.querySelectorAll('img[data-rinsp-defer-src]').forEach(img => {
            if (lazy) {
                img.setAttribute('loading', 'lazy');
            }
            if (img.dataset.orgImg) {
                if (img.dataset.rinspDeferSrc) {
                    img.dataset.orgImg = img.dataset.rinspDeferSrc;
                }
            } else {
                img.classList.add('rinsp-img-loading');
                img.addEventListener('load', () => {
                    img.classList.remove('rinsp-img-loading');
                });
                img.src = img.dataset.rinspDeferSrc;
            }
            delete img.dataset.rinspDeferSrc;
        });
    }, RESUME_IMG_LOAD_DELAY);
}

function initWhenPageRootLoaded() {
    if (!document.documentElement || document.documentElement.dataset.rinspInstalled * 1 > 0) {
        return;
    }
    document.documentElement.dataset.rinspInstalled = '1';
    if (window.matchMedia) {
        try {
            window.matchMedia('(prefers-color-scheme: dark)').addEventListener('change', () => {
                applyDarkTheme();
            });
        } catch (ignore) {}
    }
    if (applyDarkTheme()) {
        // attempt to reduce flashing of unstyled content
        document.documentElement.setAttribute('style', 'filter:invert(0.5) filter:brightness(0.2)');
    } else {
        document.documentElement.setAttribute('style', 'filter:invert(0.5) filter:brightness(1.9)');
    }
    if (isSafeMode()) {
        document.documentElement.classList.add('rinsp-safe-mode');
    }
    if (isSafeModeShowMyAvater()) {
        document.documentElement.classList.add('rinsp-safe-mode-allow-myself');
    }
}
function initWhenPageHeadLoaded() {
    if (!document.head || document.documentElement.dataset.rinspInstalled * 1 > 1) {
        return;
    }
    document.documentElement.dataset.rinspInstalled = '2';
    addGlobalStyle(document.head, customCss);
    addGlobalStyle(document.head, darkThemeCss);
    if (DEV_MODE) {
        addGlobalStyle(document.head, devCss);
    }
}
function initWhenPageDomLoaded() {
    if (!document.body || document.documentElement.dataset.rinspInstalled * 1 > 2) {
        return;
    }
    document.documentElement.dataset.rinspInstalled = '3';

    document.documentElement.removeAttribute('style');
    if (findErrorMessage(document) === '论坛设置:刷新不要快于 1 秒') {
        document.documentElement.dataset.rinspInstalled = '9'; // skip all subsequent initialization
        setTimeout(() => document.location.reload(), 1000);
        return;
    }
    if (requiresDomainRedirect()) {
        attemptDomainRedirect().then(redirecting => {
            if (redirecting) {
                document.documentElement.dataset.rinspInstalled = '9'; // skip all subsequent initialization
            }
        });
    }

    if (isFastLoadMode()) {
        deferImageLoading(document);
        init().catch(ex => console.error(ex));
    }

}
function initWhenPageComplete() {
    if (document.readyState !== 'complete' || document.documentElement.dataset.rinspInstalled * 1 > 3) {
        return;
    }
    document.documentElement.dataset.rinspInstalled = '4';
    document.documentElement.removeAttribute('style'); // unhide those used to be unstyled content

    if (isFastLoadMode()) {
        resumeImageLoading(document);
    } else {
        init().catch(ex => console.error(ex));
    }
}

function begin() {
    if (redirectToDesktopVersion()) {
        return;
    }
    if (attemptPicWallRedirect()) {
        return;
    }
    
    let fallbackTimer = null;
    function handleInitStages() {
        if (DEBUG_MODE) console.info('[STAGE] ' + document.readyState);
        switch (document.readyState) {
            case "loading":
                initWhenPageRootLoaded();
                initWhenPageHeadLoaded();
                break;
            case "interactive":
                initWhenPageRootLoaded();
                initWhenPageHeadLoaded();
                initWhenPageDomLoaded();
                break;
            case "complete":
                if (fallbackTimer) {
                    clearTimeout(fallbackTimer);
                    fallbackTimer = null;
                }
                document.removeEventListener('readystatechange', handleInitStages);
                initWhenPageRootLoaded();
                initWhenPageHeadLoaded();
                initWhenPageDomLoaded();
                initWhenPageComplete();
                break;
        }
    }
    handleInitStages();

    if (document.readyState !== 'complete') {
        document.addEventListener('readystatechange', handleInitStages);
        // force a recheck after N seconds, in case a dodgy GM script plugin is in use
        fallbackTimer = setTimeout(() => {
            handleInitStages();
        }, 5000);
    }

    if (DEBUG_MODE) {
        document.addEventListener('DOMContentLoaded', () => {
            console.info('[STAGE] DOMContentLoaded');
        });
    }

}

if (document.location.pathname !== '/admin.php' && document.location.href !== 'about:blank') {
    begin();
}

})();