我的搜索

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

当前为 2023-05-19 提交的版本,查看 最新版本

您需要先安装一个扩展,例如 篡改猴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      5.2
// @description  打造订阅式搜索,让我的搜索,只搜精品!
// @license MIT
// @author       zhuangjie
// @exclude  http://127.0.0.1*
// @exclude  http://localhost*
// @match      *://*/*
// @exclude  http://192.168.*
// @icon         
// @require      https://cdn.bootcdn.net/ajax/libs/jquery/3.6.2/jquery.min.js
// @require      https://unpkg.com/pinyin-pro

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

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

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

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

// ==/UserScript==

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




    // 脚本引入css文件
    GM_addStyle(GM_getResourceText("code-css"));
    GM_addStyle(GM_getResourceText("markdown-css"));


    // 正则捕获
    // 正则捕获
    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;
    }


    // 重写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】
    const converter = new showdown.Converter({
        simpleLineBreaks:true,
        openLinksInNewWindow: true,
        metadata:true
    });

    // 引入js
    /*$(document.head).html($(document.head).html()+`
       <script type="text/javascript" src="https://cdn.jsdelivr.net/gh/zh-lx/[email protected]/dist/pinyin-pro.js"></script>
    `)*/
    // 提取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];
            return {protocol,domain,path,params}
        }
        return null;
    }

    // 检查网站是否可用
    function checkUsability(templateUrl,isStopCheck = false) {
        return new Promise(function (resolve, reject) {
            // 判断是否要检查
            if(isStopCheck) {
                reject(null);
                return;
            }
            var img=document.createElement("img");
            img.src = templateUrl.replace("<url>","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 = {
        get(key) {
            return GM_getValue(key);
        },
        set(key,value) {
            GM_setValue(key,value);
        },
        jGet(key) {
            let value = GM_getValue(key);
            if( value == null) return value;
            return JSON.parse(value);
        },
        jSet(key,value) {
            value = JSON.stringify(value)
            GM_setValue(key,value);
        },
        remove(key) {
            GM_deleteValue(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 "";
        }
    }
    // ==偏业务工具函数==
    // 根据反馈的错误项调整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;
    }


    // 全局注册表
    let ERROR = {
        tell(info) {
            console.error("ERROR " + info)
        }
    }


    //registry.searchData.SEARCH_DATA_KEY
    //registry.view.onViewFirstShow
    let registry = {// registry.view.onViewFirstShow
        view: {
            viewVisibilityController: () => { ERROR.tell("视图未初始化,但你使用了它的未初始化的注册表信息!") },
            viewDocument: null,
            setButtonVisibility: () => { ERROR.tell("按钮未初始化!") },
            titleHandlerFuns: [],
            onViewFirstShow: [],
            menuActive: false,
            // 视图延时隐藏时间
            delayedHideTime: 200,
            initialized: false
        },
        other: {  // registry.other.UPDATE_CDNS_CACHE_KEY
            UPDATE_CDNS_CACHE_KEY: "UPDATE_CDNS_CACHE_KEY"
        },
        searchData: { //registry.searchData.keyword  registry.searchData.OLD_SEARCH_DATA_KEY
            data: [],
            // 旧的新数据
            OLD_SEARCH_DATA_KEY: "OLD_SEARCH_DATAS_KEY",
            // 标签数据缓存KEY
            DATA_ITEM_FLAGS_CACHE_KEY: "DATA_ITEM_FLAGS_CACHE_KEY",
            // 用户维护的不关注标签列表,缓存KEY
            USER_UNFOLLOW_LIST_CACHE_KEY: "USER_UNFOLLOW_LIST_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:{},
                send(keyword) {
                    keyword = (keyword??"").trim().toUpperCase();
                    let fun = this.event[keyword];
                    if(fun != null) {
                        return fun();
                    }
                    return null;
                }
            },
            // 新数据设置的过期天数
            NEW_DATA_EXPIRE_DAY_NUM:7,
            // 搜索逻辑,可用来手动触发搜索
            triggerSearchHandle: function (keyword=""){
                // 获取input元素
                const inputEl = document.getElementById('my_search_input');
                // 如果有传入搜索值,就要设置值
                if(keyword != null) {
                    inputEl.value = keyword;
                }
                // 创建一个事件对象
                const event = new Event('input', { bubbles: true });
                // 手动触发input事件
                inputEl.dispatchEvent(event);
                // 维护全局搜索keyword
                this.keyword = keyword;
            },
            // 事件函数
            dataChange: [],
            onSearch: [],
            // 第一次处理数据块前
            onFirstHandleNewDataBlock: [],
            // 新数据块处理完成事件
            onNewDataBlockHandleAfter: [],
            // 新数据的flag
            NEW_ITEMS_FLAG: "[新]",
            // 搜索的keyword
            keyword: "",
            // 持久化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: ["https://ico.di8du.com/get.php?url=<url>","https://www.google.com/s2/favicons?domain=<url>","<url>/favicon.ico"],
            CACHE_FAVICON_SOURCE_KEY: "CACHE_FAVICON_SOURCE_KEY",
            CACHE_FAVICON_SOURCE_TIMEOUT: 1000*60*60*12, // 12个小时重新检测一下favicon源/过期时间
            getFaviconAPI: (function(){
                let faviconUrlTemplate = "<url>/favicon.ico";
                let isRemoteTemplate = false;
                // 查看是否已经检查模板
                function checkTemplateAndUpdateTemplate() {
                    if( !isRemoteTemplate && cache.get(registry.searchData.CACHE_FAVICON_SOURCE_KEY) != null ) {
                        faviconUrlTemplate = cache.get(registry.searchData.CACHE_FAVICON_SOURCE_KEY).sourceTemplate;
                        // 设置已经是远程Favicon模板
                        isRemoteTemplate = true;
                    }
                }
                return function(url) {
                    checkTemplateAndUpdateTemplate();
                    // 去掉模板,才是真正的URL
                    url = registry.searchData.clearUrlSearchTemplate(url);
                    // 获取网站的favicon没有使用第三方的api,而是直接请求网站的 协议://域名/favicon.ico 来获取
                    // 备选的第三方api: https://github.com/antongunov/favicongrabber.com 或 https://www.google.com/s2/favicons?domain=<url>
                    // return `https://s.qwant.com/fav/q/q/${getUrlRoot(url,true,true)}.ico`;
                    return faviconUrlTemplate.replace("<url>",url);
                }
            })(),
            tmpVar: null, // 用于防抖
            getDataLength(target = "SELECT") {
                // 持久化的数据项数
                let cacheData = cache.get(this.SEARCH_DATA_KEY) == null?[]:cache.get(this.SEARCH_DATA_KEY).data;
                let tmpData = this.data.length === 0?cacheData:this.data;
                let dataLength = (tmpData == null)?0:tmpData.length;
                // 全部的输入提示
                let inputDescs = ["我的搜索,只搜精品"];
                // 当前应用“输入提示”
                let inputDesc = inputDescs[Math.floor(Math.random()*inputDescs.length)];
                if(target == "UPDATE") {
                    if(this.tmpVar != null) {
                        clearTimeout(this.tmpVar);
                    }
                    this.tmpVar = setTimeout(()=>{
                        $("#my_search_input").attr("placeholder",this.getDataLength());
                    },1200)
                    return `可以搜索( 🔁 数据库更新到 ${dataLength}条)`;
                }
                return inputDesc;

            },
            searchBoundary: " : ",
            dataAddBeforeHandler: {
                handlers: [],
                handler(items) {
                    for(let handlerFun of this.handlers) {
                        items = handlerFun(items);
                    }
                    return items;
                }
            },
            // 存储着text转pinyin的历史  registry.searchData.TEXT_PINYIN_KEY
            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")??{});
                }
            })(),
            isSearchPro: false,
            searchProFlag: "[可搜索]"

        }

    }
    let dao = {}

    // 新版本执行代码
    // 定义一个函数,用于检查版本号并执行指定代码块
    function versionHandleFun() {
        // cache.remove(registry.searchData.SEARCH_DATA_KEY)
        // cache.remove(registry.searchData.OLD_SEARCH_DATA_KEY)
        // cache.remove(registry.searchData.SEARCH_NEW_ITEMS_KEY)
    }
    // 检查当前版本是否满足条件执行 versionHandleFun 函数
    (function () {
        // 当前版本号
        let currentVersion = GM_info.version;
        if(currentVersion == null ) return;
        // 从 localStorage 中获取上一个版本号
        var previousVersion = localStorage.getItem("version");
        // 如果上一个版本号不存在或与当前版本号不同,则执行指定代码块,并将当前版本号保存到 localStorage 中
        //console.log("版本信息:",currentVersion,previousVersion)
        if (previousVersion == null || previousVersion !== currentVersion) {
            // 执行指定代码块
            versionHandleFun();
            // 将当前版本号保存到 localStorage 中
            localStorage.setItem("version", currentVersion);
        }
    })();

    // 判断是否只是url且不应该是URL文本 (用于查看类型)
    function isUrlNoUrlText(str = "") {
        str = str.trim().split("#")[0];
        // 不能存在换行符,如果存在不满足
        if(str.indexOf("\n") != -1 ) return false;
        // 被“空白符”切割后只能有一个元素
        if(str.split(/\s+/).length != 1) return false;
        // 如果不满足url,返回false
        if(! /^https?:\/\/.+/i.test(str) ) return false;
        return true;
    }
    /*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;
        if(cacheFaviconSourceData == null || currentTime - cacheFaviconSourceData.updateTime > timeout ) {
            if(cacheFaviconSourceData != null) {
                console.logout(`==超时${(currentTime - cacheFaviconSourceData.updateTime - timeout)/1000}s,重新设置Favicon源==`);
            }
            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
                    })
                }
            }
            let faviconSources = registry.searchData.faviconSources;
            let pos = 0;
            let promise = null;
            // 去测试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.onViewFirstShow.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);
    }
    // 异步函数
    function asyncExecFun(fun,time = 20) {
        setTimeout(()=>{
            fun();
        },time)
    }
    // 同步执行函数
    let syncExecFun = (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;
    }
    function removeDuplicates(objs,selecter) {
        let itemType = objs[0] == null?false:typeof objs[0];
        // 比较两个属性相等
        function compareObjects(obj1, obj2) {
            if(selecter != null ) return selecter(obj1) == selecter(obj2);
            if(itemType != "object" ) return obj1 == obj2;
            // 如果是对象且selecter没有传入时,比较对象的全部属性
            const keys1 = Object.keys(obj1);
            const keys2 = Object.keys(obj2);

            if (keys1.length !== keys2.length) {
                return false;
            }

            for (let key of keys1) {
                if (!(key in obj2) || obj1[key] !== obj2[key]) {
                    return false;
                }
            }
            return true;
        }
        for(let i = 0; i< objs.length; i++ ) {
            let item1 = objs[i];
            for(let j = i+1; j< objs.length; j++ ) {
                let item2 = objs[j];
                if(item2 == null ) continue;
                if( compareObjects(item1,item2) ) {
                    objs[i] = null;
                    break;
                }
            }
        }
        // 去掉无效新数据(item == null)-- 必须先去重
        return objs.filter((item, index) => item != null);
    }
    // 【追加原型函数】
    // Date对象转年月日
    Date.prototype.toDateString = function(separator = "-") {
        const year = this.getFullYear();
        const month = this.getMonth() + 1;
        const day = this.getDate();

        return `${year}${separator}${month}${separator}${day};`;
    };

    // 往字符原型中添加新的方法 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 = "\\$.*?{.*?"+key+".*?}";
            resultUrl = resultUrl.replace(new RegExp(regexStr),obj[key]);
        }
        if(/\$.*?{.*?}/.test(resultUrl)) return null;
        return resultUrl;
    }
    // 比较两个数组是否相等(顺序不相同不影响)
    Array.prototype.isEqual = function (arr2) {
        let arr1 = this;
        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;
    }
    Array.prototype.diff = function ( arr2, idFun = () => null,diffRange = 3) { // diffRange值:“1”是左边多的,“2”是右边数组多的,3是左右合并,0是相同的部分,30是两个数组去重的
        if(window.hashString == null) {
            window.hashString = function (obj) {
                let str = JSON.stringify(obj);
                let hash = 0;
                for (let i = 0; i < str.length; i++) {
                    hash += str.charCodeAt(i);
                }
                return hash;
            }
        }
        let arr1 = this;
        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 = window.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(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(`

/*定义字体*/

 @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;
}

