【etai2019】明里小秘书

评论区、优优收藏馆、紬宝图片馆。侧边展示、简洁排版和无限翻页

질문, 리뷰하거나, 이 스크립트를 신고하세요.
// ==UserScript==
// @name         【etai2019】明里小秘书
// @version      0.1.7
// @description  评论区、优优收藏馆、紬宝图片馆。侧边展示、简洁排版和无限翻页
// @author       etai2019
// @license      GPL-3.0 License
// @namespace    etai2019

// @match        *://*.sehuatang.net/forum.php?mod=viewthread&tid=*
// @match        *://*.sehuatang.org/forum.php?mod=viewthread&tid=*
// @match        *://*.sehuatang.*/forum.php?mod=viewthread&tid=*
// @match        *://*.jq2t4.com/forum.php?mod=viewthread&tid=*
// @match        *://*.0krgb.com/forum.php?mod=viewthread&tid=*
// @match        *://*.xxjsnc.co/forum.php?mod=viewthread&tid=*
// @match        *://*.o4vag.com/forum.php?mod=viewthread&tid=*
// @match        *://*.weterytrtrr.*/forum.php?mod=viewthread&tid=*
// @match        *://*.qweqwtret.*/forum.php?mod=viewthread&tid=*
// @match        *://*.retreytryuyt.*/forum.php?mod=viewthread&tid=*
// @match        *://*.qwerwrrt.*/forum.php?mod=viewthread&tid=*
// @match        *://*.ds5hk.app/forum.php?mod=viewthread&tid=*
// @match        *://*.30fjp.com/forum.php?mod=viewthread&tid=*
// @match        *://*.18stm.cn/forum.php?mod=viewthread&tid=*
// @match        *://*.xo6c5.com/forum.php?mod=viewthread&tid=*
// @match        *://*.mzjvl.com/forum.php?mod=viewthread&tid=*
// @match        *://*.9xr2.app/forum.php?mod=viewthread&tid=*
// @match        *://*.kzs1w.com/forum.php?mod=viewthread&tid=*
// @match        *://*.nwurc.com/forum.php?mod=viewthread&tid=*
// @match        *://*.zbkz6.app/forum.php?mod=viewthread&tid=*
// @match        *://*.ql75t.cn/forum.php?mod=viewthread&tid=*
// @match        *://*.0uzb0.app/forum.php?mod=viewthread&tid=*
// @match        *://*.d2wpb.com/forum.php?mod=viewthread&tid=*
// @match        *://*.5aylp.com/forum.php?mod=viewthread&tid=*
// @match        *://*.8otvk.app/forum.php?mod=viewthread&tid=*

// @match        *://*.sehuatang.net/thread*
// @match        *://*.sehuatang.org/thread*
// @match        *://*.sehuatang.*/thread*
// @match        *://*.jq2t4.com/thread*
// @match        *://*.0krgb.com/thread*
// @match        *://*.xxjsnc.co/thread*
// @match        *://*.o4vag.com/thread*
// @match        *://*.weterytrtrr.*/thread*
// @match        *://*.qweqwtret.*/thread*
// @match        *://*.retreytryuyt.*/thread*
// @match        *://*.qwerwrrt.*/thread*
// @match        *://*.ds5hk.app/thread*
// @match        *://*.30fjp.com/thread*
// @match        *://*.18stm.cn/thread*
// @match        *://*.xo6c5.com/thread*
// @match        *://*.mzjvl.com/thread*
// @match        *://*.9xr2.app/thread*
// @match        *://*.kzs1w.com/thread*
// @match        *://*.nwurc.com/thread*
// @match        *://*.zbkz6.app/thread*
// @match        *://*.ql75t.cn/thread*
// @match        *://*.0uzb0.app/thread*
// @match        *://*.d2wpb.com/thread*
// @match        *://*.5aylp.com/thread*
// @match        *://*.8otvk.app/thread*

// @icon         

// @require      https://unpkg.com/vue@3/dist/vue.global.js
// @grant        GM_addStyle
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_listValues
// @grant        GM_deleteValue
// @grant        GM_xmlhttpRequest
// ==/UserScript==

