我的搜索

打造订阅式搜索,让我的搜索,只搜精品!

当前为 2025-08-15 提交的版本,查看 最新版本

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

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

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         我的搜索
// @namespace    http://tampermonkey.net/
// @version      7.8.0
// @description  打造订阅式搜索,让我的搜索,只搜精品!
// @license MIT
// @author       zhuangjie
// @exclude  http://127.0.0.1*
// @exclude  http://localhost*
// @match      *://*/*
// @exclude  http://192.168.*
// @icon         
// @require https://cdn.jsdelivr.net/npm/[email protected]/dist/jquery.min.js

// @require      https://cdn.jsdelivr.net/npm/[email protected]/dist/showdown.min.js
// @resource markdown-css https://cdnjs.cloudflare.com/ajax/libs/github-markdown-css/5.8.1/github-markdown.min.css

// @require      https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.9.0/highlight.min.js
// @resource code-css https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.9.0/styles/github.min.css

// @require https://update.greasyfork.org/scripts/530877/1560004/ms-pinyin.js
// @require https://update.greasyfork.org/scripts/501646/1642319/string-overlap-matching-degree.js
// @noframes

// @grant        window.onurlchange
// @grant        GM_setValue
// @grant        GM_xmlhttpRequest
// @connect      *
// @grant        GM_getValue
// @grant        GM_addStyle
// @grant        GM_getResourceText

// @grant        GM_getResourceURL
// @grant GM_deleteValue
// @grant GM_registerMenuCommand
// @grant        GM_info

// ==/UserScript==