.resultItem {
	animation-duration: 0.5s;
	animation-name: resultItem;
}

@-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;
}
/*当视图大于等于1400.1px时*/
@media (min-width: 1400.1px) {
  #my_search_box {
    left: 27%;
    right:27%;
  }
}
/*当视图小于等于1400px时*/
@media (max-width: 1400px) {
  #my_search_box {
    left: 24%;
    right:24%;
  }
}
/*当视图小于等于1200px时*/
@media (max-width: 1200px) {
  #my_search_box {
    left: 21%;
    right:21%;
  }
}
/*当视图小于等于1100px时*/
@media (max-width: 1100px) {
  #my_search_box {
    left: 18%;
    right:18%;
  }
}
/*当视图小于等于800px时*/
@media (max-width: 800px) {
  #my_search_box {
    left: 15%;
    right:15%;
  }
}
/*输入框右边按钮*/
#controlButton {
    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;
    outline: none;
}
#controlButton img {
   display: block;
   width: 25px;
}

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


    //防抖函数模板
    function debounce(fun, wait) {
        let timer = null;
        return function (...args) {
            // 清除原来的定时器
            if (timer) clearTimeout(timer)
            // 开启一个新的定时器
            timer = setTimeout(() => {
                fun.apply(this, args)
            }, wait)
        }
    }
    // 判断是否为指定指令
    function isInstructions(val,cmd) {
        return val == ":"+cmd;
    }



    // 向数据项中加入拼音项 如:title加了titlePinyin, desc加了descPinyin
    function genDataItemPinyin(threadHandleItems){
        let textPinyinMap = registry.searchData.getGlobalTextPinyinMap();
        // console.logout("分配的预热item:",threadHandleItems)
        asyncExecFun(()=>{
            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.data;
        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.dataChange.push(refresh);

    // 从数据库获取新数据
    let newDataLine = [];
    function getNewDataLine(data) {

    }
    registry.searchData.dataChange.push(getNewDataLine);
    // 实现模块一:使用快捷键触发指定事件
    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.toReplaceAll("alt", "").toReplaceAll("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);
    }

    // 解决有些网站不加载图片
    // 插入 meta 标签
    /*    var oMeta = document.createElement('meta');
    oMeta.content = "default-src 'self' data: * 'unsafe-eval'; style-src 'self' 'unsafe-inline'; media-src * ;img-src *";
    oMeta['http-equiv'] = "Content-Security-Policy";
    document.getElementsByTagName('head')[0].appendChild(oMeta);
*/
    // 【数据初始化】
    // 获取存在的订阅信息
    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 = pageTextHandleChains.init(subscribe);
        let tisHasFetchFun = pageTextHandleChainsY.parseSingleTab("tis","fetchFun");
        let tisNotFetchFun = pageTextHandleChainsY.parseSingleTabValue("tis");

        let tis = [...tisHasFetchFun, ...tisNotFetchFun];
        // 生成订阅信息存储
        let subscribeText = "\n";
        for(let aTis of tisHasFetchFun) {
            subscribeText += `<tis::${aTis.tabValue} fetchFun="${aTis.attrValue}" />\n`
        }
        for(let aTis of tisNotFetchFun) {
            subscribeText += `<tis::${aTis.tabValue} />\n`
        }
        // 持久化
        let newSubscribeInfo = subscribeText.replace(/\n+/gm,"\n\n");
        cache.set(registry.searchData.subscribeKey,newSubscribeInfo);
        return tis.length;
    }
    // 存储订阅信息,当指定 sLineFetchFun 时,表示将解析“直接页”的配置,如果没有指定 sLineFetchFun 时,只解析内容
    // 在提取函数中 \n 要改写为 \\n
    let dataSources = getSubscribe()+ `
       <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 current_build_search_item_resource = "";
                let point = 0; // 指的是上面的 current_build_search_item
                let default_desc = "--无描述--"
                function getTitleLineData(titleLine) {
                   const regex = /^# ([^()()]+)[((]?([^()()]*)[^))]?/;
                   let matchData =  regex.exec(titleLine)
                   return {
                      title: matchData[1],
                      desc: ((matchData[2]==null || matchData[2] == "")?default_desc:matchData[2])
                   }
                }
                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;
                    // 向当前搜索项目容器追加当前行
                    current_build_search_item_resource += (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;
                       // 打包装箱
                       search_data_lines.push(current_build_search_item);
                    }
                }
                // 添加种类
                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>
    `;


    // 判断是否是github文件链接
    let githubUrlFlag = "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);
        // 提供的加速模板(顺序会在后面的请求中进行重排序-请求错误反馈的使重排序)
        let initCdnrs = ["https://ghproxy.net/${protocol}${domain}${path}","https://ghps.cc/${protocol}${domain}${path}","https://github.moeyy.xyz/${protocol}${domain}${path}"];
        // 如果我们修改了最开始提供的加速模板,比如新添加/删除了一个会使用新的
        if(cdnrs == null || ! initCdnrs.isEqual(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(githubUrlFlag)) ) {
                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};
        }
    })();

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


    // 使用责任链模式——对pageText进行操作的工具
    const pageTextHandleChains = {
        pageText: "",
        setPageText(newPageText) {
            this.pageText = newPageText;
        },
        getPageText() {
            return this.pageText;
        },
        init(newPageText = "") {
            // 深拷贝一份实例
            let wo = {...this};
            // 初始化
            wo.setPageText(newPageText);
            return wo;
        },
        // 解析双标签-获取指定标签下指定属性下的值
        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;
        },
        // 获取指定单标签指定属性与标签值(标签::值)
        parseSingleTab(tabName,attrName) {
            // 返回指定标签下指定属性下的值
            const regex = RegExp(`<${tabName}::([^\\s<>]*)\\s*${attrName}="([^"<>]*)"\\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],
                    attrValue: m[2]
                })

                const newPageText =this.pageText.replace(m[0], "");
                this.pageText = newPageText;
            }

            return tabNameArr;
        },
        parseSingleTabValue(tabName) {
            // 返回指定标签下指定属性下的值
            const regex = RegExp(`<${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;
        },

        // 清除指定单双标签
        cleanTabByTabName(tabName) {
            const regex = RegExp(`<\\s*${tabName}[^<>]*>([^<>]*)(<\/[^<>]*>)*`,"gm");
            // 替换的内容
            const subst = ``;
            // 被替换的值将包含在结果变量中
            const cleanedText = this.pageText.replace(regex, subst);
            this.pageText = cleanedText;

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

        let config = {
            // {url、fetchFun属性}
            tis: [],
            // {name与fetchFun属性}
            fetchFuns: []
        }
        // 从config中放在返回对象中
        let pageTextHandleChainsX = pageTextHandleChains.init(pageText);
        let fetchFunTabDatas = pageTextHandleChainsX.parseDoubleTab("fetchFun","name");
        for(let fetchFunTabData of fetchFunTabDatas) {
            config.fetchFuns.push( { name:fetchFunTabData.attrValue,fetchFun:fetchFunTabData.tabValue } )
        }
        // 获取tis
        let tisHasFetchFun = pageTextHandleChainsX.parseSingleTab("tis","fetchFun");
        let tisNotFetchFun = pageTextHandleChainsX.parseSingleTabValue("tis");
        let tisArr = [...tisHasFetchFun, ...tisNotFetchFun]
        for(let tis of tisArr) {
            config.tis.push( { url:tis.tabValue, fetchFun:tis.attrValue } )
        }

        return config;

    }
    // 将url转为文本(url请求得到的就是文本),当下面的dataSourceUrl不是http的url时,就会直接返回,不作请求
    function urlToText(dataSourceUrl) {

        // dataSourceUrl 转text
        return new Promise(function (resolve, reject) {
            if((dataSourceUrl.trim().indexOf("http") != 0 ) ) 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;
                $.ajax({
                    url: `${url}?t=${+new Date().getTime()}`,
                    timeout: 5000, // 设置超时时间为 5 秒钟
                    success: function (result) {
                        resolve(result)
                    },
                    error: function(xhr, status, errorThrown){
                        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 = [];
    registry.searchData.dataAddBeforeHandler.handlers.push(function (items) {
        for(let searchItem of items) {
            let resource = searchItem.resource;
            let isSearchableItem = /\[\[[^\[\]]+keyword[^\[\]]+\]\]/.test(resource);
            // 判断是否为可搜索
            if( ! isSearchableItem || searchItem.type == "sketch" || /<\s*br\s*\/\s*>/.test(resource) ) continue;
            searchItem.title = registry.searchData.searchProFlag+searchItem.title;
        }
        return items;
    })
    // 防抖函数->处理新数据
    const triggerRefreshNewData = debounce(()=>{
        // 排序(根据配置在obj中的权重)
        registry.searchData.onNewDataBlockHandleAfter.sort((obj1,obj2)=>obj2.weights-obj1.weights);
        for(let obj of registry.searchData.onNewDataBlockHandleAfter) syncExecFun(obj.fun);
    }, 1200)
    // 缓存数据
    function cacheSearchData(newSearchData) {
        console.logout("触发了缓存,当前数据",registry.searchData.data)
        // 当有数据加入到全局数据容器时,会触发缓存,当前函数会执行
        let SEARCH_DATA_KEY = registry.searchData.SEARCH_DATA_KEY;
        cache.remove(SEARCH_DATA_KEY)
        cache.set(SEARCH_DATA_KEY,{
            data: registry.searchData.data,
            expire: new Date().getTime() + (1000*60*60*12) // 12个小时
            //expire: new Date().getTime() + (2000) // 测试,一秒过期
        })
    }
    // 将解析出来的部分数据push 到 registry.searchData.data的操作队列 (push到全局不能并发,所以这里必须是单线程操作)
    let searchDataController = {
        // 加入到全局的等待队列
        queueData: [],
        // 是否空闲
        isIdle: true,
        // 是否第一次 pushToGlobal
        isFirsPush: true,
        // 给外面触发,加入到队列中
        pushToGlobal:function(newItems) {
            // 如果是第一次push,清空全局数据
            if(this.isFirsPush) {
                this.isFirsPush = false;
                // 清空全局数据容器的数据
                registry.searchData.data = [];
                // 触发第一次处理Block事件
                for(let f of registry.searchData.onFirstHandleNewDataBlock) f();
            }
            // 在添加前,进行额外处理添加,如给有”{keyword}“的url搜索项添加”可搜索“标签
            newItems = registry.searchData.dataAddBeforeHandler.handler(newItems);
            this.queueData.push(...newItems);
            // 如果当前不是空闲的,其它线程当会处理刚才push的数据到全局中
            if(!this.isIdle) return;
            // 设置当前为工作模式
            this.isIdle = false;
            // 处理队列中的数据
            let newItem = null;
            while((newItem = this.queueData.pop() ) != null) {
                // 下一个编号索引号
                let nextIndex = registry.searchData.data.length;
                // newItem.index = nextIndex;
                registry.searchData.data[nextIndex] = newItem;
            }
            // 设置当前空闲
            this.isIdle = true;
            // 更新视图显示条数
            $("#my_search_input").attr("placeholder",registry.searchData.getDataLength("UPDATE"));
            // 缓存新数据块
            cacheSearchData(newItems);
            // 触发“新数据块处理完成事件”
            triggerRefreshNewData();

        }
    }
    // 转义与恢复,数据进行解析前进行转义,解析后恢复——比如文本中出现“/”,就会出现: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();

    function dataSourceHandle(resourcePageUrl,tisTabFetchFunName) {
        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.url,tis.fetchFun);
                }
                // 清理内容
                pageTextHandleChains.setPageText("");
            }else {
                // --> 是内容 <--
                // 解析内容
                if(tisTabFetchFunName === "") return;
                let fetchFunStr = getFetchFunGetByName(tisTabFetchFunName);

                let search_data_line =(new Function('text', "return ( " + fetchFunStr + " )(`"+callBeforeParse.escape(text)+"`)"))();
                // 将之前修改为 <wrapLine> 改为真正的换行符 \n
                // 处理并push到全局数据容器中
                for(let item of search_data_line) {
                    item.title = callBeforeParse.recovery(item.title);
                    item.desc = callBeforeParse.recovery(item.desc);
                    item.resource = callBeforeParse.recovery(item.resource);
                }
                // 加入到push到全局的搜索数据队列中,等待加入到全局数据容器中
                searchDataController.pushToGlobal(search_data_line);
                // 触发搜索数据改变事件(做缓存等操作,观察者模式)
                for(let fun of registry.searchData.dataChange) {
                    fun(search_data_line);
                }
            }
        })


    }
    // 根据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) {
            // 缓存信息不为空,深入判断是否使用缓存的数据
            let dataExpireTime = dataPackage.expire;
            let currentTime = new Date().getTime();
            // console.logout("缓存的数据:",dataBox.data)
            // 数据多大时,才开启缓存
            const TRIGGER_CACHE_DATA_LENGTH = 300;
            // 判断是否有效,有效的话放到全局容器中
            let isValid = (dataExpireTime != null && dataExpireTime > currentTime && dataPackage.data != null && dataPackage.data.length > 0);
            // 如果网站比较特殊,忽略数据过期时间
            if(!isValid && window.location.host.toUpperCase().indexOf("GITHUB.COM") >= 0) {
                isValid = true;
            }
            // 挂载数据
            // 从缓存中将挂载数据挂载 (条件是视图已经初始化)
            if(registry.view.initialized) registry.searchData.data = dataPackage.data;
            // 如果数据过期,或数据量不满足缓存大小,会去请求数据
            // 检查是否已经执行初始化
            if(isValid && dataPackage.data.length >= TRIGGER_CACHE_DATA_LENGTH ) {
                console.logout(`${registry.view.initialized?'数据加载':'数据检查'}:数据有效期还有${parseInt((dataExpireTime - currentTime)/1000/60)} 分钟!`,dataPackage.data);
                return
            };
        }
        // 在去网络请求获取数据前-检查是否已经执行初始化
        if(! askIsExpiredByTopic("SEARCH_DATA_INIT",10*1000)) return;
        // 内部将使用递归,解析出信息
        dataSourceHandle(dataSources,null);
    }
    // 检查数据有效性,且只有数据无效时挂载到数据
    dataInitFun();
    // 当视图第一次显示时,再执行
    registry.view.onViewFirstShow.push(dataInitFun);

    // 该函数作用是为了防止一个页面多次加载数据,导致页面一直在加载
    function checkIsCanInit() {
        let initFlagKey = "initFlagKey";
        let initFlagValue = cache.get(initFlagKey)??0;
        let currentTime = new Date().getTime();
        let vailTime = 8*1000;

        console.logout("是否过期(为负过期):",initFlagValue , currentTime,initFlagValue +vailTime - currentTime)
        let b = initFlagValue == 0 || initFlagValue + vailTime < currentTime;
        if(b) cache.set(initFlagKey,currentTime);
        return b;
    }
    // 判断是否要直接执行初始化函数-如果没有数据,这里要直接执行
    /*(function() {
        console.logout("请求初始化数据!");
        if(cache.get( registry.searchData.SEARCH_DATA_KEY) == null && checkIsCanInit() ) {
           console.logout("===初始化数据!===");
          // 执行初始化函数
          dataInitFun();
        }

    })();*/


    const refreshNewData = function () {
        console.log("==3:数据对比==")
        // 新数据加载完成-进行数据对比
        // 旧数据,也就是上一次数据,用于与本次比较,得出新添加数据
        let oldSearchData = cache.get(registry.searchData.OLD_SEARCH_DATA_KEY)??[];
        // 当前最新数据,用于搜索
        let currentSearchData = registry.searchData.data;
        // 当前时间戳
        let currentTime = new Date().getTime();
        // 数组差异-获取不同的元素比较的基值
        let idFun = function(item) { // 自定义比较
            if(item == null || !( item instanceof Object && item.title != null)) return null;
            return item.title.toReplaceAll(registry.searchData.NEW_ITEMS_FLAG,"")+item.desc;
        }
        // 准备一个存储新数据项的容器
        let newDataItems = [];
        // 只要 oldSearchData 与 currentSearchData有就可以比较,得到新数据
        if( oldSearchData != null && oldSearchData.length != 0 && currentSearchData != null && currentSearchData.length != 0 ) {
            console.logout("开始差异对比(上次与这次数据):",oldSearchData.length,currentSearchData.length)
            // 这里没有旧的
            newDataItems = currentSearchData.diff(oldSearchData,idFun,1);
            // 给新添加的过期时间(新数据有效期)
            let dayNumber = registry.searchData.NEW_DATA_EXPIRE_DAY_NUM;
            newDataItems.forEach(item=>item.expires=(currentTime++) + (1000*60*60*24*dayNumber));
            console.log("新差异项:",JSON.parse(JSON.stringify(newDataItems)));
        }
        // 过滤掉新数据中带有“带注释”的项
        newDataItems = newDataItems.filter(item=> !item.title.startsWith("#"));
        // 以前的新增数据
        let oldNewItems = cache.get(registry.searchData.SEARCH_NEW_ITEMS_KEY)??[];
        // 确保oldNewItems是合法值
        if( ! oldNewItems instanceof Array) oldNewItems = [];
        // 如果还没有过期的,保留下来放在最新数据中
        for(let item of oldNewItems) {
            if(item != null && item.expires > currentTime) newDataItems.push(item);
        }
        // 总新增去重 (标记 - 过滤标记的 )
        newDataItems = removeDuplicates(newDataItems,(item)=>item.title+item.desc);
        // 忽略新数据条件
        if( newDataItems.length/currentSearchData.length < 0.3 || newDataItems.length < 10 ) {
            // 重新缓存“New Data”
            cache.set(registry.searchData.SEARCH_NEW_ITEMS_KEY,newDataItems);
        }else {
            // 清空数据,不清掉,又不缓存新数据的索引将失效
            cache.set(registry.searchData.SEARCH_NEW_ITEMS_KEY,[]);
        }
        // 改变全局搜索并持久化
        // 1)为全局数据的新数据添加新数据标签
        for(let nItem of newDataItems) {
            for(let cItem of currentSearchData) {
                if(nItem.title === cItem.title && nItem.desc === cItem.desc) {
                    // 修改全局搜索数据中New Data数据添加“新数据”标签
                    if (! cItem.title.startsWith(registry.searchData.NEW_ITEMS_FLAG)) {
                        cItem.title = registry.searchData.NEW_ITEMS_FLAG+cItem.title;
                    }
                    break;
                }
            }
        }
        // 缓存全局数据
        let cacheData = cache.get(registry.searchData.SEARCH_DATA_KEY)??{};
        cacheData.data = currentSearchData;
        cache.set(registry.searchData.SEARCH_DATA_KEY,cacheData);
        // 更新“旧全局数据”:currentSearchData 追加-> oldSearchData
        let newOldSearchData = currentSearchData.diff(oldSearchData,idFun,30);
        cache.set(registry.searchData.OLD_SEARCH_DATA_KEY,newOldSearchData??[]);
    }
    // ############ 使用用户操作的规则对加载出来的数据过滤:(责任链中的一块)
    registry.searchData.onNewDataBlockHandleAfter.push({weights:300 ,fun:refreshNewData});
    // 解析出传入的所有项标签数据
    function parseFlags(data = [],selecterFun = (_item)=>_item) {
        let isArray = Array.isArray(data);
        let items = isArray?data:[data];
        let flagsMap = {
            /*
            "程序员": {
               name: "程序员",
               status: 0/1, // 1正常,0禁搜
               visible: false/true
            }
            */
        }
        // 解析 item.name中包含的标签
        items.forEach(function(item) {
            let captureGroups = captureRegEx(/\[\s*(([^'\]\s]*)\s*')?\s*([^'\]]*)\s*'?\s*]/gm,selecterFun(item));
            captureGroups.forEach(function(group) {
                let params = group[2]??"";
                let label = group[3];
                // 判断是否已经存在
                if(label != null && flagsMap[label] == null ) {
                    let currentHandleFlagObj = flagsMap[label] = {
                        name: label,
                        status: 1, // 正常
                        //visible: params.includes("h"), // 参数中包含h字符表示可见
                        count: 1
                        //params: params
                        //items: [item]
                    }
                    // 如果传入的不是一个数组,那设置下面参数才有意义
                    if(! isArray) {
                        currentHandleFlagObj.params = params;
                    }
                }else {
                    if(flagsMap[label] != null) {
                        flagsMap[label].count++;
                        //flagsMap[label].items.push(item);
                    }

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

    const refreshFlags = function (){
        console.log("==1:解析出数据标签==")
        // 将现有的所有标签提取出来
        // 当前最新数据
        let currentSearchData = registry.searchData.data;
        // 解析
        let dataItemFlags = parseFlags(currentSearchData,(_item=>_item.title));
        // 缓存
        if(dataItemFlags.length > 0) {
            cache.set(registry.searchData.DATA_ITEM_FLAGS_CACHE_KEY,dataItemFlags)
        }
    }
    // ################# 执行顺序从大到小 1000 -> 500
    registry.searchData.onNewDataBlockHandleAfter.push({weights:500 ,fun:refreshFlags});
    // 清理标签(参数中有h的)
    function clearHideFlag(data,get = (_item)=>_item,set = (item,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;
            const subst = ``;
            let cleanedTarget = target.replace(regex, subst);
            if (! isArray) {
                return cleanedTarget;
            }
            set(item,cleanedTarget);
        }
        return isArray?items:data;
    }
    const filterSearchData = function () {
        const filterDataByUserUnfollowList = (itemsData,userUnfollowList = []) => {
            var userUnfollowMap = userUnfollowList.reduce(function(result, item) {
                result[item] = '';
                return result;
            }, {});
            // 开始过滤
            return itemsData.filter(item=>{
                let flags = parseFlags(item.title);
                for(let flag of flags){
                    if(userUnfollowMap[flag.name] != null){
                        // 被过滤
                        return false;
                    }
                }
                return true;
            })
        }
        console.log("==2:去除用户不关注的数据项==")
        // 将现有的所有标签提取出来
        // 当前最新数据
        let currentSearchData = registry.searchData.data;
        // 用户维护的取消关注标签列表
        let userUnfollowList = cache.get(registry.searchData.USER_UNFOLLOW_LIST_CACHE_KEY)?? registry.searchData.USER_DEFAULT_UNFOLLOW;
        // 利用用户维护的取消关注标签列表 过滤 搜索数据
        let filteredSearchData = filterDataByUserUnfollowList(currentSearchData,userUnfollowList);
        // 去标签(参数h),清理每个item中title属性的flag
        let cleanedFlagSearchData = clearHideFlag(filteredSearchData,(_item)=>_item.title,(item,cleaned)=>{item.title=cleaned});
        // 改变全局
        registry.searchData.data = cleanedFlagSearchData;

    }
    // ############### 执行顺序从大到小 1000 -> 500
    registry.searchData.onNewDataBlockHandleAfter.push({weights:400 ,fun:filterSearchData});


    const refreshIndex = function () {
        // 当前最新数据,用于搜索
        let currentSearchData = registry.searchData.data;
        let newDataItems = cache.get(registry.searchData.SEARCH_NEW_ITEMS_KEY);
        // 将 index 给 newDataItems ,不然new中的我们选择与实际选择的不一致问题 !
        // 给全局数据创建索引
        currentSearchData.forEach((item,index)=>{item.index=index});
        // 给NEW建索引
        newDataItems.forEach(NItem=>{
            for(let CItem of currentSearchData) {
                if( CItem.title.includes(NItem.title) && NItem.desc === CItem.desc) {
                    NItem.index = CItem.index;
                    break;
                }
            }
        })
        // 忽略新数据条件
        if( newDataItems.length/currentSearchData.length < 0.3 || newDataItems.length < 10 ) {
            // 重新缓存“New Data”
            cache.set(registry.searchData.SEARCH_NEW_ITEMS_KEY,newDataItems);
        }else {
            // 清空数据,不清掉,又不缓存新数据的索引将失效
            cache.set(registry.searchData.SEARCH_NEW_ITEMS_KEY,[]);
        }
        // 重新缓存全局数据
        let cacheData = cache.get(registry.searchData.SEARCH_DATA_KEY)??{};
        cacheData.data = currentSearchData;
        cache.set(registry.searchData.SEARCH_DATA_KEY,cacheData);
    }
    // ############### 执行顺序从大到小 1000 -> 500
    registry.searchData.onNewDataBlockHandleAfter.push({weights:0 ,fun:refreshIndex});

    // 模块二
    registry.view.viewVisibilityController = (function() {

        // 整个视图对象
        let viewDocument = null;
        let searchInputDocument = null;
        let matchItems = null;
        let searchBox = null;

        let isInitializedView = false;
        let controlButton = null;
        let textShow = null;
        let matchResult = null;
        let initView = function () {

            // 初始化视图
            let view = document.createElement("div")
            view.id = "my_search_box";
            let menu_icon = "";
            view.innerHTML = (`
             <div id="my_search_view">
                <div id="searchBox" >
                    <input placeholder="${registry.searchData.getDataLength()}" id="my_search_input" />
                    <button id="controlButton" >
                       <img src="${menu_icon}" />
                    </button>
                </div>
                <div id="matchResult">
                    <ol id="matchItems">
                       <!-- <li><a href="#">webGL水族馆(测试电脑性能)</a></li> -->
                    </ol>
                </div>
                <!--加“markdown-body”是使用了github-markdown.css 样式!加在markdown文档父容器中-->
                <div id="text_show" class="markdown-body">

                </div>
             </div>
         `)
            // 设置样式
            view.style = `
             position: fixed;top:50px;
             border:2px solid #cecece;z-index:10000;
             background: #ffffff;
             overflow: hidden;
         `;

            // 挂载到文档中
            document.body.appendChild(view)
            // 整个视图对象放在组件全局中/注册表中
            registry.view.viewDocument = viewDocument = view;



            // 搜索框对象
            searchInputDocument = $("#my_search_input")
            matchItems = $("#matchItems");
            searchBox = $("#searchBox")
            controlButton = $("#controlButton")
            textShow = $("#text_show")
            matchResult = $("#matchResult");
            // 菜单函数(点击输入框右边按钮时会调用)
            controlButton.click( function () {
                registry.view.menuActive = true;
                // alert("小彩蛋:可以搜索一下“系统项”了解脚本基本使用哦~");
                // 调用手动触发搜索函数,如果已经搜索过,搜索空串(清理)
                let keyword = "[系统项]";
                registry.searchData.triggerSearchHandle(searchInputDocument.val()==keyword?'':keyword);
                setTimeout(function(){ registry.view.menuActive = false;},registry.view.delayedHideTime+100);
            })

            searchBox.css({
                "height": "45px",
                "background": "#ffffff",
                "padding": "0px",
                "box-sizing": " border-box",
                "z-index": "10001",
                "position":"relative",
                "display": "flex",
                "justify-content": "space-between",
                "align-items": "center",
                "flex-wrap": "nowrap"

            })

            searchInputDocument.css({
                "width": "100%",
                "height": "100%",
                "border": "none",
                "outline": "none",
                "font-size": "15px",
                "background": "#fff",
                "padding": "0px 10px",
                "box-sizing": " border-box",
                "color":"rgba(0,0,0,.87)",
                "font-weight":"400",
                "margin":"0px"
            })
            $("#matchResult").css({
                "display":"none"
            })
            $("#matchResult > ol").css({
                "margin": "0px",
                "padding": "0px 15px 5px"
            })


            textShow.css({
                "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"
            })
            // 图片放大/还原
            textShow.on("click","img",function(e) {
                let target = e.target;
                if(target.isEnlarge??false) {
                    $(this).animate({
                        width: "100%"
                    });
                    // 还原
                    target.isEnlarge = false;
                }else {
                    $(this).animate({
                        width: "900px"
                    });
                    target.isEnlarge = true;
                }
            });
            // 设置视图已经初始化
            registry.view.initialized = true;


            // 在搜索的结果集中上下选择移动然后回车(相当点击)
            searchInputDocument.keyup(function(event){
                let keyword = $(event.target).val().trim();
                // 当不为空时,放到全局keyword中
                if(keyword != "" && keyword != null) {
                    registry.searchData.keyword = event.target.value;
                }

                // 处理keyword中的":"字符
                if(keyword.endsWith("::") || keyword.endsWith("::")) {
                    keyword = keyword.replace(/::|::/,registry.searchData.searchBoundary).replace(/\s+/," ");
                    // 每次要形成一个" : "的时候去掉重复的" : : " -> " : "
                    keyword = keyword.replace(/((\s{1,2}:)+ )/,registry.searchData.searchBoundary);
                    $(event.target).val(keyword.toUpperCase());
                    // 设置当前为“搜索PRO”模式
                    registry.searchData.isSearchPro = true;
                    // registry.view.showControlButton("PRO模式")

                }

                // 判断是否要退出搜索模式
                if($(event.target).val().indexOf(registry.searchData.searchBoundary) == -1) {
                    // registry.view.hideControlButton();
                    registry.searchData.isSearchPro = false;
                }

            });
            searchInputDocument.keydown(function (event){
                let e = event || window.event || arguments.callee.caller.arguments[0];

                // 判断一个输入框的东西,如果如果按下的是删除,判断一下是不是"搜索模式"
                let keyword = $(event.target).val().trim();

                if(e.key == "Backspace" ) {
                    // 按的是删除键
                    if(event.target.value.endsWith(registry.searchData.searchBoundary)) {
                        // 取消默认事件
                        e.preventDefault();
                        return;
                    }
                }
                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){ // 回车选择的元素
                    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 = $($("#matchItems > li")[registry.searchData.pos-1]);
                // if(activeItem == null) return;
                // 设置活跃背景颜色
                let activeBackgroundColor = "#dee2e6";
                //let activeFontColor = "rgb(26, 13, 171)";
                // 如果是搜索项,可用别的颜色
                //if(activeItem.find("a").attr("href").indexOf("keyword") != -1) activeFontColor = "rgb(251,182,54)"
                activeItem.css({
                    "background":activeBackgroundColor
                })
                /*activeItem.find("a").css({
                     "color":activeFontColor
                 })*/

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

                if(e && e.keyCode==13 && activeItem.find("a").length > 0){ // 回车
                    // 点击当前活跃的项,点击
                    activeItem.find("a")[0].click();
                }
                // 取消冒泡
                e.stopPropagation();
                // 取消默认事件
                e.preventDefault();

            });
            // 将输入框的控制按钮设置可见性函数公开放注册表中
            registry.view.setButtonVisibility = function (buttonVisibility = false) {
                // registry.view.setButtonVisibility
                controlButton.css({
                    "display": buttonVisibility?"block":"none"
                })
            }
            // 向搜索事件(只会触发一个)中添加一个“NEW”搜索关键词
            registry.searchData.searchEven.event["NEW"] = function() {
                let showNewData = null;
                let activeSearchData = registry.searchData.data;
                // 如果当前注册表中全局搜索数据为空,使用缓存的数据
                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 = newItems.diff(activeSearchData,function(item) {
                        // 在这里只要desc与resource两个相等,那就是相等的
                        if(item == null || !( item instanceof Object && item.title != null)) return null;
                        return item.title[item.title.length-1]+item.desc+(item.resource??'').substring(0,20);
                    },0)
                }
                if(showNewData == null) return [];
                // 对数据进行排序
                showNewData.sort(function(item1, item2){return item2.expires - item1.expires});
                // 添加标签
                showNewData.map((item,index)=>{
                    let flag = index <= 0?"[最新一条]":registry.searchData.NEW_ITEMS_FLAG;
                    let dayNumber = registry.searchData.NEW_DATA_EXPIRE_DAY_NUM;
                    item.title = flag+item.title+ " | " + Math.floor( (new Date().getTime() - (item.expires - 1000*60*60*24*dayNumber) )/(1000*60*60*24) )+"天前"; //toDateString
                    return item;
                })
                return showNewData;
            }
            function searchUnitHandler(beforeData = [],keyword = "") {
                // 触发搜索事件
                for(let e of registry.searchData.onSearch) e(keyword);
                // 如果没有搜索内容,返回空数据
                keyword = keyword.trim().toUpperCase();
                if(keyword == "" || registry.searchData.data.length == 0 ) return [];

                // 看有没有观察者想要直接返回结果
                let showItemData = registry.searchData.searchEven.send(registry.searchData.keyword);
                let isSendSuccess = showItemData != null && showItemData instanceof Array;
                //if( isSendSuccess ) registry.searchData.isSearchAll = true;
                if( isSendSuccess ) return showItemData;
                // registry.searchData.isSearchAll = false;


                // 切割搜索内容以空格隔开,得到多个 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 = new Date().getTime();
                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.isSearchPro ) break;
                    */
                    // 将数据放在指定搜索层级数据上
                    if (
                        (( getPinyinByKeyword(dataItem.title,true).indexOf(pinyinKeyword) >= 0 || dataItem.title.toUpperCase().indexOf(keyword) >= 0 ) && searchLevelData[0].push(dataItem) )
                        || (( getPinyinByKeyword(dataItem.desc,true).indexOf(pinyinKeyword) >= 0 || dataItem.desc.toUpperCase().indexOf(keyword) >= 0) && searchLevelData[1].push(dataItem) )
                        || ( dataItem.resource.length <= 1000 && dataItem.resource.toUpperCase().indexOf(keyword) >= 0 && searchLevelData[2].push(dataItem) )
                    ) {
                        // 向满足条件的数据对象添加在总数据中的索引
                    }
                }
                let searchEnd = new Date().getTime();
                console.logout("搜索主逻辑耗时:"+(searchEnd - searchBegin ) +"ms");
                console.log("搜索:",searchLevelData,searchResultData)
                // 将【搜索项】放到上面
                function searchItemToTop(items) {
                    // 只有是搜索PRO模式 才干扰排序
                    if(!registry.searchData.isSearchPro) return;
                    let searchableItem = [];
                    let noSearchableItem = [];
                    let currentTop = 0;
                    for(let i = 0; i < items.length; i++) {
                        let item = items[i];
                        if(item.title.trim().indexOf(registry.searchData.searchProFlag) != -1 ) {
                            // 替换
                            let tmp = items[currentTop];
                            items[currentTop] = items[i];
                            items[i] = tmp;
                            currentTop++;
                        }
                    }
                }
                // 将上面层级数据放在总容器中
                searchResultData.push(...searchLevelData[0]);
                searchResultData.push(...searchLevelData[1]); searchItemToTop(searchResultData); // 搜索PRO模式时会干扰排序
                searchResultData.push(...searchLevelData[2]);
                if(searchUnits.length > 0 && searchUnits[searchUnits.length-1].trim() != ":") {
                    // 递归搜索
                    searchResultData = searchUnitHandler(searchResultData,searchUnits.join(" "));
                }
                return searchResultData;
            }
            // 给输入框加事件
            // 执行 debounce 函数返回新函数
            let handler = function (e) {
                let key = e.target.value.trim().split(/\s+/).reverse().join(" ");
                // 过滤
                // 数据出来的总数据
                let searchData = []
                // 递归搜索,根据空字符切换出来的多个keyword
                let searchResultData = searchUnitHandler(registry.searchData.data,key)
                // console.logout("搜索总数据:",searchResultData)
                // 放到视图上
                // 置空内容
                matchItems.html("")
                // 最多显示条数
                let show_item_number = registry.searchData.showSize ;
                function getFlag(searchResultItem) {
                    let resource = searchResultItem.resource.trim();
                    let isSketch = ! isUrlNoUrlText(resource);
                    let sketchFavicon = "";
                    if(isSketch) return `<img src="${sketchFavicon}"  />`;
                    function loaded() {
                        alert("loaded!")
                    }
                    return `<img src="${registry.searchData.getFaviconAPI(resource)}"  class="searchItem" />`
                }
                /*       $("#matchItems").on("load","li img", function (){
                   alert("加载完成!")
                })*/

                // 标题flag颜色选择器
                function titleFlagColorMatchHandler(flagValue) {
                    let vcObj = {
                        "系统项":"rgb(0,210,13)",
                        "非最佳":"#fbbc05",
                        "推荐":"#ea4335",
                        "装机必备":"#9933E5",
                        "好物":"rgb(247,61,3)",
                        "Adults only": "rgb(244,201,13)",
                        "可搜索":"#4c89fb",
                        "新":"#f70000",
                        "最新一条":"#f70000"
                    }
                    let resultFlagColor = "#5eb95e";
                    Object.getOwnPropertyNames(vcObj).forEach(function(key){
                        if(key == flagValue) {
                            resultFlagColor = vcObj[key];
                        }
                    });
                    return resultFlagColor;
                }

                // 标题内容处理程序
                function titleFlagHandler(title) {
                    if(!(/[\[]?/.test(title) && /[\]]?/.test(title))) return -1;
                    // 格式是:[flag]title(desc):resource 这种的
                    const regex = /(\[[^\[\]]*\])/gm;
                    let m;
                    let resultTitle = title;
                    while ((m = regex.exec(title)) !== null) {
                        // 这对于避免零宽度匹配的无限循环是必要的
                        if (m.index === regex.lastIndex) {
                            regex.lastIndex++;
                        }
                        let flag = m[0];
                        if(flag == null || flag.length == 0) return -1;
                        let flagCore = flag.substring(1,flag.length - 1);
                        // 正确提取
                        let style = `
                            background: ${titleFlagColorMatchHandler(flagCore)};
                            color: #fff;
                            height:30px;
                            line-height: 30px;
                            font-size: 10px;
                            padding: 3px 6px;
                            border-radius: 5px;
                            font-weight: 700;
                            box-sizing: border-box;
                            margin-right: 3.5px;
                        `;
                        resultTitle = resultTitle.toReplaceAll(flag,`<span style="${style}">${flagCore}</span>`);
                    }
                    return resultTitle;
                }
                // 标题前面带“#”的titleHandler
                function title井Handler(title) {
                    // 去掉flag
                    title = title.replace(/\[.*\]/,"").trim();
                    if(title.indexOf("#") == 0) {
                        let style = `text-decoration:line-through;color:#a8a8a8;`;
                        return `<span style="${style}">${title.replace("#","")}</span>`;
                    }
                    return -1;
                }

                function titleHandler(title) {
                    let titleHandlerFuns = registry.view.titleHandlerFuns;
                    for(let titleHandlerFun of titleHandlerFuns) {
                        let result = titleHandlerFun(title.trim());
                        if(result != -1) return result;
                    }
                    return title;

                }
                // 添加标题处理器 title井Handler (优化级较高)
                registry.view.titleHandlerFuns.push(title井Handler);
                // 添加标题处理器 titleFlagHandler
                registry.view.titleHandlerFuns.push(titleFlagHandler);

                for(let searchResultItem of searchResultData ) {
                    // 限制条数
                    if(show_item_number-- <= 0 && !registry.searchData.isSearchAll) {
                        break;
                    }
                    // 将数据放入局部容器中
                    searchData.push(searchResultItem)
                    let isSketch = !isUrlNoUrlText(searchResultItem.resource);//  searchResultItem.resource.trim().toUpperCase().indexOf("HTTP") != 0;
                    // 将符合的数据装载到视图  //
                    let item = `
                    <li class="resultItem">${getFlag(searchResultItem)}
                        <a href="${isSketch?'':searchResultItem.resource}" target="_blank" title="${searchResultItem.desc}" index="${searchResultItem.index}" >
                             ${titleHandler(searchResultItem.title)}
                             <span class="desc">(${searchResultItem.desc})</span>
                        </a>
                    </li>`
                    matchItems.html(matchItems.html() + item);



                }
                let loadErrorFlagIcon = "";
                //let loadErrorFlagIcon = "";
                // 给刚才添加的img添加事件
                for(let imgObj of $("#matchItems").find('img')) {
                    // 加载完成事件,去除加载背景
                    imgObj.onload = function(e) {
                        $(e.target).css({
                            "background": "rgba(0,0,0,0)"
                        })
                    }
                    // 加载失败,设置自定义失败的本地图片
                    imgObj.onerror = function(e,a,b,c) {
                        $(e.target).attr("src",loadErrorFlagIcon)
                    }
                }

                // 隐藏文本显示视图
                textShow.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;
                // 设置li样式
                $("#matchResult li").css({
                    "line-height": "30px",
                    "height": "30px",
                    "color": "#0088cc",
                    "list-style": "none", //decimal
                    "width":"100%",
                    "margin":"0px",
                    "display":"flex",
                    "justify-content":"left",
                    "align-items":"center",
                    "padding":"0px",
                    "margin":"0px"

                })
                $("#matchResult li>a").css({
                    "display":"inline-block",
                    "font-size":"15px",
                    "color": "#1a0dab",
                    "text-decoration":"none",
                    "text-align":"left",
                    "overflow":"hidden", //超出的文本隐藏
                    "text-overflow":"ellipsis", //溢出用省略号显示
                    "white-space":"nowrap", //溢出不换行
                    "cursor":"pointer",
                    "font-weight":"400",
                    "background":"rgb(255 255 255 / 0%)"
                })
                $("#matchResult .desc").css({
                    "color":"#4d5156"
                })
                $("#matchResult img").css({
                    "display": "inline-block",
                    "width": "22px",
                    "height":"22px",
                    "margin":"0px 7px 0px 5px",
                    "box-shadow": "0 0 2px rgba(0,0,0,0.5)",
                    "border-radius": "30%",
                    "box-sizing": " border-box",
                    "border":"3px solid #fff0",
                    "flex-shrink":"0" // 当容量不够时,不压缩图片的大小

                })
            }


            // 简述内容转markdown前
            function sketchResourceToHtmlBefore(txtStr = "") {
                // 1、“换行”转无意义中间值
                txtStr = txtStr.replace(/<\s*br\s*\/\s*>/gm,"?br?"); // 单行简述下的换行,注意要在"<",">"转意前就要做了,注意顺序
                // 2、特殊字符 转无意义中间值
                txtStr = txtStr.replace(/</gm,"?lt?").replace(/>/gm,"?gt?").replace(/"/gm,"?quot?").replace(/'/gm,"?#39?");
                return txtStr;
            }
            //简述内容转markdown
            function sketchResourceToHtmlAfter(txtStr = "") {
                // 1、链接变超链接,这里必需要使用“先匹配再替换”
                const regexParam = /[^("?>]\s*(https?:\/\/[^\s()()\[\]<>"`]+)/gm;
                let m;
                let textStrClone = txtStr;
                while ((m = regexParam.exec(textStrClone)) !== null) {
                    // 这对于避免零宽度匹配的无限循环是必要的
                    if (m.index === regexParam.lastIndex) {
                        regexParam.lastIndex++;
                    }
                    let match = m[0];
                    // 为简讯内容的url添加可链接
                    const regex = /(https?:\/\/[^\s()()\[\] `]+)/gm;
                    const subst = `<a href="$1" target="_blank">$1</a>`;
                    // 被替换的值将包含在结果变量中
                    let aTab = match.replace(regex, subst);
                    txtStr = txtStr.replace(match, aTab);
                }
                // 2、无意义中间值 转有意符
                function revert(text) {
                    let obj = {
                        "?lt?":"&lt;",
                        "?gt?":"&gt;",
                        "?quot?":"&quot;",
                        "?#39?":"&#39;",
                        "?br?":"<br />"
                    }
                    for(let key in obj) {
                        text = text.toReplaceAll(key,obj[key]);
                    }
                    return text;
                }
                txtStr = revert(txtStr);
                return txtStr;
            }
            $("#matchItems").on("click","li > a",function(e) {
                // 取消默认事件,全部都是手动操作
                e.preventDefault();
                // 取消冒泡
                window.event? window.event.cancelBubble = true : e.stopPropagation();
                // 设置为阅读模式
                // $("#my_search_input").val(":read");
                // 获取当前结果在搜索数组中的索引
                let dataIndex = parseInt($(e.target).attr("index"));
                let itemData = registry.searchData.data[dataIndex];
                // 如果是简述搜索信息,那就取消a标签的默认跳转事件
                if( ! isUrlNoUrlText(itemData.resource) ) {
                    // 取消默认事件
                    //e.preventDefault();
                    matchResult.css({
                        "display": "none"
                    })
                    textShow.css({
                        "display":"block"
                    })
                    textShow.html("<span style='color:red'>标题</span>:"+itemData.title+"<br />"+ "<span style='color:red'>描述:</span>"+itemData.desc+"<br />"+"<span style='color:red'>简述内容:</span><br />"+sketchResourceToHtmlAfter(converter.makeHtml(sketchResourceToHtmlBefore(itemData.resource))) );
                    textShow.find("img").css({"width":"100%"})
                    /*使用code代码块样式*/
                    document.querySelectorAll('#text_show pre code').forEach((el) => {
                        hljs.highlightElement(el);
                    });
                    return;
                }
                // 隐藏视图
                registry.view.viewVisibilityController(false)
                // 解析URL(主要将keyword补充上去)
                let targetObj = e.target;
                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.searchBoundary).length < 2
                   || registry.searchData.keyword.split(registry.searchData.searchBoundary)[1].trim() == "" ) {
                    url = registry.searchData.clearUrlSearchTemplate(initUrl);
                }
                // 跳转(url如果有模板,可能已经去掉模板,取决于是“搜索模式”)
                window.open(url);

            })
            //registry.searchData.searchHandle = handler;
            const refresh = debounce(handler, 460)
            // 第一次触发 scroll 执行一次 fn,后续只有在停止滑动 1 秒后才执行函数 fn
            searchBox.on('input', refresh)

            // 初始化后将isInitializedView变量设置为true
            isInitializedView = true;
        }
        let hideView = function () {

            // 隐藏视图
            // 如果视图还没有初始化,直接退出
            if (!isInitializedView) return;
            // 如果正在查看查看“简讯”,先退出简讯
            if($("#text_show").css("display")=="block") {
                // 让简讯隐藏
                $("#text_show").css({"display":"none"})
                // 让搜索结果显示
                $("#matchResult").css({
                    "display":"block",
                    "overflow": "hidden",
                })
                return;
            }
            // 让视图隐藏
            viewDocument.style.display = "none";
            // 将输入框内容置空,在置空前将值备份,好让未好得及的操作它
            searchInputDocument.val("")
            // 将之前搜索结果置空
            matchItems.html("")
            // 隐藏文本显示视图
            textShow.css({
                "display":"none"
            })
            // 让搜索结果显示
            matchResult.css({
                "display":"none"
            })
        }
        let showView = function () {
            // 让视图可见
            viewDocument.style.display = "block";
            //聚焦
            searchInputDocument.focus()
            // 当输入框失去焦点时,隐藏视图
            searchInputDocument.blur(function() {
                setTimeout(function(){
                    // 判断输入框的内容是不是":debug"或是否正处于阅读模式,如果是,不隐藏
                    if(isInstructions(searchInputDocument.val(),"debug") || isInstructions(searchInputDocument.val(),"read")) return;
                    // 当前视图是否在展示数据,如搜索结果,简述内容?如果在展示不隐藏
                    let isNotExhibition = (($("#matchResult").css("display") == "none" || $("#matchItems > li").length == 0 ) && ($("#text_show").css("display") == "none" || $("#text_show").text().trim() == "") );
                    if(!isNotExhibition || registry.view.menuActive ) return;
                    registry.view.viewVisibilityController(false);
                },registry.view.delayedHideTime)
            });
        }

        // 返回给外界控制视图显示与隐藏
        return function (isSetViewVisibility) {
            if (isSetViewVisibility) {
                // 让视图可见 >>>
                // 如果还没初始化先初始化   // 初始化数据 initData();
                if (!isInitializedView) {
                    // 初始化视图
                    initView();
                    // 初始化数据
                    // initData();
                }
                // 让视图可见
                showView();
            } else {
                // 隐藏视图 >>>
                if (isInitializedView) hideView();
            }
        }
    })();
    // 触发策略——快捷键
    let useKeyTrigger = function (viewVisibilityController) {
        let isFirstShow = true;
        // 将视图与触发策略绑定
        triggerAndEvent("ctrl+alt+s", function () {
            // 让视图可见
            viewVisibilityController(true);
            // 触发视图首次显示事件
            if(isFirstShow) {
                for(let e of registry.view.onViewFirstShow) e();
                isFirstShow = false;
            }
        })
        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);
        }
    })();

    // 打开视图进行配置
    // 显示配置视图
    // 是否显示进度 - 进度控制
    GM_registerMenuCommand("订阅管理",function() {
        showConfigView();
    });
    GM_registerMenuCommand("清理缓存",function() {
        cache.remove(registry.searchData.SEARCH_DATA_KEY);
    });

    // 显示配置规则视图

    function giveFlagsStatus(flagsOfData,userUnfollowList) {
        // 赋予flags一个是否选中状态
        // 将 userUnfollowList 转为以key为userUnfollowList的item.name值是Item的方便检索
        let userUnfollowMap = userUnfollowList.reduce(function(result, item) {
            result[item] = '';
            return result;
        }, {});
        flagsOfData.forEach(item=>{
            if(userUnfollowMap[item.name] != null ) {
                // 默认都是选中状态,如果item在userUnfollowList上将此flag状态改为未选中状态
                item.status = 0;
            }
        })
        return flagsOfData;
    }
    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;
        // 显示视图
        var configViewContainer = document.createElement("div");
        // 用户维护的取消关注标签列表
        let userUnfollowList = cache.get(registry.searchData.USER_UNFOLLOW_LIST_CACHE_KEY)?? registry.searchData.USER_DEFAULT_UNFOLLOW;
        // 当前数据所有的标签
        let flagsOfData = cache.get(registry.searchData.DATA_ITEM_FLAGS_CACHE_KEY);
        // 使用 userUnfollowList 给 flagsOfData中的标签一个是否选中状态,在userUnfollowList中不选中,不在选中,添加一个属性到flagsOfData用boolean表达
        flagsOfData = giveFlagsStatus(flagsOfData,userUnfollowList);
        // 生成多选框html
        let flagsCheckboxHtml = "";
        flagsOfData.forEach(item=>{
            flagsCheckboxHtml += `
               <div>
                   <input type="checkbox" id="${item.name}" name="_flagsCheckBox" value="${item.name}" ${item.status==1?'checked':''} >
                   <label for="${item.name}">${item.name} (${item.count})</label>
               </div>
            `
        })
        configViewContainer.style=`
            width: 500px;
            max-height: 100%;
            max-width:100%; background:pink;
            position: fixed;right: 0px; top: 0px;
            z-index:10000;
            padding: 20px;
            box-sizing: border-box;
            border-radius: 14px;
            text-align: left;
        `
        configViewContainer.innerHTML = `
           <div class="_topController">
              <span id="topController_close">X</span>
           </div>
           <p class="_title">订阅总览:</p>
           <textarea id="all_subscribe" ></textarea>
           <p class="_title">关注标签(<a title="保存后下次重新网络加载数据生效" style="cursor: pointer;">说明</a>):</p>
           <div class="flagsCheckBoxDiv">
               ${flagsCheckboxHtml}
           </div>
           <button id="subscribe_save" >保存并应用</button>
        `;

        // 设置样式
        document.body.appendChild(configViewContainer);
        $("._title").css({
            "margin-bottom": "10px",
            "font-size":"16px"
        })
        $("._topController").css({
            "width": "100%",
            "position": "absolute",
            "top": "0px",
            "right": "0px",
            "text-align": "right",
            "padding": "15px",
            "cursor": "pointer"
        })
        $(".flagsCheckBoxDiv > div").css({
            "width":"32%",
            "display": "inline-block",
            "margin": "0px",
            "padding": "0px",
            // 超出隐藏
            "overflow": "hidden",
            "text-overflow":"ellipsis",
            "white-space": "nowrap"
        })
        document.getElementById("all_subscribe").style="width:100%;height:150px";
        document.getElementById("subscribe_save").style=" margin-top: 20px; border: none; border-radius: 3px; padding: 4px 20px; cursor: pointer;";
        // 回显
        document.getElementById("all_subscribe").value = getSubscribe();
        // 保存
        function configViewClose() {
            configViewContainer.remove();
        }
        document.getElementById("subscribe_save").onclick=function() {
            // 保存用户选择的关注标签(维护数据)
            // 获取所有多选框元素
            var checkboxes = document.querySelectorAll(".flagsCheckBoxDiv input");
            // 初始化已选中和未选中的数组
            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 = document.getElementById("all_subscribe").value;
            let validCount = editSubscribe(allSubscribe);
            // 清除视图
            configViewClose();
            alert("保存配置成功!有效订阅数:"+validCount);
        }
        // 关闭
        $("#topController_close").click(configViewClose)
    }

})();