(function() {
    'use strict';

    // 添加样式
    GM_addStyle(`
        .app {
            padding: .5em;
            position: fixed;
            top: 4%;
            right: 2%;
            background-color: #fff;
            border: 1px solid black;
            border-radius: 10px;
            overflow: hidden;
            opacity: .6;
        }
        .app:hover {
            opacity: 1;
            z-index: 2;
        }

        .panel-toggle
        {
            text-align: center;
            font-size: 1.2em;
            width: 3.5em;
            margin-left: .2em;
            line-height: 2.2em;
        }
        .toggle-group>.hover,
        .panel-toggle:hover {
            cursor: pointer;
            font-weight: bold;
        }

        .panel {
            height: 90vh;
        }

        .scroll-view {
            height: 91%;
            overflow: auto;
        }

        .panel-header {
            display: flex;
            flex-direction: column;
            border-bottom: 1px solid #eee;
            padding: .5em .5em;
        }

        .app-header-title {
            font-size: 1.4em;
            font-weight: bold;
        }

        .scroll-body .pstatus,
        .scroll-body .pstatus~br {
            display: none;
        }

        .comment-item {
            padding: 10px 5px;
            display: flex;
        }

        .comment-item:hover .footer-action {
            display: inline;
        }

        .comment-left {
            width: 50px;
            height: 50px;
            padding-top: .8em;
        }

        .comment-left>img {
            width: 100%;
            height: 100%;
            border-radius: 50%;
            object-fit: contain;
        }

        .comment-right {
            flex: 1;
            padding: 5px 10px;
            border-bottom: 1px solid #ddd;
            overflow: hidden;
        }

        .comment-basic {
            display: flex;
            justify-content: space-between;

            color: grey;
        }

        .comment-user a {
            color: grey;
            font-weight: normal;
        }

        .comment-content {
            padding: .4em 0;
            color: black;
            font-size: 1.2em;
            max-height: 20em;
            overflow: auto;
        }

        .comment-content img[smilieid] {
            max-width: 1.5em;
        }

        .comment-time {
            color: grey;
        }

        .scroll-footer {
            text-align: center;
            font-size: 12px;
            padding: 0 0 20px 0;
        }

        .footer-action {
            color: grey;
            padding: 0 .4em;
            display: none;
        }

        .footer-action:hover {
            text-decoration: none;
        }

        .panel-wrapper a:hover{
            text-decoration: none;
        }

        .panel-message {
            display: none;
        }
        .panel-message a {
            font-size: .8em;
            font-weight: normal;
        }

        .panel-title:hover {
            cursor: pointer;
        }

        .panel-avatar {
            border-radius: 50%;
            margin-right: .5em;
        }
        .panel-avatar:hover {
            cursor: pointer;
        }

        .app-header {
            display: flex;
            justify-content: space-between;

        }

        .toggle-group {
            display: flex;
            align-items: center;
        }

        .page-link {
            padding: 0 .2em;
        }
        .page-link:hover {
            decoration: none;
            cursor: pointer;
        }

        .rate-popup {
            background: rgba(0, 0, 0, 0.2);
            position: fixed;
            display: flex;
            top: 0;
            left: 0;
            width: 100%;
            height: 100%;
            justify-content: center;
            align-items: center;
        }
        .rate-popup-body {
            width: 30%;
            height: 20%;
            background-color: white;
            font-size: 2em;
            display: flex;
            flex-direction: column;
            justify-content: space-around;
            padding: 0 20px;
            border: 1px solid #000;
            border-radius: 10px;
            box-shadow: 10px 5px 5px red;
        }
        .rate-popup div:hover {
            cursor: pointer
        }

        .image-gallery {
            position: fixed;
            top: 20%;
            left: 20%;
            height: 60%;
            width: 60%;
            padding-bottom: 4em;
            background-color: black;
            border-radius: .5em;
            border: 1px solid #000;
            box-sizing: border-box;
            overflow: hidden;
            display: none;
        }

        .gallery-image-box {
            width: 20%;
            height: 30%;
            position: relative;
            display: inline-block;
        }
        .gallery-image-box img{
            height: 100%;
            width: 100%;
            position: relative;
            object-fit: contain;
        }
        .gallery-image-box-goto {
            position:absolute;
            bottom: 0px;
            right: 0px;
            background-color: purple;
            padding: .5em;
            border-radius: 50%;
            color: #fff;
        }
        .gallery-image-box-goto:hover {
            cursor: pointer;
        }

        .gallery-top-image {
            position: absolute;
            top: 0;
            left: 0;
            height: 100%;
            width: 100%;
            display: none;
            background-color: rgba(0,0,0,0.8);
            z-index: 2;
        }

        .gallery-top-image img {
            height: 100%;
            width: 100%;
            object-fit: contain;
        }

        .gallery-header {
            background-color: #f2f2f2;
            display: flex;
            justify-content: space-between;
            padding: .5em;
            font-size: 1.5em;
            border-bottom: 1px solid #000;
        }

        .gallery-header div:last-of-type {
            padding: 0 1em;
        }

        .gallery-header div:last-of-type:hover {
            cursor: pointer;
        }
    `);

    const wd = window.unsafeWindow || document.defaultView || window;
    wd.Vue = Vue;

    const app$ = document.createElement('div');
    app$.id = "app";
    document.body.appendChild(app$);

    const template = `
    <div class="rate-popup" v-if="showPopup">
        <div class="rate-popup-body">
            <div style="text-align: center" @click="jumpPopup">好用请一定点我回个贴吧!小秘书真的非常需要你的回帖来完善(只弹一次)</div>
            <div style="text-align: center" @click="closePopup">点我关闭</div>
        </div>
    </div>

    <div ref="bubbles" class="bubbles">
        <img style="border-radius:50%; opacity: 0.6;" :src="avatar2" @click="onShowImageGallery"></img>
    </div>
    <div ref="gallery$" class="image-gallery">
        <div class="gallery-header">
            <div>紬宝图片馆 - 已装载 {{imageGalleryLinks.length}} 张图片</div>
            <div @click="onShowImageGallery">关闭</div>
        </div>
        <div style="height: 100%; overflow: auto;">
            <div ref="galleryTopImage$" class="gallery-top-image" @click="$refs.galleryTopImage$.style.display='none'">
                <img :src="galleryTopImageLink"></img>
            </div>
            <div class="gallery-image-box" v-for="link in imageGalleryLinks" :key="link">
                <img :src="link" @click="onShowTopImage(link)"></img>
                <div class="gallery-image-box-goto" @click="onGotoImage(link)">Go</div>
            </div>
        </div>
    </div>

    <div class="app" ref="app">
        <div class="app-header">
            <div class="toggle-group">
                <img class="panel-avatar" :src="avatar" @click="onShowImageGallery">
                <div v-if="toggleOn" class="app-header-title">{{ title }}</div>
            </div>
            <div class="toggle-group">
                <div class="panel-toggle" :class=" {hover:activePanel == '评论区'} " @click="onToggleTab('评论区')">评论区</div>
                <div class="panel-toggle" :class=" {hover:activePanel == '收藏馆'} " @click="onToggleTab('收藏馆')">收藏</div>
            </div>
        </div>
        <div class="panel-wrapper">
            <!-- 评论区 -->
            <div v-if="toggleOn && activePanel=='评论区'" class="panel">
                <div class="panel-header">
                    <div style="display: flex; justify-content: space-between">
                        <div>
                            <select @change="onJumpToPage(selectedPage)" v-model="selectedPage">
                                <option v-for="i in totalPages" :selected="i == selectedPage" :key="i">{{i}}</option>
                            </select>
                            <span>/ {{ totalPages }} 页</span>

                            <span v-if="selectedPage < totalPages"class="page-link" @click="onJumpToPage(totalPages)">末页</span>
                            <span v-if="selectedPage < totalPages"class="page-link" @click="onJumpToPage(selectedPage + 1)">下一页</span>
                            <span v-if="selectedPage > 1" class="page-link" @click="onJumpToPage(selectedPage - 1)" >上一页</span>
                        </div>
                        <div>
                            <a :href="replyThreadUrl">回复帖子</a>
                        </div>
                    </div>
                    <div class="content-length-slider" style="display: flex">
                        <input style="flex:1" type="range" min="1" max="30" v-model="minContentLength" @change="onMinContentLengthChange" />
                        <span>长度 ≥ {{ minContentLength }} 字 </span>
                        <span>(已过滤:{{ comments.length - filteredComments.length }})</span>
                    </div>
                </div>
                <div class="scroll-view" @scroll="onScroll">
                    <div class="scroll-body">
                        <div class="comment-item" v-for="(item, i) in filteredComments" :key="i">
                            <div class="comment-left" v-html="item.avatar"></div>
                            <div class="comment-right">
                                <div class="comment-basic">
                                    <div class="comment-user" v-html="item.user"></div>
                                    <div class="comment-post-number">{{item.postNum}}</div>
                                </div>

                                <div class="comment-content" v-html="item.content"></div>
                                <div class="comment-footer">
                                    <span class="comment-time" v-html="item.time"></span>
                                    <a class="footer-action" :href="item.postUrl">跳转</a>
                                    <a class="footer-action" v-if="item.replyUrl" :href="item.replyUrl" onclick="if (!window.__cfRLUnblockHandlers) return false; showWindow('reply', this.href)">回复</a>
                                </div>
                            </div>
                        </div>
                    </div>
                    <div class="scroll-footer">{{ isLoading ? "紬宝正在努力加载下一页..." : "紬宝:没有更多评论啦~" }}</div>
                </div>

                <div class="panel-footer" style="display: flex">
                    <span>宽度: {{ appWidth }}%</span>
                    <input style="flex:1" type="range" min="15" max="50" v-model="appWidth" @change="onWidthSliderChange" />
                </div>
            </div>

            <!-- 收藏馆 -->
            <div v-if="toggleOn && activePanel=='收藏馆'" class="gallery">
                <div class="fs-body">
                    <div class="fs-message" style="text-align: center">{{ keywordDetail || "紬宝:今天试试「" + keywords[Math.floor(Math.random() * keywords.length)] + "」吧?" }}</div>

                    <div class="fs-button-group">
                        <div class="fs-button-group-item" v-for="(item, index) in keywords" :key="item">
                            <div class="fs-button" @mouseover="hoveredKeyword=item"  @mouseleave="hoveredKeyword=''" @click="onSearch(item)">
                                {{ item }}
                            </div>
                            <div class="fs-button-close" @click="onRemove(item)">-</div>
                        </div>
                    </div>

                    <div v-if="loading" class="fs-mask">唤起搜索中,请等待...</div>
                </div>
                <div class="fs-footer"><input v-model="addKeyword" :placeholder="inputHint" /><div class="fs-button" @click="onAdd">+</div></div>

                <div class="panel-footer" style="display: flex">
                    <span>宽度: {{ appWidth }}%</span>
                    <input style="flex:1" type="range" min="15" max="50" v-model="appWidth" @change="onWidthSliderChange" />
                </div>
            </div>


        </div>
    </div>
    `

    const app = Vue.createApp({
        template: template,
        data() {
            return {
                activePanel: "",
                avatar: "",
                avatar2: "",
                toggleOn: true,
                commentsPerPage: 9,
                showPopup: false,
                lifetimePages: 0,

                // 图片展
                isShowImageGallery: false,
                imageGalleryLinks: [],
                galleryTopImageLink: '',

                // 评论区
                appWidth: 22,
                comments: [],
                isLoading: false,
                nextPageUrl: "",
                totalComments: 0,
                totalPages: 1,
                selectedPage: 1,
                replyThreadUrl: "",
                minContentLength: 0,
                hasInit: false,

                // 收藏馆
                keywords: ['明里'],
                addKeyword: "",
                loading: false,
                inputHint: "添加收藏快捷搜索~",
                keywordStats: {},
                hoveredKeyword: "",
            }
        },
        computed: {
            title() {
                if (this.activePanel == "评论区") {
                    return `${this.totalComments} 条评论`
                } else if (this.activePanel == "收藏馆") {
                    return `${this.keywords.length} 项收藏`
                } else {
                    return ""
                }
            },
            filteredComments() {
                return this.comments.filter(x => x.contentText.length >= this.minContentLength);
            },
            keywordDetail() {
                if (this.hoveredKeyword == "") return "";
                if (!(this.hoveredKeyword in this.keywordStats)) return `没有搜索过 ${this.hoveredKeyword} 哦~`;

                const stat = this.keywordStats[this.hoveredKeyword]
                return `搜索次数:${stat["clicks"]},最近搜索:${new Date(stat["last"]).toLocaleDateString()}`;
            }
        },
        methods: {
            applyGalleryDisplay() {
                this.$refs.gallery$.style.display = this.isShowImageGallery ? 'block' : 'none';
            },
            onGotoImage(link){
                const el = document.querySelector(`img[file^="${link}"]`);
                el.scrollIntoView({ behavior: "smooth", block: "center", inline: "center" });
                // console.log(el)
                this.isShowImageGallery = false;
                this.applyGalleryDisplay();
            },
            onShowTopImage(link) {
                this.galleryTopImageLink = link;
                this.$refs.galleryTopImage$.style.top = this.$refs.gallery$.scrollTop + "px";
                this.$refs.galleryTopImage$.style.display = 'block';
            },
            onShowImageGallery(){
                this.isShowImageGallery = !this.isShowImageGallery;
                this.applyGalleryDisplay();
            },
            jumpPopup() {
                window.open(`https://${window.location.host}/thread-1960202-1-1.html`);
                this.closePopup();
            },
            closePopup() {
                this.showPopup = false;
            },
            onWidthSliderChange(){
                this.saveToConfig();
                this.$refs.app.style.width = this.appWidth + "%";
            },
            async checkAndLoadMoreCommentsInFirstPage() {
                if (this.filteredComments.length < this.commentsPerPage) {
                    await this.loadNextPage();
                }
            },
            async onMinContentLengthChange(){
                await this.checkAndLoadMoreCommentsInFirstPage();
                this.saveToConfig();
            },
            addComments(comments$, pageUrl) {
                const newComments = []
                for (const comment$ of comments$) {
                    try {
                        const avatar = comment$.querySelector('.avatar img').outerHTML;
                        const user = comment$.querySelector('.authi').innerHTML;
                        const content = comment$.querySelector('.t_f').innerHTML;
                        const contentText = comment$.querySelector('.t_f').innerText.trim();
                        const time = comment$.querySelector('[id^="authorpost"]').innerHTML;
                        const postUrl = pageUrl.split('#')[0] + '#' + comment$["id"];
                        const replyUrl = comment$.querySelector('.fastre');
                        const postNum = comment$.querySelector('.pi strong').textContent;

                        if (postNum.trim() == "楼主") continue;

                        newComments.push({
                            avatar,
                            user,
                            content,
                            time,
                            postUrl,
                            replyUrl,
                            postNum,
                            contentText
                        });
                    } catch (error) {
                        console.error("parse error", comment$);
                        console.log(error)
                    }
                }

                this.comments.push(...newComments);
            },
            async loadNextPage(minCommentsCount) {
                if (minCommentsCount == null) {
                    minCommentsCount = this.commentsPerPage
                }

                if (minCommentsCount <= 0 || !this.nextPageUrl) {
                  return;
                }

                if (!this.nextPageUrl.startsWith("http")) {
                    this.nextPageUrl = location.protocol + "//" + location.host + "/" + this.nextPageUrl;
                }

                this.isLoading = true;

                const response = await fetch(this.nextPageUrl);
                const text = await response.text();

                const div = document.createElement("div");
                div.innerHTML = text;

                const comments$ = Array.from(div.querySelectorAll(`div[id^="post_"]:not([id^="post_ra"], [id^="post_new"])`));
                const prevCommentsCount = this.filteredComments.length;
                this.addComments(comments$, this.nextPageUrl);

                console.log(this.nextPageUrl);
                // 页面地址格式可能不同
                if (new URL(this.nextPageUrl).searchParams.get("page") != null) {
                    // https://sehuatang.net/forum.php?mod=viewthread&tid=1947970&extra=page%3D1&page=2
                    this.selectedPage = parseInt(new URL(this.nextPageUrl).searchParams.get("page"))
                } else {
                    // https://sehuatang.net/thread-1947970-3-1.html
                    this.selectedPage = parseInt(new URL(this.nextPageUrl).pathname.split('-').at(-2));
                }

                // 载入完成
                this.nextPageUrl = div.querySelector(".nxt")?.getAttribute("href");
                this.isLoading = false;

                // 继续加载下一页
                const newMinCommentsCount = minCommentsCount - (this.filteredComments.length - prevCommentsCount);
                await this.loadNextPage(newMinCommentsCount);

                // 弹窗宣传
                ++this.lifetimePages;
                this.saveToConfig();
                if (this.lifetimePages == 50) {
                    this.showPopup = true;
                }
            },
            async onJumpToPage(pageNum) {
                if (pageNum < 1 || pageNum > this.totalPages) return;

                const url = new URL(location.href);
                url.searchParams.set("page", pageNum);
                url.hash = "";
                this.nextPageUrl = url.href;
                this.comments = []
                this.selectedPage = pageNum;
                await this.loadNextPage();
            },
            async onScroll() {
                const commentPanel$ = document.querySelector('.scroll-view');
                if (commentPanel$.clientHeight + commentPanel$.scrollTop >= commentPanel$.scrollHeight - 100 && !app.isLoading) {
                    await app.loadNextPage();
                }
            },
            loadFromConfig() {
                this.keywords = GM_getValue("fsKeywords", '明里').split("\n");
                // this.keywords.sort((a, b) => a.localeCompare(b));
                this.appWidth = GM_getValue("appWidth", 22);
                this.minContentLength = GM_getValue("minContentLength", 5);
                this.keywordStats = JSON.parse(GM_getValue("keywordStats", "{}"));
                this.lifetimePages = GM_getValue("lifetimePages", 0);
            },
            saveToConfig() {
                GM_setValue("fsKeywords", this.keywords.join("\n"));
                GM_setValue("appWidth", this.appWidth);
                GM_setValue("minContentLength", this.minContentLength);
                GM_setValue("keywordStats", JSON.stringify(this.keywordStats));
                GM_setValue("lifetimePages", this.lifetimePages);
            },
            onAdd() {
                if (this.addKeyword?.trim() == "") {
                    this.inputHint = "请至少输入一个字符哦"
                    setTimeout(() => {
                        this.inputHint = "添加收藏快捷搜索~"
                    }, 2000);
                    return;
                }

                this.loadFromConfig();
                this.keywords = [...this.keywords, this.addKeyword];
                this.keywords = [...new Set(this.keywords)];
                this.saveToConfig();

                this.loadFromConfig();
                this.addKeyword = "";
            },
            onRemove(item) {
                this.keywords = this.keywords.filter(x => x != item);
                this.saveToConfig();
                this.loadFromConfig();
            },
            onSearch(query){
                const formhash = document.querySelector('input[name="formhash"]')?.value;
                const sehuatangURL = `https://${window.location.host}`;

                if (formhash) {
                    this.loading = true;

                    const this$ = this;
                    GM_xmlhttpRequest({
                        method: "post",
                        url: sehuatangURL+"/search.php?mod=forum",
                        data: `formhash=${formhash}&srchtxt=${query}&searchsubmit=yes`,
                        headers:  {
                            "Content-Type": "application/x-www-form-urlencoded",
                            "Origin":sehuatangURL,
                            "Referer":sehuatangURL
                        },
                        onload: function(data){
                            if(data.finalUrl){
                                window.open(data.finalUrl);
                            } else{
                                window.open(`${sehuatangURL}/search.php`);
                            }

                            this$.loading = false;
                            this$.logKeyword(query);
                        },
                        onerror : function(err){
                            console.log('SearchSehuatang error')
                            console.log(err)

                            window.open(`${sehuatangURL}/search.php`);
                            this$.logKeyword(query);

                            this$.loading = false;
                        }
                    });
                } else {
                    window.open(`${sehuatangURL}/search.php`);
                    this$.logKeyword(query);
                }
            },
            async onToggleTab(panelName) {
                if (this.activePanel != panelName) {
                    this.toggleOn = true;
                    this.activePanel = panelName;
                } else {
                    this.toggleOn = !this.toggleOn;
                }

                // 面板宽度
                if (!this.toggleOn) {
                    this.activePanel = "";
                    this.$refs.app.style.removeProperty('width');
                } else {
                    this.$refs.app.style.width = this.appWidth + "%";
                }

                // 初次载入
                if (this.activePanel == "评论区" && this.toggleOn && !this.hasInit) {
                    console.log("初始化评论区");
                    this.hasInit = true;
                    const comments$ = Array.from(document.querySelectorAll(`div[id^="post_"]:not([id^="post_ra"], [id^="post_new"])`));
                    this.addComments(comments$, location.href);
                    // 尝试加载首页
                    await this.checkAndLoadMoreCommentsInFirstPage();
                }

                // GM_setValue("ToggleOn", this.toggleOn);
            },
            logKeyword(keyword) {
                this.loadFromConfig();
                if (!(keyword in this.keywordStats)) {
                    this.keywordStats[keyword] = {
                        last: Date.now(),
                        clicks: 0
                    };
                }

                this.keywordStats[keyword]['last'] = Date.now();
                this.keywordStats[keyword]['clicks']++;
                this.saveToConfig();
            },
        },
        mounted() {
            // 初始化气泡
            if (this.imageGalleryLinks.length > 0) {
                const scrolltop$ = document.querySelector('#scrolltop');
                scrolltop$.insertBefore(this.$refs.bubbles, scrolltop$.childNodes[0]);
                showTopLink();
                scrolltop$.style.visibility = 'visible';
            }
        },
        async created() {
            // 初始化图片展
            const imgs$ = Array.from(document.querySelector('.t_fsz').querySelectorAll('img[file],img[src]'))
                .filter(x => {
                    const imageLink = (x.getAttribute('file') || x.getAttribute('src'))
                    return imageLink.startsWith("http") && imageLink.indexOf("static") < 0;
                });
            imgs$.forEach(x => {
                const imageLink = x.getAttribute('file') || x.getAttribute('src')
                x.setAttribute("src", imageLink);
                x.setAttribute("lazyloaded", true);
                this.imageGalleryLinks.push(imageLink);
            });

            // 初始化收藏馆
            this.loadFromConfig();

            // 初始化评论区
            // const comments$ = Array.from(document.querySelectorAll(`div[id^="post_"]:not([id^="post_ra"], [id^="post_new"])`));
            // this.addComments(comments$, location.href);
            // // 默认加载下一页
            // await this.checkAndLoadMoreCommentsInFirstPage();

            this.nextPageUrl= document.querySelector(".nxt")?.getAttribute("href");
            this.replyThreadUrl = location.href.split('#')[0] + '#f_pst';

            this.selectedPage = parseInt(new URL(location.href).searchParams.get('page')) || 1;
            this.totalPages = parseInt(document.querySelector('.pg label span')?.textContent.match(/\d/g).join("")) || 1;

            const reply = Array.from(document.querySelectorAll('.hm .xi1')?.values()).at(1)?.innerText;
            this.totalComments = reply != null ? parseInt(reply) : this.totalPages * 10;

            // 初始化面板状态
            this.toggleOn = GM_getValue("ToggleOn", false);
            this.avatar = "";
            this.avatar2 = "";

        }
    }).mount('#app')

    // Your code here...
})();