// 参数tis: actualWindow 是真实的window,而如果在下面脚本内访问window则不是,是沙盒的XPCNativeWrapper对象,详情https://blog.rxliuli.com/p/e55a67646bf546b3900ce270a6fbc6ca/
(function(actualWindow) {
    'use strict';

    // 模块一:快捷键触发某一事件 (属于触发策略组)
    // 模块二:搜索视图(显示与隐藏)(属于搜索视图组)
    // 模块三:触发策略组触发策略触发搜索视图组视图
    // 模块四:根据用户提供的策略(策略属于数据生成策略组)生成搜索项的数据库
    // 模块五:视图接入数据库

    // 判断当前是否在iframe里面,
    function currentIsIframe() {
        if (self.frameElement && self.frameElement.tagName == "IFRAME") return true;
        if (window.frames.length != parent.frames.length) return true;
        if (self != top) return true;
        return false;
    }


    // 如果当前是ifrae,结束脚本执行
    let MY_SEARCH_SCRIPT_VIEW_SHOW_EVENT = "MY_SEARCH_SCRIPT_VIEW_SHOW_EVENT";
    if(currentIsIframe()) {
        // 虽然iframe不能初始化脚本,但可以作为父窗口的事件触发源
        triggerAndEvent("ctrl+alt+s", function () { // 通知主容器显示搜索框
            window.parent.postMessage(MY_SEARCH_SCRIPT_VIEW_SHOW_EVENT, '*');
        })
        // 结束脚本执行
        return;
    }

    // css resource加载器
    function cssLoad(prefix,css = "",{isResourceName = false,replacePrefix}) {
        if(isResourceName) css = GM_getResourceText(css);

        if(replacePrefix && prefix) {
           css = css.toReplaceAll(replacePrefix,prefix);
        }else if(prefix) {
          // 对css原始内容引入前缀
          css = `.${prefix} {
           ${css}
          }`;
        }
        GM_addStyle(css);
        return prefix;
    }
    // 正则捕获
    function captureRegEx(regex, text) {
        let m;
        let result = []; // 一组一组 [[],[],...]
        regex.lastIndex = 0; // 重置lastIndex
        while ((m = regex.exec(text)) !== null) {
            let group = [];
            group.push(...m);
            if(group.length != 0) result.push(group);
        }
        return result;
    }
    // 滚动到目标文本  可以指定容器
    function scrollToText(text, index = 0, container = document.body) {
        // 创建高亮样式(只需执行一次)
        if (!document.getElementById('highlight-style')) {
            const style = document.createElement('style');
            style.id = 'highlight-style';
            style.textContent = `.highlight-text { border-bottom: 2px solid red !important;color:red; }`;
            document.head.appendChild(style);
        }

        // 处理容器参数
        const containerElement = typeof container === 'string'
        ? document.querySelector(container)
        : container;

        if (!containerElement) {
            console.error('Container not found:', container);
            return;
        }

        // 清理旧高亮
        Array.from(containerElement.getElementsByClassName('highlight-text'))
            .forEach(span => {
            const textNode = document.createTextNode(span.textContent);
            span.parentNode.replaceChild(textNode, span);
        });

        if (!text) return;

        // 获取所有文本节点
        const textNodes = [];
        const walker = document.createTreeWalker(
            containerElement,
            NodeFilter.SHOW_TEXT,
            { acceptNode: () => NodeFilter.FILTER_ACCEPT }
        );
        while (walker.nextNode()) textNodes.push(walker.currentNode);

        // 查找所有匹配项
        const matches = [];
        const searchLen = text.length;
        for (const node of textNodes) {
            const nodeText = node.nodeValue;
            let pos = 0;
            while (pos <= nodeText.length - searchLen) {
                const idx = nodeText.indexOf(text, pos);
                if (idx === -1) break;
                matches.push({ node, start: idx, end: idx + searchLen });
                pos = idx + 1; // 允许重叠匹配
            }
        }

        // 验证索引
        if (index < 0 || index >= matches.length) {
            console.error(`Index ${index} out of range (0-${matches.length - 1})`);
            return;
        }

        // 处理目标匹配项
        const { node, start, end } = matches[index];
        const middle = node.splitText(start);
        const after = middle.splitText(end - start);
        const span = document.createElement('span');
        span.className = 'highlight-text';
        span.textContent = middle.nodeValue;
        middle.parentNode.replaceChild(span, middle);

        // 滚动到目标位置
        span.scrollIntoView({
            behavior: 'smooth',
            block: 'center',
            inline: 'nearest'
        });
    }
    // 重写console.log方法
    let originalLog = console.log;
    console.logout = function() {
        const prefix = "[我的搜索log]>>> ";
        const args = [prefix].concat(Array.from(arguments));
        originalLog.apply(console, args);
    }
    // markdown转html 转换器 【1】
    // 更多配置项:https://github.com/showdownjs/showdown
    const converter = new showdown.Converter({
        // 将换行符解析为<br>
        simpleLineBreaks:true,
        // 新窗口打开链接
        openLinksInNewWindow: true,
        metadata:true,
        // 不允许下划线变斜体
        literalMidWordUnderscores: true,
        // 识别md表格
        tables: true,
        // www.baidu.com 会识别为链接
        simplifiedAutoLink: true
    });

    function md2html(rawText) {
        // 方案二: return marked.parse(rawText);  // @require      https://cdnjs.cloudflare.com/ajax/libs/marked/9.0.2/marked.min.js
        return converter.makeHtml( rawText )
    }

    // 提取URL根域名
    function getUrlRoot(url,isRemovePrefix = true,isRemoveSuffix = true) {
        if(! (typeof url == "string" || url.length >= 3)) return url;
        // 可处理
        // 判断是否有前缀
        let prefix = "";
        let root = "";
        let suffix = "";
        // 提取前缀
        if(url.indexOf("://") != -1) {
            // 存在前缀
            let prefixSplitArr = url.split("://")
            prefix = prefixSplitArr[0];
            url = prefixSplitArr[1];
        }
        // 提取root 和suffix
        if(url.indexOf("/") != -1) {
            let twoLevelIndex = url.indexOf("/")
            root = url.substr(0,twoLevelIndex);
            suffix = url.substr(twoLevelIndex,url.length-1);
        }else {
            root = url;
            suffix = "";
        }
        return ((!isRemovePrefix && prefix != "")?(prefix+"://"):"") + root + (isRemoveSuffix?"":suffix);
    }
    // 解析出http url 结构
    function parseUrl(url) {
        const regex = /(https?:|)\/\/([^\/]*|[^\/]*)(\/[^\s\?]*|)(\??[^\s]*|)/;
        const matches = regex.exec(url);
        if (matches) {
            const protocol = matches[1];
            const domain = matches[2];
            const path = matches[3];
            const params = matches[4];
            const rootUrl = protocol+"//"+domain
            const rawUrl = url;
            return {protocol,domain,path,params,rootUrl,rawUrl}
        }
        return null;
    }
    function isHttpUrl(url = "") {
        url = url.trim();
        return /^https?:\/\/.+/i.test(url)
    }

    // 检查网站是否可用
    function checkUsability(templateUrl,isStopCheck = false) {
        return new Promise(function (resolve, reject) {
            // 判断是否要检查
            if(isStopCheck) {
                reject(null);
                return;
            }
            var img=document.createElement("img");
            img.src = templateUrl.fillByObj(parseUrl("https://www.baidu.com"));
            img.style= "display:none;";
            img.onerror = function(e) {
                setTimeout(function() {img.remove();},20)
                reject(null);
            }
            img.onload = function(e) {
                setTimeout(function() {img.remove();},20)
                resolve(templateUrl);
            }
            document.body.appendChild(img);
        });
    }



    // 数据缓存器
    let cache = {
        prefix: "",
        get(key) {
            return GM_getValue(this.prefix+key);
        },
        set(key,value) {
            this.remove(this.prefix+key);
            GM_setValue(this.prefix+key,value);
        },
        jGet(key) {
            let value = GM_getValue(this.prefix+key);
            if( value == null) return value;
            return JSON.parse(value);
        },
        jSet(key,value) {
            value = JSON.stringify(value)
            GM_setValue(this.prefix+key,value);
        },
        remove(key) {
            GM_deleteValue(this.prefix+key);
        },
        cookieSet(cname,cvalue,exdays) {
            var d = new Date();
            d.setTime(d.getTime()+exdays);
            var expires = "expires="+d.toGMTString();
            document.cookie = cname + "=" + cvalue + "; " + expires;
        },
        cookieGet(cname) {
            var name = cname + "=";
            var ca = document.cookie.split(';');
            for(var i=0; i<ca.length; i++)
            {
                var c = ca[i].trim();
                if (c.indexOf(name)==0) return c.substring(name.length,c.length);
            }
            return "";
        }
    }
    function removeDuplicates(objs, props = []) {
        if (!Array.isArray(objs) || objs.length === 0) return [];

        // 深度比较两个值是否相等
        function deepEqual(a, b) {
            if (a === b) {
                // 处理0和-0的情况
                return a !== 0 || 1 / a === 1 / b;
            }

            // 处理 NaN
            if (Number.isNaN(a) && Number.isNaN(b)) return true;

            if (typeof a !== 'object' || a === null ||
                typeof b !== 'object' || b === null) {
                return false;
            }

            const keysA = Object.keys(a);
            const keysB = Object.keys(b);
            if (keysA.length !== keysB.length) return false;

            for (const key of keysA) {
                if (!keysB.includes(key) || !deepEqual(a[key], b[key])) {
                    return false;
                }
            }
            return true;
        }

        // 比较两个元素是否相等
        function compareObjects(obj1, obj2) {
            if (props.length > 0) {
                // 按指定属性逐个深度比较
                return props.every(p => deepEqual(obj1[p], obj2[p]));
            } else {
                return deepEqual(obj1, obj2);
            }
        }

        let result = [];

        for (const item of objs) {
            if (!result.some(r => compareObjects(item, r))) {
                result.push(item);
            }
        }

        return result;
    }

    // 责任链对象工厂
    function getResponsibilityChain() {
        return {
            chains: [],
            add(chain = {weight:0,fun: (data,ref)=>data}) {
                if(chain == null ) throw new Error("[ERROR]责任链对象: 你添加了一个null Chain!")
                if(chain.weight == undefined || chain.fun == undefined) throw new Error("[ERROR]责任链对象: 你传入的Chain是无效的!")
                this.chains.push(chain)
            },
            trigger(baton) {
                // 排序,通过weight从高到低
                this.chains = this.chains.sort((a, b)=>b.weight - a.weight);
                // 开始执行
                let _baton = baton;
                let ref = {
                    stop: false
                }
                for(let chain of this.chains) {
                    if( ref.stop) {
                        break;
                    }
                    _baton = chain.fun(_baton,ref)

                }
                return _baton;
            }
        }
    }
    // 请求包装
    function request(type, url, { query, body,header = {},config ={}} = {}) {
        return new Promise(function(resolve, reject) {
            var formData = new FormData();
            var isFormData = false;

            if (body) {
                for (var key in body) {
                    if (body[key] instanceof File) {
                        formData.append(key, body[key]);
                        isFormData = true;
                    } else {
                        formData.append(key, JSON.stringify(body[key]));
                    }
                }
            }

            var ajaxOptions = {
                ...config,
                url: url + (query ? ("?" + $.param(query)) : ""),
                method: type,
                headers: header,
                success: function(response) {
                    resolve(response);
                },
                error: function(jqXHR, textStatus, errorThrown) {
                    reject(errorThrown);
                }
            };

            if (isFormData) {
                ajaxOptions.data = formData;
                ajaxOptions.processData = false;
                ajaxOptions.contentType = false;
            } else {
                ajaxOptions.data = JSON.stringify(body);
            }
            config?.crossDomain
             // ajax兼容
             ? GM_xmlhttpRequest({
                ...ajaxOptions,
                onload: (xhr)=>{
                    ajaxOptions.success(xhr.responseText,xhr)
                },
                onerror: ajaxOptions.error,
                ontimeout: ajaxOptions.error
            })
            : $.ajax(ajaxOptions);
        });
    }
    // 正则字符串匹配目录字符串-匹配工具
    function isMatch(regexStr, text) {
        // 创建正则表达式对象
        let regex = new RegExp(regexStr);
        // 使用 test 方法测试字符串是否匹配
        return regex.test(text);
    }
    // `视图渲染完成`后调用
    function waitViewRenderingComplete(callback) {
        // 这里模拟的是当下次渲染完成后执行
        setTimeout(callback,30)
    }
    // 结构化的css 转 平铺的css
    function flattenCSS(cssObject, parentSelector = '') {
        let result = '';

        for (const [selector, rules] of Object.entries(cssObject)) {
            if (typeof rules === 'object') {
                // 如果 rules 是对象,说明有嵌套,需要递归处理
                const fullSelector = parentSelector ? `${parentSelector} ${selector}` : selector;
                result += flattenCSS(rules, fullSelector);
            } else {
                // 如果 rules 不是对象,说明是样式规则
                const fullSelector = parentSelector ? `${parentSelector} { ${selector}: ${rules}; }\n` : '';
                result += fullSelector;
            }
        }

        return result;
    }
    // ref引用变量
    function ref(initValue = null) {
        return {value: initValue}
    }
    // ==偏业务工具函数==
    // 使用责任链模式——对pageText进行操作的工具
    class PageTextHandleChains {
        pageText = "";
        constructor(pageText = "") {
            this.pageText = pageText;
        }
        setPageText(newPageText) {
            this.pageText = newPageText;
        }
        getPageText() {
            return this.pageText;
        }
        // 解析双标签-获取指定标签下指定属性下的值
        parseDoubleTab(tabName,attrName) {
            // 返回指定标签下指定属性下的值
            const regex = RegExp(`<\\s*${tabName}[^<>]*\\s*${attrName}="([^<>]*)"\\s*>([\\s\\S]*?)<\/\\s*${tabName}\\s*>`,"gm");
            let m;
            let tabNameArr = [];
            let copyPageText = this.pageText;
            // 注意下面的 copyPageText 不能改变
            while ((m = regex.exec(copyPageText)) !== null) {
                // 这对于避免零宽度匹配的无限循环是必要的
                if (m.index === regex.lastIndex) {
                    regex.lastIndex++;
                }
                tabNameArr.push({
                    attrValue: m[1],
                    tabValue: m[2]
                })
                const newPageText =this.pageText.replace(m[0], "");
                this.pageText = newPageText;
            }
            return tabNameArr;
        }
        // 解析双标签-只获取值
        parseDoubleTabValue(tabName) {
            // 返回指定标签下指定属性下的值
            const regex = RegExp(`<\\s*${tabName}[^<>]*\\s*>([\\s\\S]*?)<\/\\s*${tabName}\\s*>`,"gm");
            let m;
            let tabNameArr = [];
            let copyPageText = this.pageText;
            while ((m = regex.exec(copyPageText)) !== null) {
                // 这对于避免零宽度匹配的无限循环是必要的
                if (m.index === regex.lastIndex) {
                    regex.lastIndex++;
                }
                tabNameArr.push({
                    tabValue: m[1]
                })
                const newPageText =this.pageText.replace(m[0], "");
                this.pageText = newPageText;
            }

            return tabNameArr;
        }

        // 解析所有指定的单标签
        parseAllDesignatedSingTags(parseTabName) {
            // 匹配标签的正则表达式
            const regex = /<(\w+)::([\S]+)(.*?)\/>/g;
            // 匹配属性键值对的正则表达式,支持连字符
            const attributesRegex = /([\w-]+)="(.*?)"/g;
            let matches;
            const result = [];
            let modifiedString = this.pageText;

            while ((matches = regex.exec(this.pageText)) !== null) {
                const tabName = matches[1];
                const tabValue = matches[2];
                const attributesString = matches[3];

                console.log(tabName, parseTabName);
                if (tabName !== parseTabName) continue;

                const attributes = {};
                let attrMatch;
                while ((attrMatch = attributesRegex.exec(attributesString)) !== null) {
                    attributes[attrMatch[1]] = attrMatch[2];
                }

                result.push({
                    tabName,
                    tabValue,
                    ...attributes
                });

                // 将匹配到的内容替换为空字符串
                modifiedString = modifiedString.replace(matches[0], '');
            }

            // 更新 pageText
            this.pageText = modifiedString;
            return result;
        }
        // 根据单标签的元信息进行stringify
        rebuildTags(tagMetaArr = []) {
            return tagMetaArr.map(tag => {
                const { tabName, tabValue, ...attributes } = tag;
                const attributesString = Object.entries(attributes)
                .map(([key, value]) => `${key}="${value}"`)
                .join(' ');
                return `<${tabName}::${tabValue} ${attributesString} />`;
            }).join('\n');
        }
    }

    // 根据反馈的错误项调整templates位置,使得错误的靠后
    function feedbackError(saveKey,currentErrorItem) {
        let items = cache.get(saveKey)??[];
        let foundIndex = -1; // -1-查找模式 , n-已找到 n是所在位置模式
        let foundValue = null;
        for(let i = 0; i < items.length; i++) {
            let item = items[i];
            if(foundIndex == -1 ) {
                if(item == currentErrorItem) {
                    foundIndex = i;
                    foundValue = items[i];
                }
            }else {
                items[i-1] = items[i];
                // 查看是否是最后一个
                if( i == items.length - 1 ) items[i] = foundValue;
            }
        }
        cache.set(saveKey,items);
        return items;
    }
    // 数据项选择记录器
    class SelectHistoryRecorder {
        static HISTORY_CACHE_KEY = "HISTORY_CACHE_KEY";
        static defaultIdFun = (item)=> JSON.stringify(item);
        static select(item,idFun = SelectHistoryRecorder.defaultIdFun) {
            // 记录到历史
            let key = idFun(item);
            let history = cache.get( SelectHistoryRecorder.HISTORY_CACHE_KEY )??[];
            history = history.filter(_item=>idFun(_item) != key) // 将原来的去除
            history.unshift({...item}) // 必须拷贝
            // 清理掉索引,索引只是本次数据加载有效的,而我们存储的历史数据是不随数据加载而改变的,也就是如果缓存索引会失效,没有索引它会自己找,当然我们会提供我们这里的数据给它找,如果在全局数据中匹配不到的话
            history.forEach(_item=>{
                delete _item.index;
                return _item;
            })
            // 缓存历史数据
            cache.set(SelectHistoryRecorder.HISTORY_CACHE_KEY,history);
        }
        static history(count) {
            let history = cache.get( SelectHistoryRecorder.HISTORY_CACHE_KEY )??[];
            if(count == null) return history; // 如果没有传入count,那就是全部
            let result = [];
            for(let i = 0; i < count && i+1 <= history.length; i++) result.push( history[i] ); // 将history前count个放在result数组中
            return result;
        }

    }
    // 加分、“加分(取分)”
    class DataWeightScorer {
        static ITEM_WEIGHT_CACHE_KEY = "ITEM_WEIGHT_CACHE_KEY";
        static defaultIdFun = (item)=> JSON.stringify(item);
        static SCORE_RECORD_ATTR_KEY = "weight";
        static select(item,idFun = DataWeightScorer.defaultIdFun) {
            let ItemWeightData = cache.get( DataWeightScorer.ITEM_WEIGHT_CACHE_KEY )??{};
            let key = idFun(item);
            ItemWeightData[key] = (ItemWeightData[key]??0) + 1
            cache.set(DataWeightScorer.ITEM_WEIGHT_CACHE_KEY,ItemWeightData)
        }
        static assign(items=[],idFun = DataWeightScorer.defaultIdFun) {
            let ItemWeightData = cache.get( DataWeightScorer.ITEM_WEIGHT_CACHE_KEY)??{};
            items.forEach(item=>{
                let key = idFun(item);
                item[DataWeightScorer.SCORE_RECORD_ATTR_KEY] = ItemWeightData[key]??0;
            })
            return items;
        }
        static sort(items=[],idFun = DataWeightScorer.defaultIdFun) {
            // 将权重赋于
            DataWeightScorer.assign(items,idFun);
            // 根据权重排序(高->低)
            return items.sort((a, b) => b[DataWeightScorer.SCORE_RECORD_ATTR_KEY] - a[DataWeightScorer.SCORE_RECORD_ATTR_KEY]);
        }
        // 获取高频前count项
        static highFrequency(count) {
            let ItemWeightData = cache.get( DataWeightScorer.ITEM_WEIGHT_CACHE_KEY )??{};
            let orderKeys = Object.keys(ItemWeightData).sort((a, b) => ItemWeightData[b] - ItemWeightData[a]);
            if(count != null) orderKeys = orderKeys.slice(0, count);
            // keys转items
            return registry.searchData.matchItemsByKeys(orderKeys);
        }
    }
    // 将多个
    function parseTis(bodyText) {
        // 提取整个tis标签的正则
        const regex = /(<\s*tis::http[^<>]+\/\s*>)/gm;
        let raw = captureRegEx(regex,bodyText);
        if(raw != null) {
            return raw.map(item=>item[1])
        }
        return null;
    }
    let USER_GITHUB_TOKEN_CACHE_KEY = "USER_GITHUB_TOKEN_CACHE_KEY";
    let GithubAPI = {
        token: cache.get(USER_GITHUB_TOKEN_CACHE_KEY),
        defaultToken: '',
        setToken(token) {
            if(token != null) this.token = token;
            if(this.token == null) {
                token = prompt("请输入您的github Token (只缓存在你的本地):")
                // 获取的内容无效
                if(token == null || token == "") return this;
                // 内容有效-设置
                cache.set(USER_GITHUB_TOKEN_CACHE_KEY,this.token = token);
            }
            return this;
        },
        clearToken() {
            cache.remove(USER_GITHUB_TOKEN_CACHE_KEY)
            this.token = null;
        },
        getToken(isRequest = false) {
            if(this.token == null && isRequest) this.setToken();
            return this.token;
        },
        baseRequest(type,url,{query,body}={},header = {}) {
            query = {...query}
            return request(type, url, { query,body,header });
        },
        getUserInfo() {
            return this.baseRequest("GET","https://api.github.com/user")
        },
        commitIssues(body) {
            const header = {Authorization:`Bearer ${this.token}`};
            return this.baseRequest("POST","https://api.github.com/repos/My-Search/TisHub/issues",{ body, header })
        },
        // get issues不要加 Authorization 头,可能会出现401
        getTisForIssues({keyword,state} = {}) {
            let query = null;
            if(state != null) query = {state};
            let token = this.token;
            if(token == null) token = this.defaultToken;
            return keyword
                ? new Promise((resolve,reject)=>{
                     // API兼容处理
                     this.baseRequest("GET",`https://api.github.com/search/issues?q=repo:My-Search/TisHub+state:${state}+in:title+${keyword}`)
                     .then(response=>resolve(response.items)).catch(error=>resolve([]));
                  })
                : this.baseRequest("GET","https://api.github.com/repos/My-Search/TisHub/issues",{query})
        }
    }

    // 从订阅标签中提取订阅链接
    let TisHub = {
        // 将第一个tis集与第二个tis集合并
        tisFilter(source,filterList) {
            if(typeof source == "string") source = parseTis(source);
            if(typeof filterList == "string") filterList = parseTis(filterList);
            for(let filterItem of filterList) {
                let pageTextHandler = new PageTextHandleChains(filterItem);
                let tabMetaInfos = pageTextHandler.parseAllDesignatedSingTags("tis");
                let subscribedLink = null;
                // 一个filterItem解析出元信息返回tabMetaInfos只能有一个元素,如果是多个,只取一个
                if(tabMetaInfos != null && tabMetaInfos.length > 0 ) subscribedLink = tabMetaInfos[0].tabValue;
                // 如果取不出来,说明tis无效,断言不需要解析filterItem就是subscribedLink
                if(subscribedLink == null) subscribedLink = filterItem;
                // subscribedLink 这里是filterItem用于filter source的元素实体,下面开始过滤
                source = source.filter(resultSubscribed=>! resultSubscribed.includes(subscribedLink));
            }
            return source;
        },
        getTisHubAllTis(filterList = []) {
            return new Promise((resolve,reject)=>{
                let openIssuesTisPromise = this.getOpenIssuesTis();
                let result = [];
                return Promise.all([ this.getOpenIssuesTis(), this.getClosedIssuesTis() ]).then(values=>{
                    for(let value of values) {
                        if(value == null ) continue;
                        for(let tisListObj of value) {
                            if(tisListObj != null ) result.push(...tisListObj.tisList)
                        }
                    }
                    // 过滤并提交结果
                    resolve(this.tisFilter(result,filterList));
                })
            })
        },
        // {keyword,state} .其中state {open, closed, all}
        getTisForIssues(params = {}) {
            return new Promise((resolve,reject)=>{
                GithubAPI.getTisForIssues(params).then(response=>{
                    if(response != null && Array.isArray(response)) {
                        resolve(response.map(obj=>{return {
                            owner: obj.user.login,
                            ownerProfile: obj.user.html_url,
                            title: obj.title,
                            tisList: parseTis(obj.body),
                            status: obj.state
                        }}))
                    }
                }).catch(error=>resolve([]));
            })
        },
        getOpenIssuesTis(params = {}) {
            return this.getTisForIssues({state: "open",...params});
        },
        getClosedIssuesTis(params = {}) {
            return this.getTisForIssues({state: "closed",...params});
        },
        tisListToTisText(tisList) {
            let text = "";
            for(let tis of tisList) text += tis.tisList;
            return text;
        }
    }

    // 全局注册表
    let ERROR = {
        tell(info) {
            console.error("ERROR " + info)
        }
    }
    let registry = {
        view: {
            viewVisibilityController: () => { ERROR.tell("视图未初始化,但你使用了它的未初始化的注册表信息!") },
            viewDocument: null, // 视图挂载后有值
            element: null, // 存放着视图的关键元素对象 视图挂载后有值
            tis: {
                beginTis(msg) {
                    if(msg == null || msg.length === 0) return;
                    const tisDocument = document.querySelector("#my_search_box > #tis");
                    tisDocument.innerHTML = msg;
                    tisDocument.display = "block";
                    console.log("设置结束")
                    return ()=>{
                        tisDocument.innerHTML = ""; // 置空消息内容
                        tisDocument.display = "none"; // 让tis不可见
                    }
                }
            },
            setButtonVisibility: () => { ERROR.tell("按钮未初始化!") },
            titleTagHandler: {
                handlers: [],
                // 标题tag处理器
                execute: function (title) {
                    // 去掉标题内容只剩下tags
                    let arr = captureRegEx(/(\[.*\])/gm,title)
                    if(arr == null || arr[0] == null || arr[0][0] == null) return "";
                    let tagStr = arr[0][0];
                    for(let titleTagHandler of this.handlers) {
                        let result = titleTagHandler(tagStr.trim());
                        if(result != -1) return result;
                    }
                    return tagStr;
                }
            },
            viewFirstShowEventListener: [],
            viewHideEventAfterListener: [],
            // 在查看详情内容时,返回后事件
            itemDetailBackAfterEventListener: [
                // 简讯内容隐藏-确定存在触发的事件 - 脚本环境变量置空
                () => registry.script.clearMSSE()
            ],
            // 视图延时隐藏时间,避免点击右边logo,还没显示就隐藏了
            delayedHideTime: 100,
            initialized: false,
            textView: {
                cssFillPrefix(css = "", prefix = "") {
                    const cssBlocks = css.split('}');
                    let outputCSS = '';
                    for (let block of cssBlocks) {
                        let blockLines = block.split('\n');
                        let blockOutput = '';

                        for (let line of blockLines) {
                            // 判断行末是否以 `{` 或 `,` 结尾且行首不能有空格
                            if ((line.trim().endsWith('{') || line.trim().endsWith(','))
                                && !line.startsWith(' ') && !line.trim().startsWith('@')) {
                                blockOutput += `${prefix} ${line.trim()}`; // 在当前行前加上前缀
                            } else {
                                blockOutput += line; // 其他行保持原样
                            }
                            blockOutput += '\n'; // 添加换行符,用于分隔CSS内容的各行
                        }

                        if (blockOutput.trim() !== '') {
                            outputCSS += blockOutput;
                            outputCSS += '}\n'; // 只有在当前块不为空时添加闭合大括号
                        }
                    }
                    return outputCSS;
                },
                show(html,css = "",js = "") {
                    const MS_BODY_ID = "ms-page-body";
                    html = `<style type='text/css'>${this.cssFillPrefix(css,`#${registry.view.viewDocument.id} #${registry.view.element.textView.attr('id')} #${MS_BODY_ID}` )}</style>`
                        + `<div id="${MS_BODY_ID}">${html}</div>`
                        // 这里在函数内执行js是为了在同一页面未刷新下可多次执行该js不执行,也就是对变量/函数等进行隔离
                        +`<script>(()=>{ ${js} })()</script>`
                    let my_search_box = $(registry.view.viewDocument);
                    // 视图还没有初始化
                    if(my_search_box == null) return;
                    let matchResult = registry.view.element.matchResult;
                    let textView = registry.view.element.textView
                    textView.html(html);
                    /*使用code代码块样式*/
                    document.querySelectorAll('#text_show pre code').forEach((el) => {
                        // 这里没有错,发警告不用理
                        hljs.highlightElement(el);
                    });
                    matchResult.css({
                        "display": "none"
                    })
                    textView.css({
                        "display":"block"
                    })
                    waitViewRenderingComplete(() => {
                        const currentKey = registry.searchData.searchHistory.currentKeyword();
                        if(currentKey.trim().length > 3) scrollToText(currentKey,0,"#ms-page-body");
                    });
                }
            },
            // 搜索框logo控制
            logo: {
                // logo src值
                originalLogoImgSrc: null,
                // logo按钮是否按下状态
                isLogoButtonPressedRef: ref(false),
                getLogoImg: function () {
                    let viewDocument = registry.view.viewDocument;
                    if(viewDocument == null ) return null;
                    let currentLogoImg = registry.view.element.logoButton.find("img");
                    if(this.originalLogoImgSrc == null) this.originalLogoImgSrc = currentLogoImg.attr("src");
                    return currentLogoImg;
                },
                change: function (imgResource) {
                    let logoImg = this.getLogoImg();
                    if(imgResource == null || logoImg == null ) return;
                    logoImg.attr("src",imgResource)
                },
                reset: function() {
                    let logoImg = this.getLogoImg();
                    if (logoImg == null ) return;
                    logoImg.attr("src",this.originalLogoImgSrc)
                }
            },
            modeEnum: {
                UN_INIT: -2, // 未初始化
                HIDE: -1, // 隐藏
                WAIT_SEARCH: 0, // 待搜索模式
                SHOW_RESULT: 1, // 结果显示模式
                SHOW_ITEM_DETAIL: 2 // 查看项详情 (简述内容查看/脚本页)
            },
            seeNowMode() {
               if(this.viewDocument == null) return this.modeEnum.UN_INIT;
               if(this.element.textView.css('display')!== "none") return this.modeEnum.SHOW_ITEM_DETAIL;
               if(this.element.matchResult.css('display') !== "none") return this.modeEnum.SHOW_RESULT;
               return this.viewDocument.style.display !== "none" ? this.modeEnum.WAIT_SEARCH : this.modeEnum.HIDE;
            }
        },
        other: {
            UPDATE_CDNS_CACHE_KEY: "UPDATE_CDNS_CACHE_KEY"
        },
        searchData: {
            // 处理的历史
            processHistory: [],
            // 用于数据显示后,数据又更新了
            version: 0,
            // 全局搜索数据
            data: [],
            // 数据更新后有效时长
            effectiveDuration: 1000*60*60*12,
            // 数据设置到全局中的时间
            dataMountTime: null,
            clearData() {
                this.data = [];
                this.dataMountTime = null;
            },
            setData(data) {
                if(data == null || data.length == 0) return;
                this.data = data;
                this.dataMountTime = Date.now();
            },
            getData() {
                let dataPackage = cache.get(registry.searchData.SEARCH_DATA_KEY);
                if(dataPackage == null || dataPackage.data == null) return this.data;
                // 缓存信息不为空,深入判断是否使用缓存的数据
                let updateDataTime = dataPackage.expire - this.effectiveDuration;
                // 如果数据在挂载后面已经更新了,重新加载数据到全局中
                // 全局data (即this.data)与dataMountTime是同时设置的,两者理论上是必须同时有值的
                if(this.data == null || updateDataTime > this.dataMountTime) {
                    console.logout("== 数据未加载或已检查到在其它页面已重新更新数据 ==")
                    this.setData(dataPackage.data);
                }
                return this.data;
            },
            // 根据keys(由idFun决定)从data中匹配items
            matchItemsByKeys: function (keys = []) {
                let that = this;
                if(keys.length == 0) return [];
                // 有keys转items
                let items = keys.map(key=>{
                    for(let item of that.data) {
                        if(that.idFun(item) == key) return item;
                    }
                    return null;
                })
                // 对数组keys去空,注意此时keys已经是items了
                return items.filter(item => item != null);
            },
            specialKeyword: { // 特殊的keyword
                new: "<new>",
                history: "<history>",
                highFrequency: "<highFrequency>"
            },
            // 决定着数据是否要再次初始化
            isDataInitialized: false,
            // 从可自定义搜索数据中根据title与desc进行数据匹配
            findSearchDataItem: function (title = "",desc = "",matchData) {
                if(matchData == null ) matchData = this.data;
                for(let item of matchData) {
                    if( (item.title.includes(title) || title.includes(item.title) )
                       && ( item.desc.includes(desc) || desc.includes(item.desc) )) return item;
                }
                return null;
            },
            // 数组差异-获取不同的元素比较的基值
            idFun(item) { // 自定义比较
                if(item == null || !( item instanceof Object && item.title != null)) return null;
                return item.title.replace(/\[.*\]/,"").trim()+(""+item.desc).trim();
            },
            // 旧的新数据缓存KEY
            OLD_SEARCH_DATA_KEY: "OLD_SEARCH_DATAS_KEY",
            // 标签数据缓存KEY
            DATA_ITEM_TAGS_CACHE_KEY: "DATA_ITEM_TAGS_CACHE_KEY",
            // 用户维护的不关注标签列表,缓存KEY
            USER_UNFOLLOW_LIST_CACHE_KEY: "USER_UNFOLLOW_LIST_CACHE_KEY",
            // 用户安装tishub订阅的缓存
            USE_INSTALL_TISHUB_CACHE_KEY: "USE_INSTALL_TISHUB_CACHE_KEY",
            // 默认用户不关注标签
            USER_DEFAULT_UNFOLLOW: ["成人内容","Adults only"],
            // 已经清理了用户不关注的与隐藏的标签,这是用户应真正搜索的数据
            CLEANED_SEARCH_DATA_CACHE_KEY: "CLEANED_SEARCH_DATA_CACHE_KEY",
            subscribeKey: "subscribeKey",
            showSize: 15,
            isSearchAll: false,
            searchEven: {
                event:{},
                // 搜索状态,失去焦点隐藏的一要素
                isSearching:false,
                async send(search,rawKeyword) {
                    try {
                        // 标记为搜索进行中
                        this.isSearching = true;
                        for(let subscriptionRegular of Object.keys(this.event)) {
                            const regex = new RegExp(subscriptionRegular,"i"); // 将正则字符串转换为正则表达式对象
                            if(regex.test(rawKeyword) && typeof this.event[subscriptionRegular] == "function" ) {
                                return this.event[subscriptionRegular](search,rawKeyword);
                            }
                        }
                        return await search(rawKeyword);
                    }finally {
                        // 标记为搜索结束
                        this.isSearching = false;
                    }
                }
            },
            // 新数据设置的过期天数
            NEW_DATA_EXPIRE_DAY_NUM:7,
            // 搜索逻辑,可用来手动触发搜索
            triggerSearchHandle: function (keyword){
                // 获取input元素
                const inputEl = registry.view.element?.input;
                // 如果视图还没有初始化,触发搜索无效
                if(! inputEl) return;
                if(keyword == null) {
                   keyword = inputEl.val() ?? ""
                }else {
                   inputEl.val(keyword);
                }
                // 手动触发input事件
                inputEl[0].dispatchEvent(new Event("input", { bubbles: true }));
                // 维护全局搜索keyword
                this.keyword = keyword;
            },
            // 数据改变事件
            dataChangeEventListener: [],
            // 缓存被删除事件
            dataCacheRemoveEventListener:[],
            onSearch: [],
            // 新数据块处理完成事件
            // 更新搜索数据的责任链
            USDRC: getResponsibilityChain(),
            onNewDataBlockHandleAfter: [],
            // 新数据的tag
            NEW_ITEMS_TAG: "[新]",
            // 搜索的keyword
            keyword: "",
            // 搜索的附加文件
            searchForFile: {
                files: [],
                refreshFileListView() {
                    registry.view.element.files.html('')
                    for(let [index,file] of this.files.entries()) {
                        const reader = new FileReader();
                        reader.onload = function (e) {
                            // 创建 img 元素并设置 src 为文件的 Base64 数据
                            registry.view.element.files.append(`
                                        <div class="ms-input-file" key="${index}">
                                            <img src="${e.target.result}" />
                                        </div>
                                    `);
                        };
                        reader.readAsDataURL(file);
                    }
                },
                push(file) {
                    this.files.push(file);
                    this.refreshFileListView();
                },
                delete() {
                    this.files.pop();
                    this.refreshFileListView();
                },
                clear() {
                    const trashCan = this.files;
                    this.files = [];
                    this.refreshFileListView();
                    return trashCan;
                },
                start() {
                    registry.view.element.input.on('paste', event => {
                        // 希望input外层元素无法感知到paste的动作
                        event.stopPropagation();
                        // 获取粘贴事件对象中的剪贴板数据
                        const clipboardData = event.originalEvent.clipboardData || window.clipboardData;
                        if (clipboardData) {
                            const items = clipboardData.items;
                            for (let i = 0; i < items.length; i++) {
                                const file = items[i].getAsFile();
                                if (! file.type.startsWith('text/')) this.push(file);
                            }
                        }
                    });
                    registry.view.viewHideEventAfterListener.push(() => this.clear());
                }
            },
            // 持久化Key
            SEARCH_DATA_KEY: "SEARCH_DATA_KEY",
            SEARCH_NEW_ITEMS_KEY:"SEARCH_NEW_ITEMS_KEY",
            // 搜索搜索出来的数据
            searchData: [],
            pos: 0,
            clearUrlSearchTemplate(url) {
                return url.replace(/\[\[[^\[\]]*\]\]/gm,"");
            },
            faviconSources: [ // favicon来源:https://api.cxr.cool/
                // "https://favicon.yandex.net/favicon/${domain}",  淘汰原因:当获取不到favicon时不报错,而是显示空白图标
                // "https://api.cxr.cool/ico/?url=${domain}", 淘汰原因:慢
                // "https://api.vvhan.com/api/ico?url=${domain}", 淘汰原因:快,但存在很多网站的图标无法获取
                // "https://statistical-apricot-seahorse.faviconkit.com/${domain}/32", 淘汰原因:有些图标无法获取,但会根据网站生成其网站单字符图标
                // "https://favicons.fuzqing.workers.dev/api/getFavicon?url=${rootUrl}", 淘汰原因:快,但存在很多网站的图标无法获取
                // "https://tools.ly522.com/ico/favicon.php?url=${rootUrl}", 淘汰原因:废的,10个8个获取不到
                "https://api.iowen.cn/favicon/${domain}.png", // 神
                "https://api.xinac.net/icon/?url=${rootUrl}", // 可选:但有点慢
                // "https://favicon.qqsuu.cn/${rootUrl}", 淘汰原因:外国站点不行
                // "https://api.uomg.com/api/get.favicon?url=${rootUrl}", 淘汰原因:外国站点不行
                // "https://api.vvhan.com/api/ico?url=${domain}", // 淘汰原因:存在很多网站的图标无法获取
                // "https://api.15777.cn/get.php?url=${rootUrl}",// 淘汰原因:存在很多网站的图标无法获取
                "https://ico.txmulu.com/${domain}", // 可选,但有些站点不行
                // "https://api.cxr.cool/ico/?url=${domain}", // 很多网站获取不到,国外站点不行
                "${rootUrl}/favicon.ico", // 永久有效的兜底
            ],
            CACHE_FAVICON_SOURCE_KEY: "CACHE_FAVICON_SOURCE_KEY", // 可见远程源
            CACHE_FAVICON_SOURCE_TIMEOUT: 1000*60*60*4, // 4个小时重新检测一下favicon源/过期时间,只会在呼出搜索框后检查
            getFaviconAPI: (function(){
                let defaultFaviconUrlTemplate = "${rootUrl}/favicon.ico";
                let faviconUrlTemplate = defaultFaviconUrlTemplate;
                let isRemoteTemplate = false;
                // 查看是否已经检查模板
                function checkTemplateAndUpdateTemplate() {
                    let faviconSourceCache = cache.get(registry.searchData.CACHE_FAVICON_SOURCE_KEY);
                    if( !isRemoteTemplate && faviconSourceCache != null && faviconSourceCache.sourceTemplate != null ) {
                        faviconUrlTemplate = faviconSourceCache.sourceTemplate;
                        // 设置已经是远程Favicon模板
                        isRemoteTemplate = true;
                    }
                }
                return function(url,isStandby = false) {
                    checkTemplateAndUpdateTemplate();
                    let useFaviconUrlTemplate = faviconUrlTemplate;
                    // 如果是要获取备用favicon,那直接使用上面定义的faviconUrlTemplate
                    if(isStandby) useFaviconUrlTemplate = defaultFaviconUrlTemplate;
                    // 去掉资源的“可搜索”模板,才是真正的URL
                    url = registry.searchData.clearUrlSearchTemplate(url);
                    // 将资源url放到获取favicon的源模板中
                    let urlObj = parseUrl(url)
                    return useFaviconUrlTemplate.fillByObj(urlObj);
                }
            })(),
            tmpVar: null, // 用于防抖
            searchPlaceholder(target = "SELECT",placeholder,duration = 1200) {
                // 全部的输入提示
                let inputDescs = ["我的搜索"];
                // 当前应用“输入提示”
                let inputDesc = inputDescs[Math.floor(Math.random()*inputDescs.length)];
                if(target == "UPDATE") {
                    const inputEl = registry.view.element?.input;
                    // 如果视图还没挂载,无需显示
                    if(! inputEl) return;
                    if(this.tmpVar != null) {
                        clearTimeout(this.tmpVar);
                    }
                    this.tmpVar = setTimeout(()=>{
                        inputEl.attr("placeholder",this.searchPlaceholder());
                    },duration)
                    let updateResult = placeholder==null?`🔁 数据库更新到 ${this.data==null?0:this.data.length}条`:placeholder;
                    inputEl.attr("placeholder",updateResult);
                    return updateResult;
                }
                return inputDesc;

            },
            // 存储着text转pinyin的历史  registry.searchData.subSearch.isSubSearchMode
            TEXT_PINYIN_KEY: "TEXT_PINYIN_MAP",
            // 默认数据不应初始化,不然太占内存了,只用调用了toPinyin才会初始化  getGlobalTextPinyinMap()
            getGlobalTextPinyinMap: (function() {
                let textPinyinMap = null;
                return function (){
                    if(textPinyinMap != null) return textPinyinMap;
                    return (textPinyinMap = cache.jGet("TEXT_PINYIN_MAP")??{});
                }
            })(),
            subSearch: {
                searchBoundary: " : ",
                isEnteredSubSearchMode: false, // 是否已经进入子搜索模式
                // 不传参数是看当前是否为子搜索模式 , [0] 是最近一个
                isSubSearchMode(by = undefined) {
                    let byKeyword = typeof(by) === 'string'
                        ? by // by就是keyword
                        : (by === undefined ? registry.searchData.searchHistory.currentKeyword() : registry.searchData.searchHistory.history[by]); // by是index
                    return byKeyword && byKeyword.includes(this.searchBoundary);
                },
                // 获取父级(根keyword)
                getParentKeyword(keyword) {
                    // 如果没有传入使用搜索框的value
                    if(! keyword) keyword = registry.searchData.searchHistory.currentKeyword();
                    return (keyword || "").split(this.searchBoundary)[0].trim()
                },
                // 获取子搜索keyword
                getSubSearchKeyword(keyword) {
                    // 如果没有传入使用搜索框的value
                    if(! keyword) keyword = registry.searchData.searchHistory.currentKeyword();
                    let _arr = (keyword || "").split(this.searchBoundary);
                    if( _arr.length < 2 ) return undefined;
                    return _arr[1].trim();
                }

            },
            searchHistory: {
                history: [], // 新,旧...
                add(keyword) {
                    if(! keyword) return;
                    // 维护isEnteredSubSearchMode变量状态(进入与退出)
                    const searchBoundary = registry.searchData.subSearch.searchBoundary
                    if(keyword !== searchBoundary && keyword.endsWith(searchBoundary)) {
                        registry.searchData.subSearch.isEnteredSubSearchMode = true;
                        console.logout("进入了子搜索")
                    }else if(registry.searchData.subSearch.isEnteredSubSearchMode && !keyword.includes(searchBoundary)){
                        registry.searchData.subSearch.isEnteredSubSearchMode = false;
                        console.logout("退出了子搜索")
                    }
                    // 加入到历史
                    const _history = this.history;
                    _history.unshift(keyword);
                    _history.slice(10); // 不能超过10个元素
                },
                currentKeyword() {
                    return registry.view.element.input.val()
                },
                // 当前keyword "123 : 哈哈" 与 最近"123 : 嘻嘻" 则返回true,即看左边
                seeCurrentEqualsLastByRealKeyword() {
                    // 上一次真实搜索keyword === 当前真实搜索keyword
                    return registry.searchData.subSearch.getParentKeyword(this.history[0]) === registry.searchData.subSearch.getParentKeyword(this.currentKeyword());
                }
            },
            searchProTag: "[可搜索]",
            links: {
                stringifyForSearch(links) {
                   if(links == null) return ''
                   links = links.filter(link => link != null);
                   return links.reduce((acc, cur) => acc + `${cur.text}${cur.url}${cur.title}`, '');
                }
            }
        },
        script: {
            // MSSE默认值/模板
            MS_SCRIPT_ENV_TEMPLATE: {
                event: {
                    // 发送sub keyword事件
                    sendListener: [] // 脚本页监听IPush事件
                },
                // 让脚本页可使用cache,为避免冲突,加前缀于区别
                cache: {...cache, prefix: "MS_SCRIPT_CACHE:"},
                // 让脚本页面获取脚本项数据
                getSearchDB() {
                    return [...registry.searchData.getData()]
                },
                // 让脚本页面获取选择的文本
                getSelectedText,
                // 挂载markdown
                md2html,
                // 挂载http request对象
                request,
                data: {
                   get: () => [...registry.searchData.getData()],
                   matchSearch: () => console.log('matchSearch函数未赋值'),
                   distinct: (items) => removeDuplicates(items, ['title','desc'])
                }
            },
            // 当值为undefined时表示会话未开始
            SESSION_MS_SCRIPT_ENV: undefined,
            openSessionForMSSE() {
                return (this.SESSION_MS_SCRIPT_ENV = (actualWindow.MS_SCRIPT_ENV = this.MS_SCRIPT_ENV_TEMPLATE));
            },
            clearMSSE() {
                this.SESSION_MS_SCRIPT_ENV = undefined;
            },
            // 这个函数会在脚本页渲染完成后调用
            tryRunTextViewHandler() {
                const input = registry.view.element.input;
                const rawKeyword = input.val();
                // 如果还在显示搜索的数据项 执行失败返回false
                if(registry.view.seeNowMode() === registry.view.modeEnum.SHOW_ITEM_DETAIL) {
                    // 当msg不为空发送msg消息到脚本
                    const subKeyword = registry.searchData.subSearch.getSubSearchKeyword()
                    if( subKeyword == undefined && registry.searchData.searchForFile.files.length === 0) return;
                    // 通知脚本回车send事件
                    const msg = subKeyword;
                    // actualWindow是页面真实windows对象
                    this.SESSION_MS_SCRIPT_ENV.event.sendListener.forEach(listener=>listener(msg,registry.searchData.searchForFile.clear()))
                    // clear : 清理掉send msg内容
                    input.val(rawKeyword.replace(msg,""))
                    return true;
                }
                return false;
            }
        }
    }
    let dao = {}

    // registry.registry.viewDocument
    // 页面文本选择器
    function getSelectedText(tis = '请选择页面文本') {
        function createTipElement() {
            const tipElement = document.createElement('p');
            tipElement.textContent = tis;
            // 设置行内样式
            tipElement.style.position = 'fixed';
            tipElement.style.top = '0';
            tipElement.style.left = '50%';
            tipElement.style.transform = 'translateX(-50%)';
            tipElement.style.backgroundColor = 'black';
            tipElement.style.color = 'white';
            tipElement.style.padding = '10px 20px';
            tipElement.style.fontSize = '16px';
            tipElement.style.zIndex = '9999';
            tipElement.style.borderRadius = '5px';
            document.body.appendChild(tipElement);
            return tipElement;
        }
        const view = registry.view.viewDocument;
        if(view == null) throw new Error('调用页面文本选择器异常,原因:视图隐藏不允许!')
        let tipElement = createTipElement();
        return new Promise((resolve) => {
            // 先隐藏搜索视图
            view.style.display = "none";
            // 监听鼠标抬起事件(用户结束选择)
            const onMouseUp = () => {
                const selectedText = window.getSelection().toString().trim();
                // 确保用户已选择文本
                if (selectedText) {
                    tipElement?.remove();
                    view.style.display = "block";
                    resolve(selectedText);
                    document.removeEventListener('mouseup', onMouseUp);
                }
            };

            // 监听鼠标抬起事件
            document.addEventListener('mouseup', onMouseUp);
        });
    }
    // 网页脚本自动执行函数
    let autoRunStringScript = {
        cacheKey : "autoRunStringScriptKey",
        getData() {
            let scripts = cache.get(this.cacheKey)??{};
            let keys = Object.keys(scripts);
            for(let key of keys) {
                let time = scripts[key].timeout;
                if(Date.now() > time) delete scripts[key];
            }
            cache.set(this.cacheKey,scripts);
            return scripts;
        },
        add(target,funStr,effectiveTime = 5000) {
            if(target == null || ! target.trim().startsWith("http")) return;
            let data = this.getData();
            data[target.trim()] = {
                timeout: Date.now()+effectiveTime,
                handle: funStr
            }
            cache.set(this.cacheKey,data);
        },
        run() {
            let currentPageUrl = document.URL;
            let data = this.getData();
            let keys = Object.keys(data);
            let targetObj = null;
            for(let key of keys) {
                if(key.startsWith(currentPageUrl) || currentPageUrl.startsWith(key)) targetObj = data[key];
            }
            if(targetObj != null) {
                // 从data中失去,再执行
                delete data[currentPageUrl];
                let handle = targetObj.handle;
                if(handle == null) return;
                new Function('$',handle)($);
            }
        }
    }
    // 页面加载执行
    autoRunStringScript.run();
    // 添加页面模拟脚本
    function addPageSimulatorScript(url,scriptStr) {
        scriptStr = `function exector(handle) {
            function selector(select, all = false) {
                return all ? document.querySelectorAll(select) : document.querySelector(select);
            }
            function clicker(select) {
                let element = selector(select);
                if (element != null) element.click();
            }
            function scroller(selector = 'body', topOffset = null, timeConsuming = 2000) {
                return new Promise((resolove,reject)=>{
                   var containerElement = $(selector);
                   if (containerElement.length > 0) {
                       if (topOffset !== null) {
                           $('html, body').animate({
                               scrollTop: containerElement.offset().top + topOffset
                           }, timeConsuming);
                       } else {
                           $('html, body').animate({
                               scrollTop: containerElement.offset().top
                           }, timeConsuming);
                       }
                   } else {
                       console.error('找不到指定的元素');
                   }
                   setTimeout(()=>{resolove(true)},timeConsuming)
                })
            }
            function annotator(select, styleStr = "border:5px solid red;") {
                let element = selector(select);
                if (element == null) return;
                element.style = styleStr;

            }


            handle({
                click: clicker,
                roll: scroller,
                dimension: annotator
            });
        }
        window.onload = function () {
            exector(${scriptStr})
        }`;
        autoRunStringScript.add(url,scriptStr,6000);
    }



    // 判断是否只是url且不应该是URL文本 (用于查看类型)
    function isUrl(resource) {
        // 如果为空或不是字符串,就不是url
        if(resource == null || typeof resource != "string" ) return false;
        // resource是字符串类型
        resource = resource.trim().split("#")[0];
        // 不能存在换行符,如果存在不满足
        if(resource.indexOf("\n") != -1 ) return false;
        // 被“空白符”切割后只能有一个元素
        if(resource.split(/\s+/).length != 1) return false;
        // 如果不满足url,返回false
        return isHttpUrl(resource);
    }
    /*cache.remove(registry.searchData.SEARCH_DATA_KEY);
     cache.remove(registry.searchData.SEARCH_DATA_KEY+"2");
     cache.remove(registry.searchData.SEARCH_NEW_ITEMS_KEY);
     */
    // 设置远程可用Favicon源
    let setFaviconSource = function () {
        function startTestFaviconSources(sources,pos,setFaviconUrlTemplate) {
            if(pos > sources.length - 1) return;
            console.logout(`${pos}/${sources.length-1}: 正在测试 `+sources[pos])
            checkUsability(sources[pos]).then(function(result) {
                console.logout("使用的源:"+ sources[pos])
                setFaviconUrlTemplate(result);
            }).catch(function() {
                startTestFaviconSources(sources,++pos,setFaviconUrlTemplate)
            });
        }
        let cacheFaviconSourceData = cache.get(registry.searchData.CACHE_FAVICON_SOURCE_KEY);
        let currentTime = new Date().getTime();
        let timeout = registry.searchData.CACHE_FAVICON_SOURCE_TIMEOUT;

        let faviconSources = registry.searchData.faviconSources;
        // 生成favicon源镜像
        function currentSourceArraySnapshot() {
            return JSON.stringify(faviconSources);
        }
        if(cacheFaviconSourceData == null || currentTime - cacheFaviconSourceData.updateTime > timeout || cacheFaviconSourceData.sourceArraySnapshot != currentSourceArraySnapshot()) {
            if(cacheFaviconSourceData != null) {
                console.logout(`==之前检查的已超时或源发现了修改,重新设置Favicon源==`);
            }
            let pos = 0;
            let promise = null;
            function setFaviconUrlTemplate(source = null) {
                console.logout("Test compled, set source! "+source)
                if(source != null) {
                    cache.set(registry.searchData.CACHE_FAVICON_SOURCE_KEY, {
                        updateTime: new Date().getTime(),
                        sourceTemplate: source,
                        sourceArraySnapshot: currentSourceArraySnapshot()
                    })
                }
            }
            // 去测试index=0的源, 当失败,会向后继续测试
            if(faviconSources.length < 1) return;
            startTestFaviconSources(faviconSources,0,setFaviconUrlTemplate);

        }else {

            console.logout(`Favicon源${(timeout - (currentTime - cacheFaviconSourceData.updateTime))/1000}s后测试`);
        }
    }
    // 判断是否要执行设置源,如果之前没有设置过的话就要设置,而不是通过事件触发
    if(cache.get(registry.searchData.CACHE_FAVICON_SOURCE_KEY) == null ) setTimeout(()=>{setFaviconSource();},2000);
    // 添加事件(视图在页面中初次显示时)
    registry.view.viewFirstShowEventListener.push(setFaviconSource);

    // 【函数库】
    // 加载样式
    function loadStyleString(css) {
        var style = document.createElement("style");
        style.type = "text/css";
        try {
            style.appendChild(document.createTextNode(css));
        } catch(ex) {
            style.styleSheet.cssText = css;
        }
        var head = document.getElementsByTagName('head')[0];
        head.appendChild(style);
        return style;
    }
    // 加载html
    function loadHtmlString(html) {
        // 创建一个新的 div 元素
        var newDiv = document.createElement("div");
        // 设置新的 div 的内容为要追加的 HTML 字符串
        newDiv.innerHTML = html;
        // 将新的 div 追加到 body 的末尾
        document.body.appendChild(newDiv);
        return newDiv;
    }
    // Div方式的Page页(比如构建配置面板视图)
    function DivPage(cssStr,htmlStr,handle) {
        let style = loadStyleString(cssStr);
        let div = loadHtmlString(htmlStr);
        function selector(select,isAll = false) {
            if(isAll) {
                return div.querySelectorAll(select);
            }else {
                return div.querySelector(select);
            }
        }
        function remove() {
            div.remove();
            style.remove();
        }
        handle(selector,remove);
    }
    // 异步函数
    function asyncExecFun(fun,time = 20) {
        setTimeout(()=>{
            fun();
        },time)
    }
    // 同步执行函数
    let syncActuator = function () {
        return (function () {
            let queue = [];
            let vote = 0;
            let timer = null;
            // 确保定时器已经在运行
            function ensureTimerRuning() {
                if (timer != null) return;
                timer = setInterval(async () => {
                    let taskItem = queue.pop();
                    if (taskItem != null) {
                        taskItem.active = true;
                        await taskItem.task;
                        // 任务执行完,消耗一票
                        vote--;
                        if (vote <= 0) {
                            clearInterval(timer);
                            timer = null;
                        }
                    }
                }, 100);
            }
            return function (handleFun, args, that) {
                // 让票加一
                vote++;
                // 确保定时器运行
                ensureTimerRuning();
                let taskItem = {
                    active: false,
                    task: null
                }
                taskItem.task = new Promise((resolve, reject) => {
                    let timer = null;
                    timer = setInterval(async () => {
                        if (taskItem.active) {
                            await resolve(handleFun.apply(that ?? window, args));
                            clearInterval(timer);
                        }
                    }, 30)
                })
                queue.unshift(taskItem)
                return taskItem.task;
            }
        })()
    }
    // 全页面“询问”函数
    function askIsExpiredByTopic(topic,validTime=10*1000) {
        let currentTime = new Date().getTime();
        let lastTime = cache.get(topic);
        let isExpired = lastTime == null || lastTime + validTime < currentTime;
        if(isExpired) {
            // 获取到资格,需要标记
            cache.set(topic,currentTime);
        }
        return isExpired;
    }
    
    // 【追加原型函数】
    // 往字符原型中添加新的方法 matchFetch
    String.prototype.matchFetch=function (regex,callback) {
        let str = this;
        // Alternative syntax using RegExp constructor
        // const regex = new RegExp('\\[\\[[^\\[\\]]*\\]\\]', 'gm')
        let m;
        let length = 0;
        while ((m = regex.exec(str)) !== null) {
            // 这对于避免零宽度匹配的无限循环是必要的
            if (m.index === regex.lastIndex) {
                regex.lastIndex++;
            }

            // 结果可以通过`m变量`访问。
            m.forEach((match, groupIndex) => {
                length++;
                callback(match, groupIndex);
            });
        }
        return length;
    };
    // 往字符原型中添加新的方法 matchFetch
    String.prototype.fillByObj=function (obj) {
        if(obj == null ) return null;
        let template = this;
        let resultUrl = template;
        for(let key of Object.keys(obj)) {
            let regexStr = `\\$\\s*?{[^{}]*${key}[^{}]*}`;
            resultUrl = resultUrl.replace(new RegExp(regexStr),obj[key]);
        }
        if(/\$.*?{.*?}/.test(resultUrl)) return null;
        return resultUrl;
    }
    // 比较两个数组是否相等(顺序不相同不影响)
    function isArraysEqual (arr1,arr2) {
        if( arr2 == null || arr1.length != arr2.length ) return false;
        for(let arr1Item of arr1) {
            let f = false;
            for(let arr2Item of arr2) {
                if(arr1Item == arr2Item ) {
                    f = true;
                    break;
                }
            }
            if(! f) return false;
        }
        return true;
    }

    function compareArrayDiff (arr1, arr2, idFun = () => null,diffRange = 3) { // diffRange值:“1”是左边多的,“2”是右边数组多的,3是左右合并,0是相同的部分,30是两个数组去重的
        function hashString(obj) {
            let str = JSON.stringify(obj);
            let hash = 0;
            [...str].forEach((char) => {
                hash += char.charCodeAt(0);
            });
            return "" + hash;
        }
        if (arr2 == null || arr2.length == 0) return arr1;
        // arr1与arr2都为数组对象
        // 将arr1生成模板
        let template = {};
        for (let item of arr1) {
            let itemHash = hashString(idFun(item) ?? item);

            if (template[itemHash] == null) template[itemHash] = [];
            template[itemHash].push(item);
        }
        let leftDiff = [];
        let rightDiff = [];
        let overlap = [];
        // arr2根据arr1的模板进行比对
        for (let item of arr2) {
            let itemHash = hashString(idFun(item) ?? item);
            let hitArr = template[itemHash];
            let item2Json = idFun(item) ?? JSON.stringify(item);
            if (hitArr != null) {
                // 模板中存在
                for (let hitIndex in hitArr) {
                    let hashItem = hitArr[hitIndex];
                    // 判断冲突是否真的相同
                    let item1Json = idFun(hashItem) ?? JSON.stringify(hashItem);
                    if (item1Json == item2Json) {
                        // 命中-将arr1命中的删除
                        delete hitArr.splice(hitIndex, 1);
                        overlap.push( {...item, ...hashItem} );
                        break;
                    }
                }
            } else {
                // 模板不存在,是差异项
                rightDiff.push(item);
            }
        }
        // 将模板中未命中的收集
        for (let templateKey in template) {
            let templateValue = template[templateKey]; //templateValue 是数组
            if (templateValue == null || !(templateValue instanceof Array)) continue;
            for (let templateValueItem of templateValue) {
                leftDiff.push(templateValueItem);
            }
        }
        // 根据参数,返回指定的数据
        switch (diffRange) {
            case 0:
                return overlap;
                break;
            case 1:
                return leftDiff;
                break;
            case 2:
                return rightDiff;
                break;
            case 3:
                return [...leftDiff, ...rightDiff];
                break;
            case 30:
                return [...leftDiff, ...rightDiff, ...overlap];
        }
    }
    // 保证replaceAll方法替换后也可以正常
    String.prototype.toReplaceAll = function(str1,str2) {
        return this.split(str1).join(str2);
    }
    // 向原型中添加方法:文字转拼音
    String.prototype.toPinyin = function (isOnlyFomCacheFind= false,options = { toneType: 'none', type: 'array' }) {
        let textPinyinMap = registry.searchData.getGlobalTextPinyinMap();
        // 查看字典中是否存在
        if(textPinyinMap[this] != null) {
            // console.logout("命中了")
            return textPinyinMap[this];
        }
        // 如果 isOnlyFomCacheFind = true,那返回原数据
        if(isOnlyFomCacheFind) return null;

        // console.logout("字典没有,将进行转拼音",Object.keys(textPinyinMap).length)
        let {pinyin} = pinyinPro;
        let text = this;
        let space = "<Space>"
        let spaceChar = " ";
        text = text.toReplaceAll(spaceChar,space)
        let pinyinArr = pinyin(text,options);
        // 保存到全局字典对象 ( 会话级别 )
        textPinyinMap[this] = pinyinArr.join("").toReplaceAll(space,spaceChar).toUpperCase();
        return textPinyinMap[this];
    }
    // 加载全局样式
    loadStyleString(`
/*搜索视图样式*/
#searchBox {
  height: 45px;
  background: #ffffff;
  padding: 0 10px;
  box-sizing: border-box;
  z-index: 10001;
  position: relative;
  display: flex;
  align-items: center;
  flex-wrap: nowrap;
}
#searchBox #ms-input-files {
  display: flex;
  flex-wrap: nowrap;
  align-items: center;
  height: 100%;
}
#ms-input-files .ms-input-file {
  height: 70%;
  display: flex;
  align-items: center;
  margin-right: 2px;
}
#ms-input-files .ms-input-file img {
  height: 100%;
  box-sizing: border-box;
  padding: 3px;
  background: #d3d3d3;
}

#my_search_input {
  text-align: left;
  width: 100%;
  height: 100%;
  border: none;
  outline: none;
  font-size: 15px;
  background: #fff;
  padding: 0px;
  box-sizing: border-box;
  color: rgba(0, 0, 0, .87);
  font-weight: 400;
  margin: 0px;
}

#matchResult {
  display: none;
}

#matchResult > ol {
  margin: 0px;
  padding: 0px 15px 5px;
}

#text_show {
  display: none;
  width: 100%;
  box-sizing: border-box;
  padding: 5px 10px 7px;
  font-size: 15px;
  line-height: 25px;
  max-height: 450px;
  overflow: auto;
  text-align: left;
  color: #000000;
  user-select: text !important; /* 允许用户选中复制 */
}
#text_show img {
   width: 100%;
}
#text_show .copy-btn {
   position: absolute;
   top: 8px;
   right: 8px;
   background: #7f7f7fa1;
   color: white;
   border: none;
   padding: 3px 10px;
   font-size: 12px;
   cursor: pointer;
   border-radius: 3px;
   opacity: 0.8;
   transition: opacity 0.3s;
}

#text_show .copy-btn:hover {
  opacity: 1;
}

/*定义字体*/
@font-face {
    font-family: 'HarmonyOS';
    src: url('https://s1.hdslb.com/bfs/static/jinkela/long/font/HarmonyOS_Medium.a1.woff2');
}
#my_search_view {
   font-family: 'HarmonyOS', sans-serif !important;
}
.searchItem {
	background-image: url();
    background-size: 100% 100%;
	background-clip: content-box;
	background-origin: content-box;
}

#my_search_input {
	animation-duration: 1s;
	animation-name: my_search_view;
    outline: none;
    border: none;
    box-shadow: none;
}
#my_search_input:focus{
    outline: none;
    border: none;
    box-shadow: none;
}

.resultItem {
	animation-duration: 0.5s;
	animation-name: resultItem;
}
.resultItem  .enter_main_link{
    display: flex !important ;
    justify-content: start;
    align-items: center;
    flex-grow:3;
}
/*关联图标样式*/
.resultItem .vassal {
    /*对下面的svg位置进行调整*/
    display: flex !important;
    align-items: center;
    flex-shrink:0;
    margin-right:2px;
}
.related-links {
    margin: 0 3px;
    display: flex;
    gap: 4px; /* 增加链接之间的间距 */
}

.related-links > a {
    white-space: nowrap;
    line-height: 16px;
    font-size: 12px;
    padding: 3px 10px;
    background: #f0f4ff;
    color: #3578FE;
    text-decoration: none;
    transition: all 0.3s ease;
}

.related-links > a:hover {
    background: #e6f0ff; /* 悬停时背景颜色变浅 */
    border-color: #cce1ff; /* 改变边框颜色 */
    color: #255ec8; /* 深蓝色字体加深 */
    box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); /* 增加阴影效果 */
}

.resultItem svg{
   width: 16px;
   height:16px;
}
@-webkit-keyframes my_search_view {

	0% {
		width: 0px;
	}

	50% {
		width: 50%;
	}

	100% {
		width: 100%;
	}
}

@-webkit-keyframes resultItem {

	0% {
		opacity: 0;
	}

	40% {
		opacity: 0.6;
	}

	50% {
		opacity: 0.7;
	}

	60% {
		opacity: 0.8;
	}

	100% {
		opacity: 1;
	}
}

/*简述超链接样式*/
#text_show a {
	color: #1a0dab !important;
    text-decoration:none;
}
/*自定义markdown的html样式*/
#text_show>p>code {
    padding: 2px 0.4em;
    font-size: 95%;
    background-color: rgba(188, 188, 188, 0.2);
    border-radius: 5px;
    line-height: normal;
    font-family: SFMono-Regular,Consolas,'Liberation Mono',Menlo,monospace;
    color: #558eda;
}
#my_search_input::placeholder {
  color: #757575;
}
/*简述文本颜色为统一*/
#text_show p {
	color: #202122;
}
/*让简述内容的li标签不受页面样式影响*/
#text_show > ul > li {
   list-style-type: disc !important;
}
#text_show > ul > li > ul > li {
   list-style-type: circle !important;
}
#text_show > ol > li {
   list-style-type: decimal !important;
}
/*当视图大于等于1400.1px时*/
@media (min-width: 1400.1px) {
  #my_search_box {
    left: 24%;
    right:24%;
  }
}
/*当视图小于等于1400px时*/
@media (max-width: 1400px) {
  #my_search_box {
    left: 20%;
    right:20%;
  }
}
/*当视图小于等于1200px时*/
@media (max-width: 1200px) {
  #my_search_box {
    left: 15%;
    right:15%;
  }
}
/*当视图小于等于800px时*/
@media (max-width: 800px) {
  #my_search_box {
    left: 10%;
    right:10%;
  }
}
/*输入框右边按钮*/
#logoButton {
    position: absolute;
    font-size: 12px;
    right: 5px;
    padding: 0px;
    border: none;
    display: block;
    background: rgba(255, 255, 255, 0);
    margin: 0px 7px 0px 0px;
    cursor: pointer !important;
    outline: none;
}
#logoButton:active {
  opacity: 0.4;
}
#logoButton img {
   display: block;
   width: 25px;
}

/*代码颜色*/
#text_show code,#text_show pre{
   color:#5f6368;
}


/* 滚动条整体宽度 */
#text_show::-webkit-scrollbar,
#text_show pre::-webkit-scrollbar {
  -webkit-appearance: none;
  width: 5px;
  height: 5px;
}

/* 滚动条滑槽样式 */
#text_show::-webkit-scrollbar-track,
#text_show pre::-webkit-scrollbar-track {
  background-color: #f1f1f1;
}

/* 滚动条样式 */
#text_show::-webkit-scrollbar-thumb,
#text_show pre::-webkit-scrollbar-thumb {
 background-color: #c1c1c1;
}

#text_show::-webkit-scrollbar-thumb:hover,
#text_show pre::-webkit-scrollbar-thumb:hover {
  background-color: #a8a8a8;
}

#text_show::-webkit-scrollbar-thumb:active,
#text_show pre::-webkit-scrollbar-thumb:active {
  background-color: #a8a8a8;
}
/*结果项样式*/
#matchResult li {
   line-height: 30.2px;
   height: 30.2px;
   color: #0088cc;
   list-style: none;
   width: 100%;
   padding: 0.5px;
   display: flex;
   justify-content: space-between;
   align-items: center;
   margin: 0.5px 0 !important;
}

#matchResult li > a {
  display: inline-block;
  font-size: 15.5px;
  text-decoration: none;
  text-align: left;
  cursor: pointer;
  font-weight: 400;
  background: rgb(255 255 255 / 0%);
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
  margin: 0 3px;
}
#matchResult .flag {
  color: #fff;
  height: 21px;
  line-height: 21px;
  font-size: 10px;
  padding: 0px 6px;
  border-radius: 5px;
  font-weight: 600;
  box-sizing: border-box;
  margin-right: 3.5px;
  box-shadow: rgba(0, 0, 0, 0.3) 0px 0px 0.5px;
}
#matchResult .item_title {
  color: #1a0dab;
}
#matchResult .obsolete {
  text-decoration:line-through;
  color:#a8a8a8;
}
#matchResult .item_desc {
  color: #474747;
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
}

#matchResult img {
    display: inline-block;
    width: 24px;
    height: 24px;
    margin: 0 6px 0 3px;
    box-shadow: 0 0 2px rgba(0, 0, 0, 0.5);
    border-radius: 30%;
    box-sizing: border-box;
    padding: 3px;
    flex-shrink: 0;   /* 当容量不够时,不压缩图片的大小 */
}
#my_search_box {
    position: fixed;top:50px;
    border:2px solid #cecece;z-index:2147383656;
    background: #ffffff;
}
#my_search_box > #tis {
    position: absolute;
    left: 5px;
    top: -20px;
    font-size: 12px;
    color: #d5a436;
    font-weight: bold;
}

    `)


    //防抖函数模板
    function debounce(fun, wait) {
        let timer = null;
        return function (...args) {
            // 清除原来的定时器
            if (timer) clearTimeout(timer)
            // 开启一个新的定时器
            timer = setTimeout(() => {
                fun.apply(this, args)
            }, wait)
        }
    }
    // 判断是否为指定指令
    function isInstructions(cmd) {
        let searchInputDocument = registry.view.element.input;
        if(searchInputDocument == null) return false;
        let regexString = "^\\s*:" + cmd + "\\s*$";
        let regex = new RegExp(regexString,"i");
        return regex.test(searchInputDocument.val());
    }



    // 获取一个同步执行器实例
    let pinyinActuator = syncActuator();
    // 向数据项中加入拼音项 如:title加了titlePinyin, desc加了descPinyin
    function genDataItemPinyin(threadHandleItems){
        let textPinyinMap = registry.searchData.getGlobalTextPinyinMap();
        // console.logout("分配的预热item:",threadHandleItems)
        pinyinActuator(()=>{
            if(threadHandleItems.length < 1) return;
            for(let item of threadHandleItems) {
                // 查看字典是否存在,只有没有预热过再预热
                if( textPinyinMap[threadHandleItems.title] != null ) continue;
                item.title.toPinyin();
                item.desc.toPinyin();
            }
            // 持久化-textPinyinMap字典 (这里需要判断是否值已经被初始化)
            if(textPinyinMap != null ) {
                cache.jSet(registry.searchData.TEXT_PINYIN_KEY,textPinyinMap);
            }
        });
    }
    // 当页面加载完成时触发-转拼音库操作
    const refresh = debounce(()=>{
        console.logout("==pinyin word==")
        let threadHandleItemSize = 100;
        let threadHandleItems = [];
        let currentSize = 0;
        let data = registry.searchData.getData();
        for(let item of data) {
            // 加入处理容器中
            threadHandleItems.push(item);
            currentSize++;
            // 判断是否已满
            if(currentSize >= threadHandleItemSize || data[data.length-1] == item ) {
                // 已满-去操作
                genDataItemPinyin(threadHandleItems);
                // 重置数据
                currentSize = 0;
                threadHandleItems = [];
            }
        }
    }, 2000)
    registry.searchData.dataChangeEventListener.push(refresh);

    // 实现模块一:使用快捷键触发指定事件
    function triggerAndEvent(goKeys = "ctrl+alt+s", fun, isKeyCode = false) {
        // 监听键盘按下事件

        let handle = function (event) {
            let isCtrl = goKeys.indexOf("ctrl") >= 0;
            let isAlt = goKeys.indexOf("alt") >= 0;
            let lastKey = goKeys.replace("alt", "").replace("ctrl", "").replace(/\++/gm,"").trim();
            // 判断 Ctrl+S
            if (event.ctrlKey != isCtrl || event.altKey != isAlt) return;
            if (!isKeyCode) {
                // 查看 lastKey == 按下的key
                if (lastKey.toUpperCase() == event.key.toUpperCase()) fun();
            } else {
                // 查看 lastKey == event.keyCode
                if (lastKey == event.keyCode) fun();
            }

        }
        // 如果使用 document.onkeydown 这种,只能有一个监听者
        $(document).keyup(handle);
    }
    function codeCopyMount(elementSelector) {
        document.querySelectorAll(`${elementSelector} pre code`).forEach((codeBlock) => {
            // 创建复制按钮
            const copyButton = document.createElement("button");
            copyButton.innerText = "复制";
            copyButton.classList.add("copy-btn");

            // 复制代码逻辑
            copyButton.addEventListener("click", () => {
                const text = codeBlock.innerText || codeBlock.textContent;
                navigator.clipboard.writeText(text).then(() => {
                    copyButton.innerText = "已复制";
                    setTimeout(() => (copyButton.innerText = "复制"), 2000);
                }).catch(err => {
                    console.error("复制失败:", err);
                });
            });

            // 让 <pre> 相对定位,以便按钮放在右上角
            const pre = codeBlock.parentElement;
            pre.style.position = "relative";

            // 添加按钮到 <pre> 容器
            pre.appendChild(copyButton);
        });
    }

    // 【数据初始化】
    // 获取存在的订阅信息
    function getSubscribe() {
        // 查看是否有订阅信息
        let subscribeKey = registry.searchData.subscribeKey;
        let subscribeInfo = cache.get(subscribeKey);
        if(subscribeInfo == null ) {
            // 初始化订阅信息(初次)
            subscribeInfo = `
              <tis::https://raw.githubusercontent.com/18476305640/xiaozhuang/dev/%E6%88%91%E7%9A%84%E6%90%9C%E7%B4%A2%E8%AE%A2%E9%98%85%E6%96%87%E4%BB%B6.txt />
           `;
            cache.set(subscribeKey,subscribeInfo);
        }
        return subscribeInfo;
    }
    function editSubscribe(subscribe) {
        // 判断导入的订阅是否有效
        // 获取订阅信息(得到的值肯定不会为空)
        let pageTextHandleChainsY = new PageTextHandleChains(subscribe);
        let tisArr = pageTextHandleChainsY.parseAllDesignatedSingTags("tis");
        // 生成订阅信息存储
        let subscribeText = "\n" + pageTextHandleChainsY.rebuildTags(tisArr) + "\n";
        // 持久化
        let newSubscribeInfo = subscribeText.replace(/\n+/gm,"\n\n");
        cache.set(registry.searchData.subscribeKey,newSubscribeInfo);
        return tisArr.length;
    }
    // 存储订阅信息,当指定 sLineFetchFun 时,表示将解析“直接页”的配置,如果没有指定 sLineFetchFun 时,只解析内容
    // 在提取函数中 \n 要改写为 \\n
    function getDataSources() {
        let localDataSources = `
           <fetchFun name="mLineFetchFun">
             function(pageText) {
                  let type = "sketch"; // url   sketch
                  let lines = pageText.split("\\n");
                    let search_data_lines = []; // 扫描的搜索数据 {},{}
                    let current_build_search_item = {};
                    let appendTarget = "resource"; // resource 或 vassal
                    let current_build_search_item_resource = "";  // 主要内容
                    let current_build_search_item_vassal = ""; // 附加内容
                    let current_build_search_item_links = []; // 快捷链接列表
                    let point = 0; // 指的是上面的 current_build_search_item
                    let default_desc = "--无描述--"
                    function extractLinkInfo(str) {
                       const regex = /\\[(.*?)\\]\\((https?:\\/\\/[^\\s]+)\\s*(?:\\s+"([^"]+)?")?\\s*\\)/;
                       const match = str.match(regex);
                       if (match) {
                          return {
                             text: match[1],   // 链接文本
                             url: match[2],    // URL
                             title: match[3] || ''  // 标题,如果没有则为空字符串
                          };
                       } else {
                          return null;  // 如果没有匹配到内容,返回 null
                       }
                    }
                    function isOnlyLinkLine(str) {
                       // 按行拆分,并检查每一行
                       return !str.split('\\n').some(line => line.trim() !== '' && !line.trim().startsWith('> '));
                    }
                    function getTitleLineData(titleLine) {
                       try {
                          const regex = /^#\\s*([^((]+)(?:[((](.*)[))])?\\s*$/;
                       let matchData =  regex.exec(titleLine)
                       return {
                          title: matchData[1],
                          desc: ((matchData[2]==null || matchData[2] == "")?default_desc:matchData[2])
                       }
                       }catch(e) {
                          console.log('提取函数工作时遇到了问题:',e)
                       }
                    }
                    // 是否为空字符串,忽略空格/换行
                    function isBlank(str) {
                       const trimmedStr = str.replace(/\\s+/g, '').replace(/[\\n\\r]+/g, '');
                       return trimmedStr === '';
                    }
                    for (let i = 0; i < lines.length; i++) {
                        let line = lines[i];
                        if(line.indexOf("# ") == 0) {
                           // 当前新的开始工作
                           point++;
                           // 创建新的搜索项目容器
                           current_build_search_item = {...getTitleLineData(line)}
                           // 重置resource
                           current_build_search_item_resource = "";
                           continue;
                        }
                        // 如果是刚开始,没有标题的内容行,跳过
                        if(point == 0) continue;
                        // 判断是否开始为附加内容
                        if(/^\s*-{3,}\s*$/gm.test(line)) {
                           appendTarget = "vassal"
                           // 分割行不添加
                           continue
                        }

                        // 向当前搜索项目容器追加当前行
                        if(appendTarget == "resource") {
                           current_build_search_item_resource += (line+"\\n");
                        }else {
                           // 判断当前行是否为特殊行-快捷link行
                           if(isOnlyLinkLine(current_build_search_item_vassal) && (line.trim().length > 0 && isOnlyLinkLine(line)) ) {
                              current_build_search_item_links.push(extractLinkInfo(line))
                           }else {
                              current_build_search_item_vassal += (line+"\\n");
                           }
                        }

                        // 如果是最后一行,打包
                        let nextLine = lines[i+1];
                        if(i === lines.length-1 || ( nextLine != null && nextLine.indexOf("# ") == 0 )) {
                           // 加入resource,最后一项
                           current_build_search_item.resource = current_build_search_item_resource;
                           if(! isBlank(current_build_search_item_vassal)) {
                              current_build_search_item.vassal = current_build_search_item_vassal;
                           }
                           if(current_build_search_item_links.length > 0) {
                              current_build_search_item.links = current_build_search_item_links;
                           }
                           // 打包装箱
                           search_data_lines.push(current_build_search_item);
                           // 重置资源
                           appendTarget = "resource"
                           current_build_search_item_resource = "";
                           current_build_search_item_vassal = "";
                           current_build_search_item_links = [];
                        }
                    }
                    // 添加种类
                    for(let line of search_data_lines) {
                       line.type = type;
                    }
                    return search_data_lines;
             }
           </fetchFun>
           <fetchFun name="sLineFetchFun">
             function(pageText) {
                  let type = "url"; // url   sketch
                  let lines = pageText.split("\\n");
                    let search_data_lines = []
                    for (let line of lines) {

                        let search_data_line = (function(line) {
                const baseReg = /([^::\\n(())]+)[((]([^()()]*)[))]\\s*[::]\\s*(.+)/gm;
                const ifNotDescMatchReg = /([^::]+)\\s*[::]\\s*(.*)/gm;
                let title = "";
                let desc = "";
                let resource = "";

             let captureResult = null;
             if( !(/[()()]/.test(line))) {
                 // 兼容没有描述
                 captureResult = ifNotDescMatchReg.exec(line);
                 if(captureResult == null ) return;
                 title = captureResult[1];
                 desc = "-暂无描述信息-";
                resource = captureResult[2];
             }else {
                // 正常语法
                captureResult = baseReg.exec(line);
                if(captureResult == null ) return;
                title = captureResult[1];
                desc = captureResult[2];
                resource = captureResult[3];
             }
             return {
                title: title,
                desc: desc,
                resource: resource
             };
              })(line);
                        if (search_data_line == null || search_data_line.title == null) continue;
                        search_data_lines.push(search_data_line)
                    }

                    for(let line of search_data_lines) {
                       line.type = type;
                    }
                    return search_data_lines;
             }
          </fetchFun>
        ` + getSubscribe();
        return new Promise(async (resolve,reject)=>{
            // 这里请求tishub datasources
            // [ {name: "官方订阅",body: "<tis::http... />",status: ""} ] // status: disable enable
            const installHubTisList = cache.get(registry.searchData.USE_INSTALL_TISHUB_CACHE_KEY) || [];
            const installDataSources = installHubTisList.map(installTis => `${installTis.body}`).join("\n");
            resolve(installDataSources+localDataSources);
        })
    }


    // 判断是否是github文件链接
    let githubUrlTag = "raw.githubusercontent.com";
    // cdn模板+数据=完整资源加速链接 -> 返回
    function cdnTemplateWrapForUrl(cdnTemplate,initUrl) {
        let result = parseUrl(initUrl)??{};
        if(Object.keys(result) == 0 ) return null;
        return cdnTemplate.fillByObj(result);
    }
    // github CDN加速包装器
    // 根据传入的状态,返回适合的新状态(状态中包含资源加速下载链接|原始链接|null-表示不再试)
    let cdnPack = (function () { // index = 1 用原始的(不加速链接), -2 表示原始链接打不开此时要退出

        let cdnrs = cache.get(registry.other.UPDATE_CDNS_CACHE_KEY);
        // 提供的加速模板(顺序会在后面的请求中进行重排序-请求错误反馈的使重排序)
        // protocol、domain、path、params
        let initCdnrs = []; // "https://github.moeyy.xyz/${rootUrl}${path}","https://ghproxy.net/${rootUrl}${path}"
        // 如果我们修改了最开始提供的加速模板,比如新添加/删除了一个会使用新的
        if(cdnrs == null || ! isArraysEqual(initCdnrs,cdnrs) ) {
            cdnrs = initCdnrs;
            cache.set(registry.other.UPDATE_CDNS_CACHE_KEY,initCdnrs);
        }
        return function ({index,url,initUrl}) {

            if( index <= -2 ) return null;
            // 如果已经遍历完了 或  不满足github url 不使用加速
            if(index == -1 || index > cdnrs.length -1 || (index == 0 && ! url.includes(githubUrlTag)) ) {
                url = initUrl;
                index--;
                console.logout("无法加速,将使用原链接!")
                return {index,url,initUrl};
            }
            let cdnTemplate = cdnrs[index++];
            url = cdnTemplateWrapForUrl(cdnTemplate,initUrl);
            if(index == cdnrs.length) index = -1;
            return {index,url,initUrl};
        }
    })();

    // 模块四:初始化数据源

    // 从 订阅信息(或页) 中解析出配置(json)
    function getConfigFromDataSource(pageText) {

        let config = {
            // {url、fetchFun属性}
            tis: [],
            // {name与fetchFun属性}
            fetchFuns: []
        }
        // 从config中放在返回对象中
        let pageTextHandleChainsX = new PageTextHandleChains(pageText);
        let fetchFunTabDatas = pageTextHandleChainsX.parseDoubleTab("fetchFun","name");
        for(let fetchFunTabData of fetchFunTabDatas) {
            config.fetchFuns.push( { name:fetchFunTabData.attrValue,fetchFun:fetchFunTabData.tabValue } )
        }
        // 获取tis
        let tisMetaInfos = pageTextHandleChainsX.parseAllDesignatedSingTags("tis");
        config.tis.push( ...tisMetaInfos )
        return config;

    }
    // 将url转为文本(url请求得到的就是文本),当下面的dataSourceUrl不是http的url时,就会直接返回,不作请求
    function urlToText(dataSourceUrl) {
        // dataSourceUrl 转text
        return new Promise(function (resolve, reject) {
            // 如果不是URL,那直接返回
            if( ! isHttpUrl(dataSourceUrl) ) return resolve(dataSourceUrl) ;
            let allCdns = cache.get(registry.other.UPDATE_CDNS_CACHE_KEY);
            function rq( cdnRequestStatus ) {
                let {index,url,initUrl} = cdnRequestStatus??{};
                // -2 表示加速链接+原始链接都不会请求成功(异常) ,null表示index状态已经是-2了还去请求返回null
                if(index == null || index < -2 ) return;
                request("GET",url,{query: {} ,config : {timeout: 5000} }).then(resolve).catch(()=>{
                    console.log("CDN失败,不加速请求!");
                    // 反馈错误,调整请求顺序,避免错误还是访问
                    // 获取请求错误的根域名
                    let { domain } = parseUrl(url);
                    // 根据根域名从模板中找出完整域名
                    let templates = allCdns.filter(item=>item.includes(domain));
                    // 反馈
                    //  if(templates.length > 0 ) {
                        // if(index > 0 || index <= cache.get(registry.other.UPDATE_CDNS_CACHE_KEY).length ) feedbackError(registry.other.UPDATE_CDNS_CACHE_KEY,templates[0]);
                    // }
                    // console.logout("反馈重调整后:",cache.get(registry.other.UPDATE_CDNS_CACHE_KEY)); // 反馈的结果只会在下次起作用
                    // 处理失败后的回调函数代码
                    rq(cdnPack({index,url,initUrl}));
                })

            }
            rq(cdnPack({index:0,url:dataSourceUrl,initUrl:dataSourceUrl}));
        });
    }
    // 下面的 dataSourceHandle 函数
    let globalFetchFun = [];
    // tis处理队列
    let waitQueue = [];
    // 缓存数据
    function cacheSearchData(newSearchData) {
        if(newSearchData == null) return;
        console.logout("触发了缓存,当前数据",registry.searchData.data)
        // 数据加载后缓存
        cache.set( registry.searchData.SEARCH_DATA_KEY,{
            data: newSearchData,
            expire: new Date().getTime() + registry.searchData.effectiveDuration
        })
    }
    // 更新历史数据
    function compareAndPushDiffToHistory(items = [],isCompared = false) {
        // 更新“旧全局数据”:searchData 追加-> oldSearchData
        let oldSearchData = cache.get(registry.searchData.OLD_SEARCH_DATA_KEY)??[];
        let newItemList = items;
        if(! isCompared && oldSearchData.length != 0) {
            // 比较后,差异项加入(取并集)
            newItemList = compareArrayDiff(items,oldSearchData,registry.searchData.idFun,1) ;
        }
        oldSearchData.push(... newItemList)
        console.log("旧数据缓存",oldSearchData)
        cache.set(registry.searchData.OLD_SEARCH_DATA_KEY,oldSearchData);
        if(! Array.isArray(newItemList)) newItemList = [];
        return newItemList;
    }
    // 防抖函数->处理新数据
    let blocks = [];
    let processingBlock = [];
    let triggerDataChageActuator = syncActuator();
    let refreshNewData = debounce(()=>{
        if(blocks.length == 0) return;
        // 倒动作
        processingBlock = blocks;
        blocks = [];
        // 将经过处理链得到的数据放到全局注册表中
        let globalSearchData = registry.searchData.getData();
        triggerDataChageActuator(()=>{
            globalSearchData.push(... registry.searchData.USDRC.trigger(processingBlock))
            // 数据版本改变
            registry.searchData.version++;
            // 更新视图显示条数
            registry.searchData.searchPlaceholder("UPDATE")
            // 触发搜索数据改变事件(做缓存等操作,观察者模式)
            for(let fun of registry.searchData.dataChangeEventListener) fun(globalSearchData);
            // 重新搜索
            registry.searchData.triggerSearchHandle();
        })
    }, 200) // 积累时间
    const triggerRefreshNewData = (block)=>{
        // 块积累
        blocks.push(...block);

        // 开始去处理
        refreshNewData();
    }
    // 转义与恢复,数据进行解析前进行转义,解析后恢复——比如文本中出现“/”,就会出现:SyntaxError: Octal escape sequences are not allowed in template strings.
    function CallBeforeParse() {
        this.obj = {
            "`":"<反引号>",
            "\\":"<转义>",
            "$": "<美元符>"
        }
        this.escape = function(text) {
            let obj = this.obj;
            for (var key in obj) {
                text = text.toReplaceAll(key,obj[key]);
            }
            return text;
        }
        this.recovery = function(text) {
            let obj = this.obj;
            for (var key in obj) {
                text = text.toReplaceAll(obj[key],key);
            }
            return text;
        }
    }
    let callBeforeParse = new CallBeforeParse();

    // recovery作用:将之前修改为 <wrapLine> 改为真正的换行符 \n
    function contentRecovery(item) {
        item.title = callBeforeParse.recovery(item.title);
        item.desc = callBeforeParse.recovery(item.desc);
        item.resource = callBeforeParse.recovery(item.resource);
        if(item.vassal != null ) item.vassal = callBeforeParse.recovery(item.vassal);
    }
    // 如果tisMetaInfo中有"default-tag"属性表示标签有这个属性,属性处理器在此
    function defaultTagHandle(item,tisMetaInfo = {}) {
        const defaultTag = tisMetaInfo['default-tag'];
        if(!defaultTag) return;
        // 假设defaultTag是 h'游戏' 那下面processedDefaultTag是 [h'游戏']
        const processedDefaultTag = `[${defaultTag}]`
        // defaultTagContent就是 游戏
        const defaultTagContent = parseTag(processedDefaultTag)[0][3];
        // 这里看item.title是否已经有 '游戏'] 或 [游戏] 如果都没有才加,也就是子数据项如果手动加就,default-tag就不会生效
        if( !parseTag(item.title).some(captureMeta => captureMeta[3] === defaultTagContent) ) {
            item.title = processedDefaultTag + item.title;
        }
    }
    // baseUrl + relativePath(文件 ./文件  ../文件)= relativePath的绝对路径
    function resolveUrl(baseUrl, relativePath) {
        // 创建一个链接对象,方便解析路径
        const base = new URL(baseUrl);
        // 处理相对路径
        const resolvedUrl = new URL(relativePath, base);
        return resolvedUrl.href;
    }
    function dataSourceHandle(resourcePageUrl,tisMetaInfo = {}, parentResourcePageUrl) { //resourcePageUrl 可以是url也可以是已经url解析出来的资源
        const tisTabFetchFunName = tisMetaInfo && tisMetaInfo.fetchFun;
        if(! registry.searchData.isDataInitialized) {
            registry.searchData.isDataInitialized = true;
            registry.searchData.processHistory = []; // 清空处理历史
            registry.searchData.clearData(); // 清理旧数据
        }
        let processHistory = registry.searchData.processHistory; // 处理过哪些链接需要记住,避免重复
        if(processHistory.includes(resourcePageUrl)) return; // 判断
        processHistory.push(resourcePageUrl); // 记录
        // 如果不根,且不是resourcePageUrl不是httpUrl,需要将resourcePageUrl(相对路径)根据parentResourcePageUrl(绝对路径)转为http url
        if( ! tisMetaInfo.root && !isHttpUrl(resourcePageUrl) ) {
            // if(parentResourcePageUrl == null) throw new Error(`订阅异常,相对路径: ${resourcePageUrl},没有父绝对路径!`);
            resourcePageUrl = resolveUrl(parentResourcePageUrl,resourcePageUrl);
        }
        urlToText(resourcePageUrl).then(text => {
            if(tisTabFetchFunName == null) {
                // --> 是配置 <--
                let data = []
                // 解析配置
                let config = getConfigFromDataSource(text);
                console.logout("解析的配置:",config)
                // 解析FetchFun:将FetchFun放到全局解析器中
                globalFetchFun.push(...config.fetchFuns);
                // 解析订阅:将tis放到处理队列中
                waitQueue.push(...config.tis);
                let tis = null;
                while((tis = waitQueue.pop()) != undefined) {
                    // tis第一个是url,第二是fetchFun
                    dataSourceHandle(tis.tabValue,tis, resourcePageUrl);
                }
            }else {
                // --> 是内容 <--
                // 解析内容
                if(tisTabFetchFunName === "") return;
                let fetchFunStr = getFetchFunGetByName(tisTabFetchFunName);

                let searchDataItems = [];
                try {
                    searchDataItems =(new Function('text', "return ( " + fetchFunStr + " )(`"+callBeforeParse.escape(text)+"`)"))();
                }catch(e) {
                    throw new Error("我的搜索 run log: 由于页面站点限制,导致数据解析失败!",e)
                }
                // 处理并push到全局数据容器中
                for(let item of searchDataItems) {
                    // 转义-恢复
                    contentRecovery(item);
                    // "default-tag"标签属性处理器
                    defaultTagHandle(item,tisMetaInfo)
                }
                // 加入到push到全局的搜索数据队列中,等待加入到全局数据容器中
                triggerRefreshNewData(searchDataItems)
            }
        })


    }
    // 根据fetchFun名返回字符串函数
    function getFetchFunGetByName(fetchFunName) {
        for(let fetchFunData of globalFetchFun) {
            if(fetchFunData.name == fetchFunName) {
                return fetchFunData.fetchFun;
            }
        }
    }
    // 检查是否已经执行初始化
    function checkIsInitializedAndSetInitialized(secondTime) {
        let key = "DATA_INIT";
        let value = cache.cookieGet(key);
        if(value != null && value != "") return true;
        cache.cookieSet(key,key,1000*secondTime);
        return false;
    }
    // 【数据初始化主函数】
    // 调用下面函数自动初始化数据,刚进来直接检查更新(如果数据已过期就更新数据)
    function dataInitFun() {
        // 从缓存中获取数据,判断是否还有效
        // cache.remove(SEARCH_DATA_KEY)
        let dataPackage = cache.get(registry.searchData.SEARCH_DATA_KEY);
        if(dataPackage != null && dataPackage.data != null) {
            // 缓存信息不为空,深入判断是否使用缓存的数据
            let dataExpireTime = dataPackage.expire;
            let currentTime = new Date().getTime();
            // 判断是否有效,有效的话放到全局容器中
            let isNotExpire = (dataExpireTime != null && dataExpireTime > currentTime && dataPackage.data != null && dataPackage.data.length > 0);
            // 如果网站比较特殊,忽略数据过期时间
            if( window.location.host.includes("github.com") ) isNotExpire = true;

            if(isNotExpire) {
                // 当视图已经初始化时-从缓存中将挂载数据挂载 (条件是视图已经初始化)
                console.logout(`视图${registry.view.initialized?'已加载':'未加载'}:数据有效期还有${parseInt((dataExpireTime - currentTime)/1000/60)} 分钟!`,dataPackage.data);
                if( registry.view.initialized ) registry.searchData.setData(dataPackage.data);
                // 如果数据状态未过期(有效)不会去请求数据
                return;
            }

        }
        // 在去网络请求获取数据前-检查是否已经执行初始化-防止多页面同时加载导致的数据重复加载
        if(! askIsExpiredByTopic("SEARCH_DATA_INIT",6*1000)) return;
        // 清理掉当前缓存数据
        cache.remove(registry.searchData.SEARCH_DATA_KEY);
        registry.searchData.clearData();
        // 重置数据初始化状态
        registry.searchData.isDataInitialized = false;
        // 持续执行
        registry.searchData.searchPlaceholder("UPDATE","🔁 数据准备更新中...",5000)
        // 内部将使用递归,解析出信息
        getDataSources().then(dataSources=>{dataSourceHandle(dataSources,{ root: true})})
    }
    // 检查数据有效性,且只有数据无效时挂载到数据
    dataInitFun();
    // 当视图第一次显示时,再执行
    registry.view.viewFirstShowEventListener.push(dataInitFun);

    // 解析标签函数-core函数
    function parseTag(title) {
        return captureRegEx(/\[\s*(([^'\]\s]*)\s*')?\s*([^'\]]*)\s*'?\s*]/gm,title);
    }
    // 解析出传入的所有项标签数据
    function parseTags(data = [],selecterFun = (_item)=>_item,tagsMap = {}) {
        let isArray = Array.isArray(data);
        let items = isArray?data:[data];
        // 解析 item.name中包含的标签
        items.forEach(function(item) {
            let captureGroups = parseTag(selecterFun(item));
            captureGroups.forEach(function(group) {
                let params = group[2]??"";
                let label = group[3];
                // 判断是否已经存在
                if(label != null && tagsMap[label] == null ) {
                    let currentHandleTagObj = tagsMap[label] = {
                        name: label,
                        status: 1, // 正常
                        //visible: params.includes("h"), // 参数中包含h字符表示可见
                        count: 1
                        //params: params
                        //items: [item]
                    }
                    // 如果传入的不是一个数组,那设置下面参数才有意义
                    if(! isArray) {
                        currentHandleTagObj.params = params;
                    }
                }else {
                    if(tagsMap[label] != null) {
                        tagsMap[label].count++;
                        //tagsMap[label].items.push(item);
                    }

                }
            })
        });
        // 这里不能是不是数组(上面的isArray)都返回tag数组,因为一项也可能有多个标签
        return Object.values(tagsMap);
    }

    let tagsMap = {}
    const parseSearchItem = function (searchData){
        console.log("==1:解析出数据标签==")
        // 将现有的所有标签提取出来
        // 解析
        let dataItemTags = parseTags(searchData,(_item=>_item.title),tagsMap);
        // 缓存
        if(dataItemTags.length > 0) {
            cache.set(registry.searchData.DATA_ITEM_TAGS_CACHE_KEY,dataItemTags)
        }
        return searchData;
    }
    // ################# 执行顺序从大到小 1000 -> 500
    registry.searchData.USDRC.add({weight:600 ,fun:parseSearchItem});
    // 解析script项的text
    function scriptTextParser(text) {
        if (text == null) return null;
        let scriptLines = text.split("\n");
        if (scriptLines != null && scriptLines.length != 0) {
            // 可以解析
            let result = {};
            let key = null;
            let value = null;
            for (let i = 0; i < scriptLines.length; i++) {
                let line = scriptLines[i];
                // 判断是否为新的变量开始
                let captureArr = captureRegEx(/^--\s*([^-\s]*)\s*--\s*$/gm, line);
                let isStartNewVar = captureArr != null && captureArr[0] != null && captureArr[0].length >= 2;
                let isLastLine = (i + 1 == scriptLines.length);
                if(isStartNewVar) {
                    // 保存前面的
                    if (key != null) result[key] = value.trim();
                    // 开始新的
                    key = captureArr[0][1];
                    value = ""; // 重置value
                }else {
                    value += ("\n" + line);
                }

                if ( isLastLine) {
                    // 保存前面的
                    if (key != null) result[key] = value.trim();
                    return result;
                }
            }
            return result;
        }

        return null;
    }
    // 将形如“aa bb” 转为 {aa:"bb"} ,并且如果是布尔类型或数值字符串转为对应的类型
    function extractVariables(varsString) {
        const lines = varsString.split('\n');
        const result = {};

        for (const line of lines) {
            const parts = line.trim().split(/\s+/);
            if (parts.length === 2) {
                const key = parts[0].trim();
                const value = parts[1].trim();

                // 检查是否为 true/false 字符串
                if (value === 'true' || value === 'false') {
                    result[key] = value === 'true'; // 转换为布尔值
                } else if (!isNaN(value)) { // 检查是否为数值字符串
                    result[key] = parseFloat(value); // 转换为数值类型
                } else {
                    result[key] = value; // 保持原始字符串
                }
            }
        }

        return result;
    }
    const parseScriptItem = function (searchData){
        console.log("==1:简述项解析出脚本项==")
        for(let item of searchData) {
            if((item == null || item.title == null) || item.type != "sketch" ) continue;
            if( /\[\s*(.*')?\s*(脚本|script)\s*'?\s*\]/.test( item.title ) ) {
                // 是脚本项
                item.type = "script";
                // 将resource解析为对象
                item.resourceObj = scriptTextParser(item.resource);
                item.resource = "--脚本项resource已解析到resourceObj--"
                // 解析脚本中的env(环境变量)
                if(item.resourceObj.env != null) {
                    item.resourceObj.env = extractVariables(item.resourceObj.env);
                    // 将提取的icon变量放到数据项根上,这样显示时,可读取作为icon
                    let customIcon = item.resourceObj.env._icon;
                    if( customIcon != null) item.icon = customIcon;
                    let vassal = item.resourceObj.vassal;
                    if(vassal != null) item.vassal = vassal;
                }
            }
        }
        return searchData;
    }
    // ################# 执行顺序从大到小 1000 -> 500
    registry.searchData.USDRC.add({weight:599 ,fun:parseScriptItem});
    // 监听缓存被清理,当被清理时,置空之前收集的标签数据
    registry.searchData.dataCacheRemoveEventListener.push(()=>{tagsMap = {}})

    const refreshTags = function (searchData){
        // 在添加前,进行额外处理添加,如给有”{keyword}“的url搜索项添加”可搜索“标签
        for(let searchItem of searchData) {
            let resource = searchItem.resource;
            let isHttpUrl =/^[^\n]*\.[^\n]*$/.test(`${resource}`.trim());
            let isSearchable = /\[\[[^\[\]]+keyword[^\[\]]+\]\]/.test(resource);
            // 判断是否为可搜索
            if( resource == null || !isHttpUrl || !isSearchable ) continue;
            if(! searchItem.title.includes(registry.searchData.searchProTag)) searchItem.title = registry.searchData.searchProTag+searchItem.title;
        }
        return searchData;
    }

    // ################# 执行顺序从大到小 1000 -> 500
    registry.searchData.USDRC.add({weight:500 ,fun:refreshTags});
    // 清理标签(参数中有h的)
    function clearHideTag(data,get = (item)=>item.title,set = (item,cleaned)=>{item.title=cleaned}) {
        let isArray = Array.isArray(data);
        let items = isArray?data:[data];
        for(let item of items) {
            let target = get(item);
            const regex = /\[\s*[^:\]]*h[^:\]]*\s*'\s*[^'\]]*\s*'\s*]/gm;
            let cleanedTarget = target.replace(regex, '');
            set(item,cleanedTarget);
        }
        return isArray?items:items[0];
    }
    // 给title清理掉“h”标签
    function clearHideTagForTitle(rawTitle) {
        const regex = /\[\s*[^:\]]*h[^:\]]*\s*'\s*[^'\]]*\s*'\s*]/gm;
        return rawTitle.replace(regex, '');
    }
    // 解析出标题中的所有标签-返回string数组
    function extractTagsAndCleanContent(inputString = "") {
        // 使用正则表达式匹配所有方括号包围的内容
        const regex = /\[.*?\]/g;
        const tags = inputString.match(regex) || [];
        // 清理掉标签的内容
        const cleanedContent = inputString.replace(regex, '').trim();
        return {
            tags: tags,
            cleaned: cleanedContent
        };
    }
    const filterSearchData = function (searchData) {
        const filterDataByUserUnfollowList = (itemsData,userUnfollowList = []) => {
            var userUnfollowMap = userUnfollowList.reduce(function(result, item) {
                result[item] = '';
                return result;
            }, {});
            // 开始过滤
            return itemsData.filter(item=>{
                let tags = parseTags(item.title);
                for(let tag of tags){
                    if(userUnfollowMap[tag.name] != null){
                        // 被过滤
                        return false;
                    }
                }
                return true;
            })
        }
        console.log("==去除用户不关注的数据项==")
        // 用户维护的取消关注标签列表
        let userUnfollowList = cache.get(registry.searchData.USER_UNFOLLOW_LIST_CACHE_KEY)?? registry.searchData.USER_DEFAULT_UNFOLLOW;
        // 利用用户维护的取消关注标签列表 过滤 搜索数据
        let filteredSearchData = filterDataByUserUnfollowList(searchData,userUnfollowList);
        // 去标签(参数h),清理每个item中title属性的tag , 下面注释掉是因为清理后置了仅在显示时不显示
        // let clearedSearchData = clearHideTag(filteredSearchData);
        return filteredSearchData;
    }
    // ############### 执行顺序从大到小 1000 -> 500
    registry.searchData.USDRC.add({weight:400 ,fun:filterSearchData});
    let isHasLaftData = true;
    const compareBlocks = function (searchData = []) {
        let oldSearchData = cache.get(registry.searchData.OLD_SEARCH_DATA_KEY)??[];
        if(isHasLaftData) isHasLaftData = oldSearchData != null && oldSearchData.length > 0;
        console.log("块数据与旧数据对比中>>")
        // 新数据加载完成-进行数据对比
        // 旧数据,也就是上一次数据,用于与本次比较,得出新添加数据
        // 当前时间戳
        let currentTime = new Date().getTime();
        // 准备一个存储新数据项的容器
        let newDataItems = compareAndPushDiffToHistory(searchData);
        // 给新添加的过期时间(新数据有效期)
        newDataItems.forEach(item=> {
            // 添加过期时间
            item.expires = (currentTime++) + ( 1000*60*60*24*registry.searchData.NEW_DATA_EXPIRE_DAY_NUM )
        });
        console.log("数据对比-新差异项:",[...newDataItems]);
        // 过滤掉新数据中带有“带注释”的项
        newDataItems = newDataItems.filter(item=> !item.title.startsWith("#"));
        // 以前的新增数据
        let oldNewItems = cache.get(registry.searchData.SEARCH_NEW_ITEMS_KEY);
        // 如果第一次加载数据,那不要这次的最新数据
        if(oldNewItems == null) {
            cache.set(registry.searchData.SEARCH_NEW_ITEMS_KEY,[]);
            return searchData;
        }
        // 如果还没有过期的,保留下来放在最新数据中
        for(let item of oldNewItems) {
            if(item != null && item.expires > currentTime) newDataItems.push(item);
        }
        console.log("数据对比-总新数据:",[...newDataItems])
        // 总新增去重 (标记 - 过滤标记的 )
        newDataItems = removeDuplicates(newDataItems,['title','desc']);
        // 当新数据项大于registry.searchData.showSize时,进行截取
        if(! isHasLaftData) {
            // 如何是第一次安装,那不应该有新数据
            newDataItems = [];
        }else if( newDataItems.length > registry.searchData.showSize ) {
            // 如果新增超过指定数量 ,进行截取头部最新
            // 先根据expires属性降序排序
            newDataItems.sort((a, b) => b.expires - a.expires);
            // 然后截取前15条记录
            newDataItems = newDataItems.slice(0, registry.searchData.showSize );
        }
        // 重新缓存“New Data”
        cache.set(registry.searchData.SEARCH_NEW_ITEMS_KEY,newDataItems);
        // 为全局数据(注册表中)的新数据添加新数据标签
        for(let nItem of newDataItems) {
            for(let cItem of searchData) {
                if(nItem.title === cItem.title && nItem.desc === cItem.desc) {
                    // 修改全局搜索数据中New Data数据添加“新数据”标签
                    if (! cItem.title.startsWith(registry.searchData.NEW_ITEMS_TAG)) {
                        cItem.title = registry.searchData.NEW_ITEMS_TAG+cItem.title;
                    }
                    break;
                }
            }
        }
        return searchData;
    }
    // ############ 使用用户操作的规则对加载出来的数据过滤:(责任链中的一块)
    registry.searchData.USDRC.add({weight:300 ,fun:compareBlocks});

    // 索引处理与缓存
    const refreshIndex = function (globalSearchData) {
        if(globalSearchData == null || globalSearchData.length == 0 ) return;
        console.log("===刷新索引===")
        // 当前最新数据,用于搜索
        let newDataItems = cache.get(registry.searchData.SEARCH_NEW_ITEMS_KEY);
        // 去重
        globalSearchData = removeDuplicates(globalSearchData,['title','desc'])
        // 将 index 给 newDataItems ,不然new中的我们选择与实际选择的不一致问题 !
        // 给全局数据创建索引
        globalSearchData.forEach((item,index)=>{item.index=index});
        // 给NEW建索引
        newDataItems.forEach(NItem=>{
            for(let CItem of globalSearchData) {
                if( CItem.title.includes(NItem.title) && NItem.desc === CItem.desc) {
                    NItem.index = CItem.index;
                    break;
                }
            }
        })
        // 重新缓存“New Data”
        cache.set(registry.searchData.SEARCH_NEW_ITEMS_KEY,newDataItems);
        // 重新缓存全局数据
        cacheSearchData(globalSearchData);
    }
    // 加入到数据改变后事件处理
    registry.searchData.dataChangeEventListener.push(refreshIndex);

    // 模块二
    registry.view.viewVisibilityController = (function() {
        // 整个视图对象
        let viewDocument = null;
        let initView = function () {
            // 初始化视图
            let view = document.createElement("div")
            view.id = "my_search_box";
            let menu_icon = "";
            const matchResultDocumentId = "matchResult", textViewDocumentId = "text_show",searchInputDocumentId = "my_search_input",matchItemsId = "matchItems",searchBoxId = "searchBox",logoButtonId = "logoButton",msInputFilesId = "ms-input-files";
            view.innerHTML = (`
             <div id="tis"></div>
             <div id="my_search_view">
                <div id="${searchBoxId}" >
                    <div id="${msInputFilesId}">
                       <!-- <div class="ms-input-file">
                          <img src="https://greasyfork.org/vite/assets/blacklogo96-CxYTSM_T.png" />
                       </div> -->
                    </div>
                    <input placeholder="${registry.searchData.searchPlaceholder()}" id="${searchInputDocumentId}" />
                    <button id="${logoButtonId}" >
                       <img src="${menu_icon}" draggable="false" />
                    </button>
                </div>
                <div id="${matchResultDocumentId}">
                    <ol id="${matchItemsId}">
                    </ol>
                </div>
                <!--加“markdown-body”是使用了github-markdown.css 样式!加在markdown文档父容器中-->
                <div id="${textViewDocumentId}" class="${cssLoad("ms-markdown-body","markdown-css",{isResourceName:true, replacePrefix: 'markdown-body'})} ${cssLoad("ms-code-body","code-css",{isResourceName: true})}" style="min-height:auto !important;">
                </div>
             </div>
         `)
            // 挂载到文档中
            document.body.appendChild(view)
            // 整个视图对象放在组件全局中/注册表中
            viewDocument = registry.view.viewDocument = view;
            // 想要追加请看下面registry.view.element是否已经包含,没有在那下面追加即可~

            // 搜索框对象
            let searchInputDocument = $(document.getElementById(searchInputDocumentId)),
            matchItems = $(document.getElementById(matchItemsId)),
            searchBox = $(document.getElementById(searchBoxId)),
            logoButton = $(document.getElementById(logoButtonId)),
            textView = $(document.getElementById(textViewDocumentId)),
            matchResult = $(document.getElementById(matchResultDocumentId)),
            msInputFiles = $(document.getElementById(msInputFilesId));
            // 将视图对象放到注册表中
            registry.view.element = {
                input: searchInputDocument,
                logoButton,
                matchItems,
                searchBox,
                matchResult,
                textView,
                files: msInputFiles
            }
            // 开启files 粘贴事件监听
            registry.searchData.searchForFile.start();
            // 菜单函数(点击输入框右边按钮时会调用)
            function onClickLogo() {
                // alert("小彩蛋:可以搜索一下“系统项”了解脚本基本使用哦~");
                // 调用手动触发搜索函数,如果已经搜索过,搜索空串(清理)
                let keyword = "[系统项]";
                registry.searchData.triggerSearchHandle(searchInputDocument.val()==keyword?'':keyword);
                // 重新聚焦搜索框
                registry.view.element.input.focus()
            }
            const isLogoButtonPressedRef = registry.view.logo.isLogoButtonPressedRef;
            // 按下按钮时设置变量为 true
            logoButton.on('mousedown', function() {
                isLogoButtonPressedRef.value = true;
            });

            // 按钮弹起时设置变量为 false,并让输入框聚焦
            logoButton.on('mouseup', function() {
                isLogoButtonPressedRef.value = false;
                onClickLogo() // 触发logo点击事件
                searchInputDocument.focus(); // 输入框聚焦
            });

            // 防止鼠标拖出按钮后弹起无法触发 mouseup
            logoButton.on('mouseleave', function() {
                if (isLogoButtonPressedRef.value) {
                    isLogoButtonPressedRef.value = false;
                    searchInputDocument.focus(); // 输入框聚焦
                }
            });

            // 设置视图已经初始化
            registry.view.initialized = true;

            // 在搜索的结果集中上下选择移动然后回车(相当点击)
            searchInputDocument.keyup(function(event){
                let keyword = $(event.target).val().trim();
                // 当不为空时,放到全局keyword中
                if(keyword) {
                    registry.searchData.keyword = event.target.value;
                }
                // 处理keyword中的":"字符
                if(keyword.endsWith("::") || keyword.endsWith("::")) {
                    keyword = keyword.replace(/::|::/,registry.searchData.subSearch.searchBoundary).replace(/\s+/," ");
                    // 每次要形成一个" : "的时候去掉重复的" : : " -> " : "
                    keyword = keyword.replace(/((\s{1,2}:)+ )/,registry.searchData.subSearch.searchBoundary);
                    $(event.target).val(keyword.toUpperCase());
                }
            });
            // searchInputDocument.keydown:这个监听用来处理其它键(非上下选择)的。
            searchInputDocument.keydown(function (event){
                // 阻止键盘事件冒泡 | 阻止输入框外监听到按钮,应只作用于该输入框
                event.stopPropagation();
                // 判断一个输入框的东西,如果如果按下的是删除,判断一下是不是"搜索模式"
                let keyword = $(event.target).val();
                let input = event.target;
                if(event.key == "Backspace" ) { // 按的是删除键-块删除
                    if(keyword.endsWith(registry.searchData.subSearch.searchBoundary)) {
                        // 取消默认事件-删除
                        event.preventDefault();
                        return;
                    }else if(/^\s*[\[<][^\[\]<>]*[\]>]\s*$/.test( keyword )) {
                        // 如果输入框只有[xxx]或<xxx>那就清空掉输入框
                        searchInputDocument.val('')
                        // keyword重置为空字符后触发搜索
                        registry.searchData.triggerSearchHandle();
                        event.preventDefault();
                        return;
                    }else if(keyword === ""){
                        registry.searchData.searchForFile.delete();
                    }
                }else if ( ! event.shiftKey && event.keyCode === 9 ) { // Tab键
                    if(! registry.searchData.subSearch.isSubSearchMode()) {
                        // 转大写
                        event.target.value = event.target.value.toUpperCase()
                        // 添加搜索pro模式分隔符
                        event.target.value += registry.searchData.subSearch.searchBoundary
                        // 阻止默认行为,避免跳转到下一个元素
                        registry.searchData.triggerSearchHandle();
                    }
                    event.preventDefault();
                }else if (event.shiftKey && event.keyCode === 9 ) { // 按下shift + tab键时取消搜索模式
                    if(registry.searchData.subSearch.isSubSearchMode()) {
                        // 在这里编写按下shift+tab键时要执行的代码
                        let input = event.target;
                        input.value = input.value.split(registry.searchData.subSearch.searchBoundary)[0]
                        event.target.value = event.target.value.toLowerCase();
                        // 手动触发输入事件
                        input.dispatchEvent(new Event("input", { bubbles: true }));
                    }
                    event.preventDefault();
                }

            })

            // searchInputDocument.keydown:这个监听用来处理上下选择范围的操作
            searchInputDocument.keydown(function (event){
                let e = event || window.event;

                if(e && e.keyCode!=38 && e.keyCode!=40 && e.keyCode!=13) return;
                if(e && e.keyCode==38){ // 上
                    registry.searchData.pos --;

                }
                if(e && e.keyCode==40){ //下
                    registry.searchData.pos ++;
                }
                // 如果是回车 && registry.searchData.pos == 0 时,设置 registry.searchData.pos = 1 (这样是为了搜索后回车相当于点击第一个)
                if(e && e.keyCode==13 && registry.searchData.pos == 0){ // 回车选择的元素
                    // 如果当前是在搜索中就忽略回车这个操作
                    if(registry.searchData.searchEven.isSearching) return;
                    registry.searchData.pos = 1;
                }

                // 当指针位置越出时,位置重定向
                if(registry.searchData.pos < 1 || registry.searchData.pos > registry.searchData.searchData.length ) {
                    if(registry.searchData.pos < 1) {
                        // 回到最后一个
                        registry.searchData.pos = registry.searchData.searchData.length;
                    }else {
                        // 回到第一个
                        registry.searchData.pos = 1;
                    }
                }
                // 设置显示样式
                let activeItem = $(registry.view.element.matchItems.find('li')[registry.searchData.pos-1]);
                // 设置活跃背景颜色
                let activeBackgroundColor = "#dee2e6";
                activeItem.css({
                    "background":activeBackgroundColor
                })

                // 设置其它子元素背景为默认统一背景
                activeItem.siblings().css({
                    "background":"#fff"
                })

                // 看是不是item detail内容显示中,如果是回车发送send事件,否则才是结果集显示的回车选择
                if(e && e.keyCode==13 && activeItem.find("a").length > 0 && !registry.script.tryRunTextViewHandler()){ // 回车
                    if (e.ctrlKey) {
                        // Ctrl + Enter 的操作
                        activeItem.find(".vassal")[0]?.click();
                    } else {
                        // 点击当前活跃的项,点击
                        activeItem.find("a")[0].click();
                    }
                }
                // 取消冒泡
                e.stopPropagation();
                // 取消默认事件
                e.preventDefault();

            });
            // 将输入框的控制按钮设置可见性函数公开放注册表中
            registry.view.setButtonVisibility = function (buttonVisibility = false) {
                // registry.view.setButtonVisibility
                logoButton.css({
                    "display": buttonVisibility?"block":"none"
                })
            }
            // 高权重项特殊搜索关键词直达
            registry.searchData.searchEven.event[registry.searchData.specialKeyword.highFrequency] = function(search,rawKeyword) {
                return DataWeightScorer.highFrequency(45);
            }
            // 历史记录特殊搜索关键词直达
            registry.searchData.searchEven.event[registry.searchData.specialKeyword.history] = function(search,rawKeyword) {
                return SelectHistoryRecorder.history(15);
            }
            // 向搜索事件(只会触发一个)中添加一个“NEW”搜索关键词
            registry.searchData.searchEven.event["new|"+registry.searchData.specialKeyword.new] = function(search,rawKeyword) {
                let showNewData = null;
                let activeSearchData = registry.searchData.getData();
                // 如果当前注册表中全局搜索数据为空,使用缓存的数据
                if(activeSearchData == null ) {
                    let cacheAllSearchData = cache.get(registry.searchData.SEARCH_DATA_KEY);
                    if(cacheAllSearchData != null && cacheAllSearchData.data != null) activeSearchData = cacheAllSearchData.data;
                }
                // 如果最新数据都没有,使用旧数据(上一次)
                if(activeSearchData == null ) {
                    let oldCacheAllSearchData = cache.get(registry.searchData.OLD_SEARCH_DATA_KEY);
                    if(oldCacheAllSearchData != null) activeSearchData = oldCacheAllSearchData;
                }
                // 只展示 newItems 数据中data也存在的项
                let newItems = cache.get(registry.searchData.SEARCH_NEW_ITEMS_KEY)??[];
                if(newItems.length > 0 && activeSearchData.length > 0) {
                    // 返回的showNewData是左边的(activeSearchData),而不是右边的(newItems),但newItems多出来 的属性也会合并到activeSearchData的item
                    showNewData = compareArrayDiff(activeSearchData,newItems,registry.searchData.idFun,0)
                }
                if(showNewData == null) return [];
                // 对数据进行排序
                showNewData.sort(function(item1, item2){return item2.expires - item1.expires});

                showNewData.map((item,index)=>{
                    let dayNumber = registry.searchData.NEW_DATA_EXPIRE_DAY_NUM;
                    // 去掉[新] 再都加[新],使得就算没有也在显示时也是有新标签的
                    item.title = registry.searchData.NEW_ITEMS_TAG+item.title.toReplaceAll(registry.searchData.NEW_ITEMS_TAG,"")
                    // 添加“几天前”
                    item.title = item.title + " | " + Math.floor( (Date.now() - (item.expires - 1000*60*60*24*dayNumber) )/(1000*60*60*24) )+"天前"; //toDateString
                    return item;
                })
                // 将最新的一条由“新”改为“最新一条”
                showNewData[0].title = showNewData[0].title.toReplaceAll(registry.searchData.NEW_ITEMS_TAG,"[最新一条]")
                return showNewData;
            }
            // 可填充搜索模式优先路由(key是正则字符串,value为字符串类型是转发,如果是函数,是自定义搜索逻辑)
            const searchableSpecialRouting = {
                "^\\s*$": "问AI",
                "^问AI$": async (search,rawKeyword,keywordForFill0)=>{
                    return await search(keywordForFill0,{isAccurateSearch : true});
                }
            }
            // 返回undfind表示没有定义匹配对应的SpecialRouting,执行通用路由 | null表示跳过 | 返回数组表示SpecialRouting执行搜索得到的结果
            const searchableSpecialRoutingHandler = async function(search,rawKeyword){
                const keywordForFill0 = registry.searchData.subSearch.getParentKeyword(rawKeyword);
                for(let key of Object.keys(searchableSpecialRouting)) {
                    if(isMatch(key,keywordForFill0)) {
                        const value = searchableSpecialRouting[key];
                        if(typeof value === "string") {
                            registry.searchData.triggerSearchHandle(value+registry.searchData.subSearch.searchBoundary)
                            return [];
                        }
                        if(typeof value === "function") return await value(search,rawKeyword,keywordForFill0);
                    }
                }
                // 表示没有匹配到SpecialRouting
                return undefined;
            }
            registry.searchData.searchEven.event[".*"+registry.searchData.subSearch.searchBoundary+".*"] = async function(search,rawKeyword) {
                const specialRoutinResult = await searchableSpecialRoutingHandler(search,rawKeyword)
                // 当没有优先Result, 只搜索“可搜索”项
                return Array.isArray(specialRoutinResult)
                    ? specialRoutinResult
                    : await search(`${registry.searchData.searchProTag} ${registry.searchData.subSearch.getParentKeyword()}`);
            }
            // 搜索AOP
            async function searchAOP(search,rawKeyword) {
                // 转发到对应的AOP处理器中(keyword规则订阅者)
                let data = registry.searchData.getData();
                console.log("搜索data:",data)
                return await registry.searchData.searchEven.send(search,rawKeyword);
            }
            function searchUnitHandler(beforeData = [],keyword = "") {
                // 触发搜索事件
                for(let e of registry.searchData.onSearch) e(keyword);
                // 如果没有搜索内容,返回空数据
                keyword = keyword.trim().toUpperCase();
                if(keyword == "" || registry.searchData.getData().length == 0 ) return [];
                // 切割搜索内容以空格隔开,得到多个 keyword
                let searchUnits = keyword.split(/\s+/);
                // 弹出一个 keyword
                keyword = searchUnits.pop();
                // 本次搜索的总数据容器
                let searchResultData = [];
                let searchLevelData = [
                    [],[],[] // 分别是匹配标题/desc/url 的结果
                ]
                // 数据出来的总数据
                //let searchData = []
                // 前置处理函数,这里使用观察者模式
                // searchPreFun(keyword);
                // 搜索操作
                // 为实现当关键词只有一位时,不使用转拼音搜索,后面搜索涉及到的转拼音操作要使用它,而不是直接调用toPinyin
                function getPinyinByKeyword(str,isOnlyFomCacheFind=false) {
                    if(registry.searchData.keyword.length > 1 ) return str.toPinyin(isOnlyFomCacheFind)??"";
                    return str??"";
                }
                let pinyinKeyword = getPinyinByKeyword(keyword);
                let searchBegin = Date.now()
                for (let dataItem of beforeData) {
                    /* 取消注释会导致虽然是15条,但有些匹配度高的依然不能匹配
                    // 如果已达到搜索要显示的条数,则不再搜索 && 已经是本次最后一次过滤了 => 就不要扫描全部数据了,只搜出15条即可
                    let currentMeetConditionItemSize = searchLevelData[0].length + searchLevelData[1].length + searchLevelData[2].length;
                    if(currentMeetConditionItemSize >= registry.searchData.showSize && searchUnits.length == 0 && registry.searchData.subSearch.isSubSearchMode() ) break;
                    */
                    // 将数据放在指定搜索层级数据上DeepSeek
                    if (
                        (( getPinyinByKeyword(dataItem.title,true).includes(pinyinKeyword) || dataItem.title.toUpperCase().includes(keyword) ) && searchLevelData[0].push(dataItem) )
                        || (( getPinyinByKeyword(dataItem.desc,true).includes(pinyinKeyword) || dataItem.desc.toUpperCase().includes(keyword)) && searchLevelData[1].push(dataItem) )
                        || ( `${registry.searchData.links.stringifyForSearch(dataItem.links)}${dataItem.resource}${dataItem.vassal}`.substring(0, 4096).toUpperCase().includes(keyword) && searchLevelData[2].push(dataItem) )
                    ) {} // 若满足条件的数据对象则会添加到对应的盒子中
                }
                let searchEnd = Date.now();
                console.logout("常规搜索主逻辑耗时:"+(searchEnd - searchBegin ) +"ms");
                // 将上面层级数据进行权重排序然后放在总容器中
                searchResultData.push(...DataWeightScorer.sort(searchLevelData[0],registry.searchData.idFun));
                searchResultData.push(...DataWeightScorer.sort(searchLevelData[1],registry.searchData.idFun));
                searchResultData.push(...DataWeightScorer.sort(searchLevelData[2],registry.searchData.idFun));

                if(searchUnits.length > 0 && searchUnits[searchUnits.length-1].trim() != registry.searchData.subSearch.searchBoundary.trim()) {
                    // 递归搜索
                    searchResultData = searchUnitHandler(searchResultData,searchUnits.join(" "));
                }
                return searchResultData;
            }
            // ==标题tag处理==
            // 1、标题tag颜色选择器
            function titleTagColorMatchHandler(tagValue) {
                let vcObj = {
                    "系统项":"background:rgb(0,210,13);",
                    "非最佳":"background:#fbbc05;",
                    "推荐":"background:#ea4335;",
                    "装机必备":"background:#9933E5;",
                    "好物":"background:rgb(247,61,3);",
                    "安卓应用":"background:#73bb56;",
                    "Adults only": "background:rgb(244,201,13);",
                    "可搜索":"background:#4c89fb;border-radius:0px !important;",
                    "新":"background:#f70000;",
                    "最新一条":"background:#f70000;",
                    "精选好课":"background:#221109;color:#fccd64 !important;"
                };
                let resultTagColor = "background:#5eb95e;";
                Object.getOwnPropertyNames(vcObj).forEach(function(key){
                    if(key == tagValue) {
                        resultTagColor = vcObj[key];
                    }
                });
                return resultTagColor;
            }
            // 2、标题内容处理程序
            function titleTagHandler(title) {
                if(!(/[\[]?/.test(title) && /[\]]?/.test(title))) return -1;
                // 格式是:[tag]title 这种的
                const regex = /(\[[^\[\]]*\])/gm;
                let m;
                let resultTitle = title;
                while ((m = regex.exec(title)) !== null) {
                    // 这对于避免零宽度匹配的无限循环是必要的
                    if (m.index === regex.lastIndex) {
                        regex.lastIndex++;
                    }
                    let tag = m[0];
                    if(tag == null || tag.length == 0) return -1;
                    let tagCore = tag.substring(1,tag.length - 1);
                    // 正确提取
                    resultTitle = resultTitle.toReplaceAll(tag,`<span style="${titleTagColorMatchHandler(tagCore)}" class="flag">${tagCore}</span>`);
                }
                return resultTitle;
            }
            // 3、添加标题处理器 titleTagHandler
            registry.view.titleTagHandler.handlers.push(titleTagHandler)
            // 字符串重叠匹配度搜索(类AI搜索)
            async function stringOverlapMatchingDegreeSearch(rawKeyword) {
                const endTis = registry.view.tis.beginTis("(;`O´)o 匹配度模式搜索中...")
                // 这里为什么要用异步,不果不会那上面设置的tis会vk得不到渲染,先保证上面已经渲染完成再执行下面函数
                return await new Promise((resolve,reject)=>{
                    waitViewRenderingComplete(() => {
                        try {
                            // 搜索逻辑开始
                            //	`registry.searchData.getData()`会被排序desc
                            // 为什么需要拷贝data,因为全局的搜索位置不能改变!!
                            const searchBegin = Date.now();
                            let searchResult = overlapMatchingDegreeForObjectArray(rawKeyword.toUpperCase(),[...registry.searchData.getData()], (item)=>{
                                const str2ScopeMap = {}
                                const { tags , cleaned } = extractTagsAndCleanContent(`${item.title}`);
                                str2ScopeMap[cleaned.toUpperCase()] = 9;
                                str2ScopeMap[`${item.desc}${tags.join()}`.toUpperCase()] = 8;
                                str2ScopeMap[`${registry.searchData.links.stringifyForSearch(item.links)}${item.resource}${item.vassal}`.substring(0, 4096).toUpperCase()] = 2;
                                return str2ScopeMap;
                            },{onlyHasScope:true});
                            const searchEnd = Date.now();
                            console.log("启动类AI搜索结果 :",searchResult)
                            console.logout("类AI搜索主逻辑耗时:"+(searchEnd - searchBegin ) +"ms");
                            resolve(searchResult)
                        }catch (e) {
                            console.error("类AI搜索异常!",e)
                            resolve([])
                        }finally {
                            endTis()
                        }
                    })
                })
            }
            // 将匹配搜索挂载给脚本应用使用
            registry.script.MS_SCRIPT_ENV_TEMPLATE.matchSearch = stringOverlapMatchingDegreeSearch;
            // 给输入框加事件
            // 执行 debounce 函数返回新函数
            let handler = async function (e) {
                // 搜索使用的数据版本
                let version = registry.searchData.version;
                let rawKeyword = e.target.value;

                // 在本次搜索加入到历史前检查(如果之前是子搜索模式且现在还是子搜索模式那就跳过搜索,因为内是子搜索内容被修改不进行搜索)
                if(registry.searchData.subSearch.isEnteredSubSearchMode && registry.searchData.subSearch.isSubSearchMode()
                  && registry.searchData.searchHistory.seeCurrentEqualsLastByRealKeyword()) return;

                // 添加到搜索历史(维护这个历史有用是为了子搜索模式的“进”-“出”)
                registry.searchData.searchHistory.add(rawKeyword)


                // 常规方式搜索(搜索逻辑入口)
                async function search(rawKeyword,{isAccurateSearch = false} = {}) {
                    let processedKeyword = rawKeyword.trim().split(/\s+/).reverse().join(" ");
                    version = registry.searchData.version;
                    // 常规搜索
                    let searchResult = searchUnitHandler(registry.searchData.getData(),processedKeyword);
                    // 如果常规搜索不到使用类AI搜索(不能是精确搜索 && 常规搜索没有结果 && 搜索keyword不为空串)
                    if(!isAccurateSearch && (searchResult == null || searchResult.length === 0) && `${rawKeyword}`.trim().length > 0 ) {
                        searchResult = await stringOverlapMatchingDegreeSearch(rawKeyword)
                    }
                    return searchResult;
                }
                // 搜索AOP或说搜索代理
                // 递归搜索,根据空字符切换出来的多个keyword
                // let searchResultData = searchUnitHandler(registry.searchData.data,key)
                let searchResultData = await searchAOP(search,rawKeyword);
                // 如果搜索的内容无效,跳过内容的显示
                if(searchResultData == null) return;
                // 放到视图上
                // 置空内容
                matchItems.html("")
                // 最多显示条数
                let show_item_number = registry.searchData.showSize ;
                function getFaviconImgHtml(searchResultItem) {
                    if(searchResultItem == null) return null;
                    let resource = searchResultItem.resource.trim();
                    let customIcon = null;
                    if(searchResultItem.icon != null) {
                        customIcon = searchResultItem.icon;
                    }else {
                        let type = searchResultItem.type;
                        // 如果不是url,那其它类型就需要自定义图标
                        let typesAndImg = {
                            "sketch":"",
                            "script":""
                        }
                        // url与sketch类型可互转,主要看resource
                        type = (type == "url" || type == "sketch")?(isUrl(resource)?"url":"sketch"):type;
                        if(type != "url") customIcon = typesAndImg[type];
                    }
                    if(customIcon != null) {
                        return `<img src="${customIcon}" />`
                    }else {
                        return `<img src="${registry.searchData.getFaviconAPI(resource)}" standbyFavicon="${registry.searchData.getFaviconAPI(resource,true)}"  class="searchItem" />`
                    }
                }

                // 标题内容处理器
                function titleContentHandler(title) {
                    // 对标题去掉所有tag
                    const { cleaned } = extractTagsAndCleanContent(title)
                    title = cleaned
                    // 如果带#将加上删除线,通过加obsolete类名方式
                    return `<span class="item_title ${title.startsWith("#")?'obsolete':''}">${title.replace(/^#/,"")}</span>`;
                }


                let matchItemsHtml = "";
                // 真正渲染到列表的数据项
                let searchData = []
                for(let searchResultItem of searchResultData ) {
                    // 限制条数
                    if(show_item_number-- <= 0 && !registry.searchData.isSearchAll) {
                        break;
                    }
                    // 显示时清理标签-虽然在加载数据时已经清理了,但这是后备方案
                    // clearHideTag(searchResultItem);
                    // 将数据放入局部容器中
                    searchData.push(searchResultItem)

                    let isSketch = !isUrl(searchResultItem.resource);//  searchResultItem.resource.trim().toUpperCase().indexOf("HTTP") != 0;
                    let vassalSvg = `<svg t="1685187993813" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="3692" width="200" height="200"><path d="M971.904 372.736L450.901333 887.338667a222.976 222.976 0 0 1-312.576 0 216.362667 216.362667 0 0 1 0-308.736l468.906667-463.232a148.736 148.736 0 0 1 208.469333 0 144.298667 144.298667 0 0 1 0 205.824L346.752 784.469333a74.325333 74.325333 0 0 1-104.192 0 72.106667 72.106667 0 0 1 0-102.912l416.853333-411.733333-52.181333-51.456-416.853333 411.733333a144.298667 144.298667 0 0 0 0 205.781334 148.650667 148.650667 0 0 0 208.426666 0l468.906667-463.146667a216.490667 216.490667 0 0 0 0-308.736 223.061333 223.061333 0 0 0-312.661333 0L60.16 552.832l1.792 1.792a288.384 288.384 0 0 0 24.277333 384.170667c106.24 104.917333 273.322667 112.768 388.906667 23.936l1.792 1.834666L1024 424.192l-52.096-51.456z" fill="#666666" p-id="3693"></path></svg>`;
                    // 构建快捷link html
                    function buildRelatedLinksHtml(links) {
                        if (links == null || links.length === 0) return '';
                        let html = `<div class="related-links">`;
                        // 遍历 links 数组,为每个链接生成对应的 <a> 标签
                        links.forEach(link => {
                            html += `<a href="${link.url}" target="_blank" title="${link.title}">${link.text}</a>`;
                        });
                        html += '</div>';
                        return html;
                    }
                    // 将符合的数据装载到视图
                    let item = `
                    <li class="resultItem">
                         <!--图标-->
                         ${getFaviconImgHtml(searchResultItem)}
                        <a href="${isSketch?'':searchResultItem.resource}" target="_blank" title="${searchResultItem.desc}" index="${searchResultItem.index}" version="${version}" class="enter_main_link">
                            <!--tag与标题-->
                            ${registry.view.titleTagHandler.execute(clearHideTagForTitle(searchResultItem.title))}${titleContentHandler(searchResultItem.title)}
                            <!--描述信息-->
                            <span class="item_desc">(${searchResultItem.desc})</span>
                        </a>
                        ${buildRelatedLinksHtml(searchResultItem.links)}
                        ${searchResultItem.vassal !=null?'<a index="'+searchResultItem.index+'" version="'+version+'" vassal="true" class="vassal" title="查看相关联/同类项内容" target="_blank">'+vassalSvg+'</a>':''}
                    </li>`
                    matchItemsHtml += item;
                }
                matchItems.html(matchItemsHtml);

                let loadErrorTagIcon = "";

                // 给刚才添加的img添加事件
                for(let imgObj of registry.view.element.matchItems.find('img')) {
                    // 加载完成事件,去除加载背景
                    imgObj.onload = function(e) {
                        $(e.target).css({
                            "background": "#fff"
                        })
                    }
                    // 加载失败,设置自定义失败的本地图片
                    imgObj.onerror = function(e,a,b,c) {
                        let currentErrorImg = $(e.target);
                        let standbyFaviconAttr = "standbyFavicon";
                        let standbyFavicon = currentErrorImg.attr(standbyFaviconAttr);
                        if(standbyFavicon != null) {
                            // 如果备用favicon使用
                            currentErrorImg.attr("src",standbyFavicon)
                            currentErrorImg.removeAttr(standbyFaviconAttr)
                        }else {
                            // 如果备用favicon直接使用加载失败图标base64
                            currentErrorImg.attr("src",loadErrorTagIcon)
                        }
                    }
                }

                // 隐藏文本显示视图
                textView.css({
                    "display":"none"
                })
                // 让搜索结果显示
                let matchResultDisplay = "block";
                if(searchResultData.length < 1) matchResultDisplay="none";
                matchResult.css({
                    "display":matchResultDisplay,
                    "overflow":"hidden"
                })
                // 将显示搜索的数据放入全局容器中
                registry.searchData.searchData = searchData;
                // 指令归位(置零)
                registry.searchData.pos = 0;
            }

            registry.view.element.matchItems.on("click","li > a",function(e) {
                let targetObj = e.target;
                // 如果当前标签是svg标签,那委托给父节点
                while ( targetObj != null && !/^(a|A)$/.test(targetObj.tagName)) {
                    targetObj = targetObj.parentNode
                }
                // 取消默认事件,全部都是手动操作
                e.preventDefault();
                // 取消冒泡
                window.event? window.event.cancelBubble = true : e.stopPropagation();
                // 设置为阅读模式
                // $("#my_search_input").val(":read");
                // 获取当前结果在搜索数组中的索引
                let dataIndex = parseInt($(targetObj).attr("index"));
                let dataVersion = parseInt($(targetObj).attr("version"));
                let currentSearchDataVersion = registry.searchData.version;
                let itemData = registry.searchData.getData()[dataIndex];
                if(itemData == null || dataVersion != currentSearchDataVersion ) {
                    console.log("后备方案(没有找到了?"+(itemData == null)+",数据版本改变了?"+(dataVersion != currentSearchDataVersion)+")")
                    // 索引出现问题-启动后备方案-全局搜索
                    let title = $(targetObj).parent().find(".item_title").text();
                    let desc = $(targetObj).parent().find(".item_desc").text();
                    // 从全局数据中根据title与desc进行匹配
                    itemData = registry.searchData.findSearchDataItem(title,desc)
                    // 从历史数据中找,根据title与desc进行匹配
                    if(itemData == null) itemData = registry.searchData.findSearchDataItem(title,desc,SelectHistoryRecorder.history)
                }
                // 给选择的item加分,便于后面调整排序 (这里的idFun使用注册表中已经有的,也是我们确认item唯一的函数)
                if(itemData != null) DataWeightScorer.select(itemData,registry.searchData.idFun);
                // 记录选择的item项
                SelectHistoryRecorder.select(itemData,registry.searchData.idFun);
                // === 如果是简述搜索信息,那就取消a标签的默认跳转事件===
                let hasVassal = $(targetObj).attr("vassal") != null;
                // 初始化textView注册表中的对象
                function showTextPage(title,desc,body) {
                    registry.view.textView.show(`<span style='color:red'>标题</span>:${title}<br /><span style='color:red'>描述:</span>${desc}<br /><span style='color:red'>简述内容:</span><br />${md2html(body)} `)
                }
                if(hasVassal) {
                    showTextPage(itemData.title,"主项的相关/附加内容",itemData.vassal);
                    // 挂载一键code复制
                    codeCopyMount("#text_show");
                    return;
                }else if(itemData.type == "script"){
                    // 是脚本,执行脚本
                    let callBeforeParse = new CallBeforeParse();
                    let jscript = ( itemData.resourceObj == null || itemData.resourceObj.script == null ) ?"function (obj) {alert('- _ - 脚本异常!')}":itemData.resourceObj.script;
                    // 调用里面的函数,传入注册表对象
                    // 打开网址函数

                    function open(url) {
                        let openUrl = url;
                        return {
                            simulator(operate = (click, roll, dimension) => {}) { // 模拟器
                                if(openUrl == null || operate == null || typeof operate != 'function') return;
                                let pageSimulatorScript = operate.toString();
                                addPageSimulatorScript(openUrl,pageSimulatorScript); // 保存模拟操作,模拟脚本将在指定时间内打开指定网址有效
                                window.open(openUrl); // 打开网址
                                return this;
                            }
                        }
                    }
                    let view = {
                        beforeCallback: null,
                        afterCallback: null,
                        mountBefore(handle) {
                            this.beforeCallback = handle;
                            return this;
                        },
                        mountAfter(handle) {
                            this.afterCallback = handle;
                            return this;
                        },
                        // mount是脚本项-脚本js调用
                        mount() {
                            // 看脚本js是否给beforeCallback ,如果有在此执行
                            if(this.beforeCallback != null) this.beforeCallback();
                            // 挂载MS_SCRIPT_ENV 实现系统脚本API到视图
                            registry.script.openSessionForMSSE();
                            // 挂载视图
                            let viewHtml = itemData.resourceObj['view:html'];
                            let viewCss = itemData.resourceObj['view:css'];
                            let viewJs = itemData.resourceObj['view:js'];
                            registry.view.textView.show(viewHtml,viewCss,viewJs);
                            // wait view complate alfter ...
                            waitViewRenderingComplete(()=>{
                                registry.script.tryRunTextViewHandler();
                                // 看脚本js是否给afterCallback ,如果有在此执行
                                if(this.afterCallback != null) this.afterCallback();
                            })
                        }
                    }
                    // 设置logo为运行图标
                    registry.view.logo.change("")
                    try {
                        Function('obj',`(${jscript})(obj)`)({registry,cache,$,open,view})
                    } catch (error) {
                        setTimeout(()=>{alert("Ծ‸Ծ 你选择的是脚本项,而当前页面安全策略不允许此操作所依赖的函数!这种情况是极少数的,请换个页面试试!")},20)
                        console.logout("脚本执行失败!",error);
                    }
                    // logo图标还原
                    setTimeout(()=>{registry.view.logo.reset();},200)
                    return;
                }else if(! isUrl(itemData.resource)) {
                    showTextPage(itemData.title,itemData.desc,itemData.resource)
                    return;
                }
                // 隐藏视图
                registry.view.viewVisibilityController(false)

                const initUrl = itemData.resource;//$(targetObj).attr("href"); // 不作改变的URL
                let url = initUrl; // 进行修改,形成要跳转的真正url
                let temNum = url.matchFetch(/\[\[[^\[\]]*\]\]/gm, function (matchStr,index) { // temNum是url中有几个 "[[...]]", 得到后,就已经得到解析了
                    let templateStr = matchStr;
                    // 使用全局的keyword, 构造出真正的keyword
                    let keyword = registry.searchData.keyword.split(":").reverse();
                    keyword.pop();
                    keyword = keyword.reverse().join(":").trim();

                    let parseAfterStr = matchStr.replace(/{keyword}/g,keyword).replace(/\[\[+|\]\]+/g,"");
                    url = url.replace(templateStr,parseAfterStr);
                });
                // 如果搜索的真正keyword为空字符串,则去掉模板跳转
                if( registry.searchData.keyword.split(registry.searchData.subSearch.searchBoundary).length < 2
                   || registry.searchData.keyword.split(registry.searchData.subSearch.searchBoundary)[1].trim() == "" ) {
                    url = registry.searchData.clearUrlSearchTemplate(initUrl);
                }
                // 跳转(url如果有模板,可能已经去掉模板,取决于是“搜索模式”)
                window.open(url);

            })
            //registry.searchData.searchHandle = handler;
            const refresh = debounce(handler, 300)
            // 第一次触发 scroll 执行一次 fn,后续只有在停止滑动 1 秒后才执行函数 fn
            searchInputDocument.on('input', refresh)
        }
        function ensureViewHide() {
            // 隐藏视图
            // 如果视图还没有初始化,直接退出
            if (! registry.view.initialized) return;
            // 如果正在查看查看“简讯”,先退出简讯
            const nowMode = registry.view.seeNowMode();
            if(nowMode === registry.view.modeEnum.SHOW_ITEM_DETAIL) {
                // 让简讯隐藏
                registry.view.element.textView.css({"display":"none"})
                // 让搜索结果显示
                registry.view.element.matchResult.css({ display:"block",overflow: "hidden" })
                // 通知简讯back事件
                registry.view.itemDetailBackAfterEventListener.forEach(listener=>listener())
                return;
            }
            // 让视图隐藏
            viewDocument.style.display = "none";
            // 将输入框内容置空,在置空前将值备份,好让未好得及的操作它
            registry.view.element.input.val("")
            // 将之前搜索结果置空
            registry.view.element.matchItems.html("")
            // 隐藏文本显示视图
            registry.view.element.textView.css({
                "display":"none"
            })
            // 让搜索结果隐藏
            registry.view.element.matchResult.css({
                "display":"none"
            })
            // 视图隐藏-清理旧数据
            registry.searchData.clearData();
            // 触发视图隐藏事件
            registry.view.viewHideEventAfterListener.forEach(fun=>fun());
        }
        function showView() {
            // 让视图可见
            viewDocument.style.display = "block";
            //聚焦
            registry.view.element.input.focus()
            // 当输入框失去焦点时,隐藏视图
            registry.view.element.input.blur(function() {
                const isLogoButtonPressedRef = registry.view.logo.isLogoButtonPressedRef
                if(isLogoButtonPressedRef.value) {
                    console.logout("隐藏跳过,因为isLogoButtonPressedRef")
                    return
                };
                setTimeout(function(){
                    const isDebuging = isInstructions("debug");
                    const isSearching = registry.searchData.searchEven.isSearching;
                    // 当前视图是否在展示数据,如搜索结果,简述内容?如果在展示不隐藏
                    let isWaitSearch = registry.view.seeNowMode() === registry.view.modeEnum.WAIT_SEARCH;
                    if(isDebuging || isSearching || !isWaitSearch || isLogoButtonPressedRef.value) {
                        console.logout("隐藏跳过,条件列表不满足!")
                        return
                    };
                    registry.view.viewVisibilityController(false);
                },registry.view.delayedHideTime)
            });
        }

        // 返回给外界控制视图显示与隐藏
        return function (isSetViewVisibility) {
            if (isSetViewVisibility) {
                // 让视图可见 >>>
                // 如果还没初始化先初始化   // 初始化数据 initData();
                if (! registry.view.initialized) {
                    // 初始化视图
                    initView();
                    // 初始化数据
                    // initData();
                }
                // 让视图可见
                showView();
            } else {
                // 隐藏视图 >>>
                ensureViewHide();
            }
        }
    })();
    // 触发策略——快捷键
    let useKeyTrigger = function (viewVisibilityController) {
        let isFirstShow = true;
        // 将视图与触发策略绑定
        function showFun() {
            // 让视图可见
            viewVisibilityController(true);
            // 触发视图首次显示事件
            if(isFirstShow) {
                for(let e of registry.view.viewFirstShowEventListener) e();
                isFirstShow = false;
            }
        }
        window.addEventListener('message', event => {
            // console.log("父容器接收到了信息~~")
            if(event.data == MY_SEARCH_SCRIPT_VIEW_SHOW_EVENT) {
                showFun() // 接收显示呼出搜索框
            }
        });
        triggerAndEvent("ctrl+alt+s", showFun)
        triggerAndEvent("Escape", function () {
            // 如果视图还没有初始化,就跳过
            if(registry.view.viewDocument == null ) return;
            // 让视图不可见
            viewVisibilityController(false);
        })
    }

    // 触发策略组
    let trigger_group = [useKeyTrigger];
    // 初始化入选的触发策略
    (function () {
        for (let trigger of trigger_group) {
            trigger(registry.view.viewVisibilityController);
        }
    })();

    // 打开视图进行配置
    // 显示配置视图
    // 是否显示进度 - 进度控制
    function clearCache() {
        cache.remove(registry.searchData.SEARCH_DATA_KEY);
        // 如果处于debug模式,也清理其它的
        if(isInstructions("debug")) {
            cache.remove(registry.searchData.CACHE_FAVICON_SOURCE_KEY);
        }
        // 触发缓存被清理事件
        for(let fun of registry.searchData.dataCacheRemoveEventListener) fun();
    }
    GM_registerMenuCommand("订阅管理",function() {
        showConfigView();
    });
    GM_registerMenuCommand("清理缓存",function() {
        clearCache();
    });

    function giveTagsStatus(tagsOfData,userUnfollowList) {
        // 赋予tags一个是否选中状态
        // 将 userUnfollowList 转为以key为userUnfollowList的item.name值是Item的方便检索
        let userUnfollowMap = userUnfollowList.reduce(function(result, item) {
            result[item] = '';
            return result;
        }, {});
        tagsOfData.forEach(item=>{
            if(userUnfollowMap[item.name] != null ) {
                // 默认都是选中状态,如果item在userUnfollowList上将此tag状态改为未选中状态
                item.status = 0;
            }
        })
        return tagsOfData;
    }
    function showConfigView() {
        // 剃除已转关注的,添加新关注的
        function reshapeUnfollowList(userUnfollowList,userFollowList,newUserUnfollowList) {
            // 剃除已转关注的
            userUnfollowList = userUnfollowList.filter(item => !userFollowList.includes(item));
            // 添加新关注的
            userUnfollowList = userUnfollowList.concat(newUserUnfollowList.filter(item => !userUnfollowList.includes(item)));
            return userUnfollowList;
        }

        if($("#subscribe_save")[0] != null) return;
        // 显示视图
        // 用户维护的取消关注标签列表
        let userUnfollowList = cache.get(registry.searchData.USER_UNFOLLOW_LIST_CACHE_KEY)?? registry.searchData.USER_DEFAULT_UNFOLLOW;
        // 当前数据所有的标签
        let tagsOfData = cache.get(registry.searchData.DATA_ITEM_TAGS_CACHE_KEY);
        // 使用 userUnfollowList 给 tagsOfData中的标签一个是否选中状态,在userUnfollowList中不选中,不在选中,添加一个属性到tagsOfData用boolean表达
        tagsOfData = giveTagsStatus(tagsOfData,userUnfollowList);
        // 生成多选框html
        let tagsCheckboxHtml = "";
        tagsOfData.forEach(item=>{
            tagsCheckboxHtml += `
               <div>
                   <input type="checkbox" id="${item.name}" name="_tagsCheckBox" value="${item.name}" ${item.status==1?'checked':''} >
                   <label for="${item.name}">${item.name} (${item.count})</label>
               </div>
            `
        })

        DivPage(`
         #my-search-view {
            width: 500px;
            max-height: 100%;
            max-width: 100%;
            background: pink;
            position: fixed;
            right: 0px;
            top: 0px;
            z-index: 2147383656;
            padding: 20px;
            box-sizing: border-box;
            border-radius: 14px;
            text-align: left;
            button {
                cursor: pointer;
            }

            ._topController {
                width: 100%;
                position: absolute;
                top: 0px;
                right: 0px;
                text-align: right;
                padding: 15px 15px 0px;
                box-sizing: border-box;
                * {
                    cursor: pointer;
                }
                #topController_close {
                   font-sise: 15px;
                   color: #e8221e;
                }
            }
            .page {
              .control_title {
                   margin: 10px 0px 5px;
                   font-size: 17px;
                   color: black;
               }
            }
            .home {
                .submitable {
                   color: #3CB371;
                }
                .tagsCheckBoxDiv > div {
                    width: 32%;
                    display: inline-block;
                    margin: 0px;
                    padding: 0px;
                    overflow: hidden;
                    text-overflow: ellipsis;
                    white-space: nowrap;
                }
                #all_subscribe {
                    width: 100%;
                    height: 150px;
                    box-sizing: border-box;
                    border: 4px solid #f5f5f5;
                }
                #subscribe_save {
                    margin-top: 20px;
                    border: none;
                    border-radius: 3px;
                    padding: 4px 17px;
                    cursor: pointer;
                    box-sizing: border-box;
                    background: #6161bb;
                    color: #fff;
                }
                .view-base-button {
                   background: #fff;
                   border: none;
                   font-size: 15px;
                   padding: 1px 10px;
                   cursor: pointer;
                   margin: 2px;
                   color: black;
                }
                ._topController span {
                   color: #3CB371;
                }
                .home label {
                  font-size: 13px;
                }
            }
            .tis-hub {
                .logo-search {
                    display: flex;
                    flex-direction: column;
                    align-items: center;
                    img {
                        display: block;
                        width: 40px;
                        height: 40px;
                     }
                     .keyword {
                         display: flex;
                         font-size: 12px;
                         width: 70%;
                         margin-top: 5px;
                         input {
                             border: none;
                             padding: 0 6px;
                             min-width: 100px;
                             line-height: 25px;
                             height: 25px;
                             flex-grow: 1;
                         }
                         button {
                             padding: 0 12px;
                             border: none;
                             background: #f0f0f0;
                             line-height: 25px;
                             height: 25px;
                         }
                      }
                }
                .search-type {
                    display: flex;
                    padding: 10px 0;
                    label {
                        display: flex;
                        align-items: center;
                        margin-right: 20px;
                        font-size: 14px;
                        input {
                            padding: 0;
                            margin: 0 3px 0 0;
                        }
                    }
                }
                .result-list {
                   min-height: 300px;
                   padding-top: 15px;
                   .hub-tis {
                      display: flex;
                      justify-content: space-between;
                      margin-bottom: 12px;
                      align-items: center;
                      button {
                          font-size: 10px;
                          line-height: 22px;
                          height: 22px;
                          padding: 0 15px;
                          border-radius: 3px;
                          border: none;
                      }
                      .tis-info {
                         display: flex;
                         flex-direction: column;
                         .title {
                             font-size: 14px;
                             font-weight: bold;
                             color: rgb(103, 0, 0);
                         }
                         .describe {
                             font-size: 12px;
                             font-weight: 400;
                             display: block;
                             font-size: smaller;
                             margin: 0.5em 0px;
                             color: #333333;
                         }

                      }
                   }
                }
            }
        }
    `,`
    <div id="my-search-view">
        <div class="_topController">
            <span id="topController_close">X</span>
        </div>
        <div class="page home">
            <div>
                <p class="control_title">订阅总览:</p>
                <textarea id="all_subscribe"></textarea>
            </div>
            <div>
                <p class="control_title">公共仓库:</p>
                <button id="pushTis" class="view-base-button">提交我的订阅到TisHub(<span class="submitable"> - </span>)</button>
                <button id="openTisHub" class="view-base-button">Tis订阅市场</button>
                <button id="clearToken" class="view-base-button" style="display:none;">清理Token (存在)</button>
            </div>
            <div>
                <p class="control_title">关注标签:</p>
                <div class="tagsCheckBoxDiv">
                    ${tagsCheckboxHtml}
                </div>
            </div>
            <button id="subscribe_save">保存并应用</button>
        </div>
        <div class="page tis-hub">
            <p class="control_title">订阅市场</p>
            <div class="logo-search">
                <a href="https://github.com/My-Search/TisHub" target="_blank">
                    <img src="https://cdn.jsdelivr.net/gh/My-Search/TisHub/favicon.ico" title="TisHub是一个GitHub仓库,订阅以Issues的方式存在!" />
                </a>
                <div class="keyword">
                    <input name="keyword" placeholder="请输入搜索关键字..." />
                    <button id="search-tishub">搜索</button>
                </div>
            </div>
            <div class="search-type">
                <label>
                    <input type="radio" name="search-type" value="installed" checked>
                    已安装
                </label><br>
                <label>
                    <input type="radio" name="search-type" value="market">
                    市场订阅
                </label>
            </div>
            <div class="result-list">
                <div class="list-rol">
                </div>
            </div>
        </div>
    </div>

    `,function (selector,remove) {
            let subscribe_text = selector("#all_subscribe");
            let subscribe_save = selector("#subscribe_save");
            let topController_close = selector("#topController_close");
            let openTisHub = selector("#openTisHub");
            let tisHubLink = "https://github.com/My-Search/TisHub/issues";
            let pushTis = selector("#pushTis");
            let commitableTisList = null;
            let clearToken = selector("#clearToken");
            let mySearchView = selector("#my-search-view");
            let currentPage = setPage(); // 默认显示的是home页

            // 刷新页
            function setPage(page = "home") {
                $(mySearchView).find('.page').hide().filter(`.${page}`).show();
            }
            setPage("home");
            // 刷新视图状态
            async function refreshViewState() {
                // 更新token状态
                $(clearToken).css({"display":GithubAPI.getToken() == null?"none":"inline-block"})
                // 更新可提交数
                let tisList = await TisHub.getTisHubAllTis();
                if(tisList != null && tisList.length != 0) {
                    commitableTisList = TisHub.tisFilter(subscribe_text.value,tisList)??[]
                    $(pushTis).find("span").text(commitableTisList.length);
                }
            }
            // 初始化subscribe_text的值
            subscribe_text.value = getSubscribe();
            // 初始化其它状态,通过调用refreshViewState()
            refreshViewState();
            // 当SubscribeText多行输入框内容发生改变时,刷新更新可提交数,通过调用refreshViewState()
            let refreshSubscribeText = debounce(()=>{refreshViewState() }, 300)
            subscribe_text.oninput = ()=>{refreshSubscribeText();}
            // 保存
            function configViewClose() {
                remove();
            }
            // 点击保存时
            subscribe_save.onclick=function() {
                // 保存用户选择的关注标签(维护数据)
                // 获取所有多选框元素
                var checkboxes = selector(".tagsCheckBoxDiv input",true);
                // 初始化已选中和未选中的数组
                var userFollowList = [];
                var newUserUnfollowList = [];
                // 遍历多选框元素,将选中的元素的value值添加到checkedValues数组中,
                // 未选中的元素的value值添加到uncheckedValues数组中
                for (var i = 0; i < checkboxes.length; i++) {
                    if (checkboxes[i].checked) {
                        userFollowList.push(checkboxes[i].value);
                    } else {
                        newUserUnfollowList.push(checkboxes[i].value);
                    }
                }
                // 剃除已转关注的,添加新关注的
                newUserUnfollowList = reshapeUnfollowList( userUnfollowList,userFollowList,newUserUnfollowList);
                cache.set(registry.searchData.USER_UNFOLLOW_LIST_CACHE_KEY,newUserUnfollowList);

                // 保存到对象
                let allSubscribe = subscribe_text.value;
                let validCount = editSubscribe(allSubscribe);
                // 清除视图
                configViewClose();
                // 清理缓存,让数据重新加载
                clearCache();
                alert("保存配置成功!有效订阅数:"+validCount);

            }
            // 打开TitHub
            openTisHub.onclick = function() {
                // window.open(tisHubLink, "_blank");
                setPage("tis-hub");
            }
            // push到TisHub公共仓库中
            pushTis.onclick =async function () {
                if(! confirm("是否确认要提交到TisHub公共仓库?")) return;
                if(commitableTisList == null || commitableTisList.length == 0) {
                    alert("经过与TisHub中订阅的比较,本地没有可提交的订阅!")
                    return;
                }
                if(GithubAPI.getToken(true) == null) {
                    alert("获取token失败,无法继续!");
                    return;
                }
                // 组装提交的body
                let body = (()=>{
                    let _body = "";
                    for(let tis of commitableTisList) _body+=tis;
                    return _body;
                })();
                if ( body == "") return;
                let userInfo = await GithubAPI.setToken().getUserInfo();
                if(userInfo == null) {
                    alert("提交异常,请检查网络或提交的Token信息!")
                    return;
                }
                GithubAPI.commitIssues({
                    "title": userInfo.name+"的订阅",
                    "body": body
                }).then(response=>{
                    refreshViewState();
                    alert("提交成功(issues)!感谢您的参与,脚本因你而更加精彩。")
                }).catch(error=>alert("提交失败~"))
            }
            // 清理token
            clearToken.onclick = function(){
                GithubAPI.clearToken(); // 清理token
                refreshViewState(); // 刷新视图变量
            };
            // 关闭
            $(topController_close).click(configViewClose)

            // 点击搜索tis-hub
            let installedList = cache.get(registry.searchData.USE_INSTALL_TISHUB_CACHE_KEY) || []; // [ {name: "官方订阅",describe: "这是官方订阅...", body: "<tis::http... />",status: ""} ]  status: disable enable installable
            let tisSearchInput = $(".tis-hub .keyword input");
            let tisSearchBtn = $("#search-tishub");
            let searchFun = async function() {
                const keyword = tisSearchInput.val()?.trim() || '';
                // 搜索类型(installed | market)
                const searchType = $('.search-type input[name="search-type"]:checked').val();
                let resultTisList = installedList.filter(item => keyword === "" || item.name.includes(keyword));
                if(searchType === "market") {
                     let marketResult = await TisHub.getClosedIssuesTis({keyword})
                     marketResult = marketResult.map(hubTisInfo => {
                         return {
                             name: hubTisInfo.title,
                             describe: hubTisInfo.describe,
                             body: hubTisInfo.tisList.join('\n') || '',
                             state: "installable"
                         }
                     })
                    const installedMap = resultTisList.reduce((map, item) => {
                         map[item.name] = item;
                         return map;
                     }, {});

                    // 看本地是否已安装,如果已安装state就取已安装的项state
                    (resultTisList = marketResult).forEach(hubTis =>{
                        if(installedMap[hubTis.name]) hubTis.state = installedMap[hubTis.name].state;
                    });
                }
                // 列表渲染
                const resultElement = $(".tis-hub .result-list > .list-rol");
                resultElement.html('')
                // 转状态名
                function stateAsName(state) {
                    return (state === "disable" && "移除(未启用)") || (state === "enable" && "移除") || "安装";
                }

                for(let tis of resultTisList) {
                    // tis 有该订阅的名 tis.name
                    // tisMetaInfo 是tis.body 包含描述信息 tisMetaInfo.describe
                    const tisMetaInfo = new PageTextHandleChains(tis.body).parseAllDesignatedSingTags("tis")[0];
                    // 自己的搜索逻辑
                    if(! `${tis.name}`.includes(keyword) && (tisMetaInfo != null && ! `${tisMetaInfo.describe}`.includes(keyword))) return;
                    // 渲染到页面
                    resultElement.append(`
                       <div class="hub-tis">
                           <div class="tis-info">
                               <a class="title" href="${tisMetaInfo.tabValue}" target="_blank">${tis.name}</a>
                               <span class="describe">${tisMetaInfo.describe || '订阅没有描述信息,请确认订阅安全或信任后再安装!'}</span>
                           </div>
                           <button class="tis-button" tis-name="${tis.name}">${ stateAsName(tis.state) }</button>
                       </div>
                    `)
                }
                // 当点击tis-button按钮时
                $(".hub-tis .tis-button").click(function() {
                    // 使用 $(this) 获取当前被点击的元素
                    const button = $(this);
                    const tisName = button.attr("tis-name");
                    let tis = installedList.find(item=>item.name === tisName);
                    if(tis != null) {
                        // 移除
                        installedList = installedList.filter(item => item.name !== tisName);
                        tis.state = "installable";
                    }else {
                        // 安装
                        const hubTis = resultTisList.find(item=>item.name === tisName);
                        hubTis.state = "enable";
                        installedList.unshift(tis = hubTis);
                    }
                    // 更新状态
                    button.html(stateAsName(tis.state));
                    // 保存
                    console.log("保存:",installedList)
                    cache.set(registry.searchData.USE_INSTALL_TISHUB_CACHE_KEY,installedList);
                    // 清理缓存
                    clearCache()
                });
            }
            // 点击搜索
            const searchButton = tisSearchBtn.click(searchFun).click();
            // 回车键触发搜索
            tisSearchInput.on("keydown", function(event) {
                if (event.key === "Enter") {
                    searchFun(); // 调用搜索功能
                }
            });

            // 单选框值改变时,搜索
            const radioButtons = document.querySelectorAll('input[name="search-type"]');
            radioButtons.forEach(radio => {
                radio.addEventListener('change', function() {
                    if (this.checked) {
                        searchButton.click();
                    }
                });
            });

        })
    }

})(unsafeWindow);
// unsafeWindow是真实的window,作为参数传入