我的搜索

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

目前為 2023-04-09 提交的版本,檢視 最新版本

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

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

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name         我的搜索
// @namespace    http://tampermonkey.net/
// @version      4.7.0
// @description  打造订阅式搜索,让我的搜索,只搜精品!
// @license MIT
// @author       zhuangjie
// @match      *://*/*
// @exclude  http://127.0.0.1*
// @exclude  http://localhost*
// @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

// ==/UserScript==

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


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




    // 重写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);
    }
    // 检查网站是否可用
    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 "";
        }
    }



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


    //registry.searchData.SEARCH_DATA_KEY
    //registry.view.onViewFirstShow
    let registry = {// registry.view.delayedHideTime
        view: {
            viewVisibilityController: () => { ERROR.tell("视图未初始化,但你使用了它的未初始化的注册表信息!") },
            viewDocument: null,
            setButtonVisibility: () => { ERROR.tell("按钮未初始化!") },
            titleHandlerFuns: [],
            onViewFirstShow: [],
            menuActive: false,
            // 视图延时隐藏时间
            delayedHideTime: 200
        },
        searchData: { //registry.searchData.isSearchAll  registry.searchData.NEW_DATA_EXPIRE_DAY_NUM
            data: [],
            // 旧的新数据
            OLD_SEARCH_DATA_KEY: "OLD_SEARCH_DATAS_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, // 60分钟重新检测一下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 = {}

    /*cache.remove(registry.searchData.SEARCH_DATA_KEY)
    cache.remove(registry.searchData.OLD_SEARCH_DATA_KEY)
    cache.remove(registry.searchData.SEARCH_NEW_ITEMS_KEY)*/

    // 判断是否只是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 asyncFun(fun,time = 20) {
       setTimeout(()=>{
          fun();
       },time)
    }
    // 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;
    };
    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)
        asyncFun(()=>{
            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 CDN加速包装器
    function cdnPack(githubResourceUrl) {
       let githubUrlFlag = "raw.githubusercontent.com";
       // 如何不满足github url ,不加速
       if(githubResourceUrl.indexOf(githubUrlFlag) < 0) return githubResourceUrl;
       return "https://proxy.zyun.vip/"+githubResourceUrl;
    }

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


    // 使用责任链模式——对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) ;
            $.ajax({
                url: cdnPack(dataSourceUrl+"?t="+new Date().getTime()),
                success: function (result) {
                    resolve(result)
                }
            });
        });
    }
    // 下面的 dataSourceHandle 函数
    let globalFetchFun = [];
    // tis处理队列
    let waitQueue = [];
    registry.searchData.dataAddBeforeHandler.handlers.push(function (items) {
       for(let searchItem of items) {
          let url = searchItem.resource;
          let isSearchableItem = /\[\[[^\[\]]+keyword[^\[\]]+\]\]/.test(url);
          if(isSearchableItem) searchItem.title = registry.searchData.searchProFlag+searchItem.title;
       }
       return items;
    })
    // 将解析出来的部分数据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"));
          // 触发“新数据块处理完成事件”
          (function(){
              // 排序(根据配置在obj中的权重)
              registry.searchData.onNewDataBlockHandleAfter.sort((obj1,obj2)=>obj2.weights-obj1.weights);
              // 触发
              for(let obj of registry.searchData.onNewDataBlockHandleAfter) obj.fun(newItems);
          })();
          
       }
    }
    // 转义与恢复,数据进行解析前进行转义,解析后恢复——比如文本中出现“/”,就会出现: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 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*1) // 一个小时
            //expire: new Date().getTime() + (2000) // 测试,一秒过期
        })
    }
    // 检查是否已经执行初始化
    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;
    }
    // 【数据初始化主函数】
    let isInitialized = false;
    function dataInitFun() {
        // 检查是否已经执行初始化
        if(isInitialized) return;
        // 设置为已初始化,保障只初始化一次,如果不能保障,同时初始化时,出现重复数据
        isInitialized = true;
        // 从缓存中获取数据,判断是否还有效
        const SEARCH_DATA_KEY = registry.searchData.SEARCH_DATA_KEY;
        // cache.remove(SEARCH_DATA_KEY)
        let dataBox = cache.get(registry.searchData.SEARCH_DATA_KEY);

        if(dataBox != null) {
            // 只要数据不为空,不管是否过期,先用着,直接将之前缓存的数据放在全局搜索数据容器中
            registry.searchData.data = dataBox.data
            // 缓存信息不为空,深入判断是否使用缓存的数据
            let dataExpireTime = dataBox.expire;
            let currentTime = new Date().getTime();
            // console.logout("缓存的数据:",dataBox.data)
            // 数据多大时,才开启缓存
            const TRIGGER_CACHE_DATA_LENGTH = 300;
            // 判断是否有效,有效的话放到全局容器中
            let isValid = (dataExpireTime != null && dataExpireTime > currentTime && dataBox.data != null && dataBox.data.length > 0);
            // 如果网站比较特殊,忽略数据过期时间
            if(!isValid && window.location.host.toUpperCase().indexOf("GITHUB.COM") >= 0) {
               isValid = true;
            }
            // 如果数据过期,或数据量不满足缓存大小,会去请求数据
            if(isValid && dataBox.data.length >= TRIGGER_CACHE_DATA_LENGTH ) {
                console.logout("本次从缓冲中获取, 数据有效期还有"+parseInt((dataExpireTime - currentTime)/1000/60)+"分钟!", dataBox.data)
                return
            };
        }
        // 内部将使用递归,解析出信息
        dataSourceHandle(dataSources,null);
        // 监听数据改变
        registry.searchData.onNewDataBlockHandleAfter.push({weights:1000 ,fun:cacheSearchData})
    }
    // 该函数作用是为了防止一个页面多次加载数据,导致页面一直在加载
    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 = debounce(()=>{
        console.logout("开始筛选~");
        // 新数据加载完成-进行数据对比
        // 旧数据,也就是上一次数据,用于与本次比较,得出新添加数据
        let oldSearchData = cache.get(registry.searchData.OLD_SEARCH_DATA_KEY)??[];
        // 当前最新数据,用于搜索
        let currentSearchData = registry.searchData.data;
        // 当前时间戳
        let currentTime = new Date().getTime();
        // 如果脚本还没有初始化,需要初始化-也就是放开第一次加载数据,否则都是新数据
        if(cache.get("SCRIPT_INITIALIZED") == null ) {
           setTimeout(()=>cache.set("SCRIPT_INITIALIZED",true),500);
           return;
        }
        // 数组差异-获取不同的元素比较的基值
        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+(item.resource??'').substring(0,20);
            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);
        }

        // 总新增去重 (标记 - 过滤标记的 )
        for(let i = 0; i< newDataItems.length; i++ ) {
           let item1 = newDataItems[i];
           for(let j = i+1; j< newDataItems.length; j++ ) {
               let item2 = newDataItems[j];
               if(item1.title == item2.title && item1.resource == item2.resource) {
                  newDataItems[i] = null;
                  break;
               }
           }
        }
        newDataItems = newDataItems.filter((item, index) => item != null); // 与“总新增去重”顺序不能乱!!!

        console.logout("总新增:", JSON.parse(JSON.stringify(newDataItems)));
        // 如果过多,不加入最新
        if( !(newDataItems.length < 10 || newDataItems.length/currentSearchData.length < 0.4) ) {
            // 不加入新数据,但需要维护旧新数据在现在新数据的索引
            newDataItems = oldNewItems;
        }
        // 将 index 给 newDataItems ,不然new中的我们选择与实际选择的不一致问题 !
        for(let nItem of newDataItems) {
            for(let cItem of currentSearchData) {
                if(nItem.title === cItem.title && nItem.desc === cItem.desc) {
                    nItem.index = cItem.index;
                    break;
                }
            }
        }
        cache.set(registry.searchData.SEARCH_NEW_ITEMS_KEY,newDataItems);
        // 改变全局搜索并持久化
        let cacheData = cache.get(registry.searchData.SEARCH_DATA_KEY);
        for(let item of currentSearchData) {
            for(let nItem of newDataItems) {
                if(item.title == nItem.title && item.resource == nItem.resource) {
                    item.title = registry.searchData.NEW_ITEMS_FLAG+item.title;
                    break;
                }
            }
        }
        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??[]);
    }, 1300);

    registry.searchData.onNewDataBlockHandleAfter.push({weights:1 ,fun:refreshNewData});
    // 模块二
    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;
                }
            });
            // 初始化搜索数据
            dataInitFun();

            // 在搜索的结果集中上下选择移动然后回车(相当点击)
            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 result = null;
                let data = registry.searchData.data;
                let tmpData = cache.get(registry.searchData.SEARCH_DATA_KEY);
                if(data == null ) data = tmpData != null && tmpData.data ;
                if(data == null ) data = cache.get(registry.searchData.SEARCH_DATA_KEY+BACK);
                // 只展示 newItems 数据中data也存在的项
                let newItems = cache.get(registry.searchData.SEARCH_NEW_ITEMS_KEY)??[];
                console.log("==>",newItems,data)
                if(newItems.length > 0 && data.length > 0) {
                   result = newItems.diff(data,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(result == null) return [];
                // 对数据进行排序
                result.sort(function(item1, item2){return item2.expires - item1.expires});
                // 添加标签

                result.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 result;
            }
            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);
                // 搜索操作
                let pinyinKeyword = keyword.toPinyin();
                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 (
                           (( (dataItem.title.toPinyin(true)??"").indexOf(pinyinKeyword) >= 0 || dataItem.title.indexOf(keyword) >= 0 ) && searchLevelData[0].push(dataItem) )
                        || (( (dataItem.desc.toPinyin(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");
                // 将【搜索项】放到上面
                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": "21px",
                 "height":"21px",
                 "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) {
                // 设置为阅读模式
                // $("#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;
                }
                // 取消冒泡
                window.event? window.event.cancelBubble = true : e.stopPropagation();
                // 隐藏视图
                registry.view.viewVisibilityController(false)
                // 解析URL(主要将keyword补充上去)
                let targetObj = e.target;
                const initUrl = $(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存在模板
                if(temNum > 0 ) {

                    window.open(url);
                    // 取消默认事件
                   e.preventDefault();
                   return;
                }

                // 否则是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 showConfigView() {
        if($("#subscribe_save")[0] != null) return;
        // 显示视图
        var configViewContainer = document.createElement("div");
        configViewContainer.style=`
            width:450px; background:pink;
            position: fixed;right: 0px; top: 0px;
            z-index:10000;
            padding: 20px;
            border-radius: 14px;
        `
        configViewContainer.innerHTML = `
           <p id="title">订阅总览:</p>
           <textarea id="all_subscribe" ></textarea>
           <button id="subscribe_save" >保存并应用</button>
        `;

        // 设置样式
        document.body.appendChild(configViewContainer);
        document.getElementById("title").style="margin-bottom: 10px; font-size: 16px;";
        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();
        // 保存
        document.getElementById("subscribe_save").onclick=function() {
            // 保存到对象
           let allSubscribe = document.getElementById("all_subscribe").value;
           let validCount = editSubscribe(allSubscribe);
           // 清除视图
           configViewContainer.remove();
           alert("保存配置成功!有效订阅数:"+validCount);
        }
    }

})();