(function() {
    GM_addStyle(`
        .fs-button-group {
            display: flex;
            flex-wrap: wrap;
            padding: 0 1em;
        }
        .fs-button-group-item {
            display: flex;
            height: 2em;
            font-size: 1.2em;
            margin: .3em;
            align-items: center;
        }
        .fs-button {
            border: .1em solid brown;
            border-radius: .8em;
            color: black;
            padding: 0 .8em;
            line-height: 2em;
            margin-right: .2em;
        }
        .fs-button:hover {
            cursor: pointer;
            border: .1em solid orange
        }

        .fs-button-close {
            border-radius: 50%;
            height: 1em;
            width: 1em;
            line-height: 1em;
            border: .2em solid #777;
            text-align: center;
            background-color: grey;
            color: #fff;
            margin: 0 0 1em -1em;
        }
        .fs-button-close:hover {
            cursor: pointer;
            border: .2em solid orange;
        }

        .fs-edit {
            padding: 0 1em;
        }

        .fs-edit-area {
            width: 100%;
            height: 20em;
            display: block;
            box-sizing: border-box;
        }

        .fs-mask {
            background-color: rgba(0, 0, 0, 0.6);
            position: absolute;
            top: 0;
            left: 0;
            width: 100%;
            height: 100%;
            color: #eee;
            display: flex;
            justify-content: center;
            align-items: center;
        }

        .fs-footer {
            display: flex;
            justify-content: space-between;
            margin-top: .5em;
            padding: .5em 2em .2em;
            border-top: 1px solid #ddd;
            font-size: 1.2em;
        }

        .fs-footer input {
            flex: 1;
            border-radius: 1em;
            margin-right: .5em;
            text-align: center;
            border: 1px solid black;
        }
    `);
})();