// ==UserScript== // @name 我的搜索 // @namespace http://tampermonkey.net/ // @version 7.6.0 // @description 打造订阅式搜索,让我的搜索,只搜精品! // @license MIT // @author zhuangjie // @exclude http://127.0.0.1* // @exclude http://localhost* // @match *://*/* // @exclude http://192.168.* // @icon  // @require https://cdn.jsdelivr.net/npm/jquery@3.6.2/dist/jquery.min.js // @require https://cdn.jsdelivr.net/npm/showdown@2.1.0/dist/showdown.min.js // @resource markdown-css https://cdnjs.cloudflare.com/ajax/libs/github-markdown-css/5.8.1/github-markdown.min.css // @require https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.9.0/highlight.min.js // @resource code-css https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.9.0/styles/github.min.css // @require https://update.greasyfork.org/scripts/530877/1560004/ms-pinyin.js // @require https://update.greasyfork.org/scripts/501646/1429885/string-overlap-matching-degree.js // @noframes // @grant window.onurlchange // @grant GM_setValue // @grant GM_xmlhttpRequest // @connect * // @grant GM_getValue // @grant GM_addStyle // @grant GM_getResourceText // @grant GM_getResourceURL // @grant GM_deleteValue // @grant GM_registerMenuCommand // @grant GM_info // @downloadURL https://update.greasyfork.icu/scripts/457020/%E6%88%91%E7%9A%84%E6%90%9C%E7%B4%A2.user.js // @updateURL https://update.greasyfork.icu/scripts/457020/%E6%88%91%E7%9A%84%E6%90%9C%E7%B4%A2.meta.js // ==/UserScript== // 参数tis: actualWindow 是真实的window,而如果在下面脚本内访问window则不是,是沙盒的XPCNativeWrapper对象,详情https://blog.rxliuli.com/p/e55a67646bf546b3900ce270a6fbc6ca/ (function(actualWindow) { 'use strict'; // 模块一:快捷键触发某一事件 (属于触发策略组) // 模块二:搜索视图(显示与隐藏)(属于搜索视图组) // 模块三:触发策略组触发策略触发搜索视图组视图 // 模块四:根据用户提供的策略(策略属于数据生成策略组)生成搜索项的数据库 // 模块五:视图接入数据库 // 判断当前是否在iframe里面, function currentIsIframe() { if (self.frameElement && self.frameElement.tagName == "IFRAME") return true; if (window.frames.length != parent.frames.length) return true; if (self != top) return true; return false; } // 如果当前是ifrae,结束脚本执行 let MY_SEARCH_SCRIPT_VIEW_SHOW_EVENT = "MY_SEARCH_SCRIPT_VIEW_SHOW_EVENT"; if(currentIsIframe()) { // 虽然iframe不能初始化脚本,但可以作为父窗口的事件触发源 triggerAndEvent("ctrl+alt+s", function () { // 通知主容器显示搜索框 window.parent.postMessage(MY_SEARCH_SCRIPT_VIEW_SHOW_EVENT, '*'); }) // 结束脚本执行 return; } // css resource加载器 function cssLoad(prefix,css = "",{isResourceName = false,replacePrefix}) { if(isResourceName) css = GM_getResourceText(css); if(replacePrefix && prefix) { css = css.toReplaceAll(replacePrefix,prefix); }else if(prefix) { // 对css原始内容引入前缀 css = `.${prefix} { ${css} }`; } GM_addStyle(css); return prefix; } // 正则捕获 function captureRegEx(regex, text) { let m; let result = []; // 一组一组 [[],[],...] regex.lastIndex = 0; // 重置lastIndex while ((m = regex.exec(text)) !== null) { let group = []; group.push(...m); if(group.length != 0) result.push(group); } return result; } // 滚动到目标文本 可以指定容器 function scrollToText(text, index = 0, container = document.body) { // 创建高亮样式(只需执行一次) if (!document.getElementById('highlight-style')) { const style = document.createElement('style'); style.id = 'highlight-style'; style.textContent = `.highlight-text { border-bottom: 2px solid red !important;color:red; }`; document.head.appendChild(style); } // 处理容器参数 const containerElement = typeof container === 'string' ? document.querySelector(container) : container; if (!containerElement) { console.error('Container not found:', container); return; } // 清理旧高亮 Array.from(containerElement.getElementsByClassName('highlight-text')) .forEach(span => { const textNode = document.createTextNode(span.textContent); span.parentNode.replaceChild(textNode, span); }); if (!text) return; // 获取所有文本节点 const textNodes = []; const walker = document.createTreeWalker( containerElement, NodeFilter.SHOW_TEXT, { acceptNode: () => NodeFilter.FILTER_ACCEPT } ); while (walker.nextNode()) textNodes.push(walker.currentNode); // 查找所有匹配项 const matches = []; const searchLen = text.length; for (const node of textNodes) { const nodeText = node.nodeValue; let pos = 0; while (pos <= nodeText.length - searchLen) { const idx = nodeText.indexOf(text, pos); if (idx === -1) break; matches.push({ node, start: idx, end: idx + searchLen }); pos = idx + 1; // 允许重叠匹配 } } // 验证索引 if (index < 0 || index >= matches.length) { console.error(`Index ${index} out of range (0-${matches.length - 1})`); return; } // 处理目标匹配项 const { node, start, end } = matches[index]; const middle = node.splitText(start); const after = middle.splitText(end - start); const span = document.createElement('span'); span.className = 'highlight-text'; span.textContent = middle.nodeValue; middle.parentNode.replaceChild(span, middle); // 滚动到目标位置 span.scrollIntoView({ behavior: 'smooth', block: 'center', inline: 'nearest' }); } // 重写console.log方法 let originalLog = console.log; console.logout = function() { const prefix = "[我的搜索log]>>> "; const args = [prefix].concat(Array.from(arguments)); originalLog.apply(console, args); } // markdown转html 转换器 【1】 // 更多配置项:https://github.com/showdownjs/showdown const converter = new showdown.Converter({ // 将换行符解析为
simpleLineBreaks:true, // 新窗口打开链接 openLinksInNewWindow: true, metadata:true, // 不允许下划线变斜体 literalMidWordUnderscores: true, // 识别md表格 tables: true, // www.baidu.com 会识别为链接 simplifiedAutoLink: true }); function md2html(rawText) { // 方案二: return marked.parse(rawText); // @require https://cdnjs.cloudflare.com/ajax/libs/marked/9.0.2/marked.min.js return converter.makeHtml( rawText ) } // 提取URL根域名 function getUrlRoot(url,isRemovePrefix = true,isRemoveSuffix = true) { if(! (typeof url == "string" || url.length >= 3)) return url; // 可处理 // 判断是否有前缀 let prefix = ""; let root = ""; let suffix = ""; // 提取前缀 if(url.indexOf("://") != -1) { // 存在前缀 let prefixSplitArr = url.split("://") prefix = prefixSplitArr[0]; url = prefixSplitArr[1]; } // 提取root 和suffix if(url.indexOf("/") != -1) { let twoLevelIndex = url.indexOf("/") root = url.substr(0,twoLevelIndex); suffix = url.substr(twoLevelIndex,url.length-1); }else { root = url; suffix = ""; } return ((!isRemovePrefix && prefix != "")?(prefix+"://"):"") + root + (isRemoveSuffix?"":suffix); } // 解析出http url 结构 function parseUrl(url) { const regex = /(https?:|)\/\/([^\/]*|[^\/]*)(\/[^\s\?]*|)(\??[^\s]*|)/; const matches = regex.exec(url); if (matches) { const protocol = matches[1]; const domain = matches[2]; const path = matches[3]; const params = matches[4]; const rootUrl = protocol+"//"+domain const rawUrl = url; return {protocol,domain,path,params,rootUrl,rawUrl} } return null; } function isHttpUrl(url = "") { url = url.trim(); return /^https?:\/\/.+/i.test(url) } // 检查网站是否可用 function checkUsability(templateUrl,isStopCheck = false) { return new Promise(function (resolve, reject) { // 判断是否要检查 if(isStopCheck) { reject(null); return; } var img=document.createElement("img"); img.src = templateUrl.fillByObj(parseUrl("https://www.baidu.com")); img.style= "display:none;"; img.onerror = function(e) { setTimeout(function() {img.remove();},20) reject(null); } img.onload = function(e) { setTimeout(function() {img.remove();},20) resolve(templateUrl); } document.body.appendChild(img); }); } // 数据缓存器 let cache = { prefix: "", get(key) { return GM_getValue(this.prefix+key); }, set(key,value) { this.remove(this.prefix+key); GM_setValue(this.prefix+key,value); }, jGet(key) { let value = GM_getValue(this.prefix+key); if( value == null) return value; return JSON.parse(value); }, jSet(key,value) { value = JSON.stringify(value) GM_setValue(this.prefix+key,value); }, remove(key) { GM_deleteValue(this.prefix+key); }, cookieSet(cname,cvalue,exdays) { var d = new Date(); d.setTime(d.getTime()+exdays); var expires = "expires="+d.toGMTString(); document.cookie = cname + "=" + cvalue + "; " + expires; }, cookieGet(cname) { var name = cname + "="; var ca = document.cookie.split(';'); for(var i=0; idata}) { if(chain == null ) throw new Error("[ERROR]责任链对象: 你添加了一个null Chain!") if(chain.weight == undefined || chain.fun == undefined) throw new Error("[ERROR]责任链对象: 你传入的Chain是无效的!") this.chains.push(chain) }, trigger(baton) { // 排序,通过weight从高到低 this.chains = this.chains.sort((a, b)=>b.weight - a.weight); // 开始执行 let _baton = baton; let ref = { stop: false } for(let chain of this.chains) { if( ref.stop) { break; } _baton = chain.fun(_baton,ref) } return _baton; } } } // 请求包装 function request(type, url, { query, body,header = {},config ={}} = {}) { return new Promise(function(resolve, reject) { var formData = new FormData(); var isFormData = false; if (body) { for (var key in body) { if (body[key] instanceof File) { formData.append(key, body[key]); isFormData = true; } else { formData.append(key, JSON.stringify(body[key])); } } } var ajaxOptions = { ...config, url: url + (query ? ("?" + $.param(query)) : ""), method: type, headers: header, success: function(response) { resolve(response); }, error: function(jqXHR, textStatus, errorThrown) { reject(errorThrown); } }; if (isFormData) { ajaxOptions.data = formData; ajaxOptions.processData = false; ajaxOptions.contentType = false; } else { ajaxOptions.data = JSON.stringify(body); } config?.crossDomain // ajax兼容 ? GM_xmlhttpRequest({ ...ajaxOptions, onload: (xhr)=>{ ajaxOptions.success(xhr.responseText,xhr) }, onerror: ajaxOptions.error, ontimeout: ajaxOptions.error }) : $.ajax(ajaxOptions); }); } // 正则字符串匹配目录字符串-匹配工具 function isMatch(regexStr, text) { // 创建正则表达式对象 let regex = new RegExp(regexStr); // 使用 test 方法测试字符串是否匹配 return regex.test(text); } // `视图渲染完成`后调用 function waitViewRenderingComplete(callback) { // 这里模拟的是当下次渲染完成后执行 setTimeout(callback,30) } // 结构化的css 转 平铺的css function flattenCSS(cssObject, parentSelector = '') { let result = ''; for (const [selector, rules] of Object.entries(cssObject)) { if (typeof rules === 'object') { // 如果 rules 是对象,说明有嵌套,需要递归处理 const fullSelector = parentSelector ? `${parentSelector} ${selector}` : selector; result += flattenCSS(rules, fullSelector); } else { // 如果 rules 不是对象,说明是样式规则 const fullSelector = parentSelector ? `${parentSelector} { ${selector}: ${rules}; }\n` : ''; result += fullSelector; } } return result; } // ref引用变量 function ref(initValue = null) { return {value: initValue} } // ==偏业务工具函数== // 使用责任链模式——对pageText进行操作的工具 class PageTextHandleChains { pageText = ""; constructor(pageText = "") { this.pageText = pageText; } setPageText(newPageText) { this.pageText = newPageText; } getPageText() { return this.pageText; } // 解析双标签-获取指定标签下指定属性下的值 parseDoubleTab(tabName,attrName) { // 返回指定标签下指定属性下的值 const regex = RegExp(`<\\s*${tabName}[^<>]*\\s*${attrName}="([^<>]*)"\\s*>([\\s\\S]*?)<\/\\s*${tabName}\\s*>`,"gm"); let m; let tabNameArr = []; let copyPageText = this.pageText; // 注意下面的 copyPageText 不能改变 while ((m = regex.exec(copyPageText)) !== null) { // 这对于避免零宽度匹配的无限循环是必要的 if (m.index === regex.lastIndex) { regex.lastIndex++; } tabNameArr.push({ attrValue: m[1], tabValue: m[2] }) const newPageText =this.pageText.replace(m[0], ""); this.pageText = newPageText; } return tabNameArr; } // 解析双标签-只获取值 parseDoubleTabValue(tabName) { // 返回指定标签下指定属性下的值 const regex = RegExp(`<\\s*${tabName}[^<>]*\\s*>([\\s\\S]*?)<\/\\s*${tabName}\\s*>`,"gm"); let m; let tabNameArr = []; let copyPageText = this.pageText; while ((m = regex.exec(copyPageText)) !== null) { // 这对于避免零宽度匹配的无限循环是必要的 if (m.index === regex.lastIndex) { regex.lastIndex++; } tabNameArr.push({ tabValue: m[1] }) const newPageText =this.pageText.replace(m[0], ""); this.pageText = newPageText; } return tabNameArr; } // 解析所有指定的单标签 parseAllDesignatedSingTags(parseTabName) { // 匹配标签的正则表达式 const regex = /<(\w+)::([\S]+)(.*?)\/>/g; // 匹配属性键值对的正则表达式,支持连字符 const attributesRegex = /([\w-]+)="(.*?)"/g; let matches; const result = []; let modifiedString = this.pageText; while ((matches = regex.exec(this.pageText)) !== null) { const tabName = matches[1]; const tabValue = matches[2]; const attributesString = matches[3]; console.log(tabName, parseTabName); if (tabName !== parseTabName) continue; const attributes = {}; let attrMatch; while ((attrMatch = attributesRegex.exec(attributesString)) !== null) { attributes[attrMatch[1]] = attrMatch[2]; } result.push({ tabName, tabValue, ...attributes }); // 将匹配到的内容替换为空字符串 modifiedString = modifiedString.replace(matches[0], ''); } // 更新 pageText this.pageText = modifiedString; return result; } // 根据单标签的元信息进行stringify rebuildTags(tagMetaArr = []) { return tagMetaArr.map(tag => { const { tabName, tabValue, ...attributes } = tag; const attributesString = Object.entries(attributes) .map(([key, value]) => `${key}="${value}"`) .join(' '); return `<${tabName}::${tabValue} ${attributesString} />`; }).join('\n'); } } // 根据反馈的错误项调整templates位置,使得错误的靠后 function feedbackError(saveKey,currentErrorItem) { let items = cache.get(saveKey)??[]; let foundIndex = -1; // -1-查找模式 , n-已找到 n是所在位置模式 let foundValue = null; for(let i = 0; i < items.length; i++) { let item = items[i]; if(foundIndex == -1 ) { if(item == currentErrorItem) { foundIndex = i; foundValue = items[i]; } }else { items[i-1] = items[i]; // 查看是否是最后一个 if( i == items.length - 1 ) items[i] = foundValue; } } cache.set(saveKey,items); return items; } // 数据项选择记录器 class SelectHistoryRecorder { static HISTORY_CACHE_KEY = "HISTORY_CACHE_KEY"; static defaultIdFun = (item)=> JSON.stringify(item); static select(item,idFun = SelectHistoryRecorder.defaultIdFun) { // 记录到历史 let key = idFun(item); let history = cache.get( SelectHistoryRecorder.HISTORY_CACHE_KEY )??[]; history = history.filter(_item=>idFun(_item) != key) // 将原来的去除 history.unshift({...item}) // 必须拷贝 // 清理掉索引,索引只是本次数据加载有效的,而我们存储的历史数据是不随数据加载而改变的,也就是如果缓存索引会失效,没有索引它会自己找,当然我们会提供我们这里的数据给它找,如果在全局数据中匹配不到的话 history.forEach(_item=>{ delete _item.index; return _item; }) // 缓存历史数据 cache.set(SelectHistoryRecorder.HISTORY_CACHE_KEY,history); } static history(count) { let history = cache.get( SelectHistoryRecorder.HISTORY_CACHE_KEY )??[]; if(count == null) return history; // 如果没有传入count,那就是全部 let result = []; for(let i = 0; i < count && i+1 <= history.length; i++) result.push( history[i] ); // 将history前count个放在result数组中 return result; } } // 加分、“加分(取分)” class DataWeightScorer { static ITEM_WEIGHT_CACHE_KEY = "ITEM_WEIGHT_CACHE_KEY"; static defaultIdFun = (item)=> JSON.stringify(item); static SCORE_RECORD_ATTR_KEY = "weight"; static select(item,idFun = DataWeightScorer.defaultIdFun) { let ItemWeightData = cache.get( DataWeightScorer.ITEM_WEIGHT_CACHE_KEY )??{}; let key = idFun(item); ItemWeightData[key] = (ItemWeightData[key]??0) + 1 cache.set(DataWeightScorer.ITEM_WEIGHT_CACHE_KEY,ItemWeightData) } static assign(items=[],idFun = DataWeightScorer.defaultIdFun) { let ItemWeightData = cache.get( DataWeightScorer.ITEM_WEIGHT_CACHE_KEY)??{}; items.forEach(item=>{ let key = idFun(item); item[DataWeightScorer.SCORE_RECORD_ATTR_KEY] = ItemWeightData[key]??0; }) return items; } static sort(items=[],idFun = DataWeightScorer.defaultIdFun) { // 将权重赋于 DataWeightScorer.assign(items,idFun); // 根据权重排序(高->低) return items.sort((a, b) => b[DataWeightScorer.SCORE_RECORD_ATTR_KEY] - a[DataWeightScorer.SCORE_RECORD_ATTR_KEY]); } // 获取高频前count项 static highFrequency(count) { let ItemWeightData = cache.get( DataWeightScorer.ITEM_WEIGHT_CACHE_KEY )??{}; let orderKeys = Object.keys(ItemWeightData).sort((a, b) => ItemWeightData[b] - ItemWeightData[a]); if(count != null) orderKeys = orderKeys.slice(0, count); // keys转items return registry.searchData.matchItemsByKeys(orderKeys); } } // 将多个 function parseTis(bodyText) { // 提取整个tis标签的正则 const regex = /(<\s*tis::http[^<>]+\/\s*>)/gm; let raw = captureRegEx(regex,bodyText); if(raw != null) { return raw.map(item=>item[1]) } return null; } let USER_GITHUB_TOKEN_CACHE_KEY = "USER_GITHUB_TOKEN_CACHE_KEY"; let GithubAPI = { token: cache.get(USER_GITHUB_TOKEN_CACHE_KEY), defaultToken: '', setToken(token) { if(token != null) this.token = token; if(this.token == null) { token = prompt("请输入您的github Token (只缓存在你的本地):") // 获取的内容无效 if(token == null || token == "") return this; // 内容有效-设置 cache.set(USER_GITHUB_TOKEN_CACHE_KEY,this.token = token); } return this; }, clearToken() { cache.remove(USER_GITHUB_TOKEN_CACHE_KEY) this.token = null; }, getToken(isRequest = false) { if(this.token == null && isRequest) this.setToken(); return this.token; }, baseRequest(type,url,{query,body}={},header = {}) { query = {...query} return request(type, url, { query,body,header }); }, getUserInfo() { return this.baseRequest("GET","https://api.github.com/user") }, commitIssues(body) { const header = {Authorization:`Bearer ${this.token}`}; return this.baseRequest("POST","https://api.github.com/repos/My-Search/TisHub/issues",{ body, header }) }, // get issues不要加 Authorization 头,可能会出现401 getTisForIssues({keyword,state} = {}) { let query = null; if(state != null) query = {state}; let token = this.token; if(token == null) token = this.defaultToken; return keyword ? new Promise((resolve,reject)=>{ // API兼容处理 this.baseRequest("GET",`https://api.github.com/search/issues?q=repo:My-Search/TisHub+state:${state}+in:title+${keyword}`) .then(response=>resolve(response.items)).catch(error=>resolve([])); }) : this.baseRequest("GET","https://api.github.com/repos/My-Search/TisHub/issues",{query}) } } // 从订阅标签中提取订阅链接 let TisHub = { // 将第一个tis集与第二个tis集合并 tisFilter(source,filterList) { if(typeof source == "string") source = parseTis(source); if(typeof filterList == "string") filterList = parseTis(filterList); for(let filterItem of filterList) { let pageTextHandler = new PageTextHandleChains(filterItem); let tabMetaInfos = pageTextHandler.parseAllDesignatedSingTags("tis"); let subscribedLink = null; // 一个filterItem解析出元信息返回tabMetaInfos只能有一个元素,如果是多个,只取一个 if(tabMetaInfos != null && tabMetaInfos.length > 0 ) subscribedLink = tabMetaInfos[0].tabValue; // 如果取不出来,说明tis无效,断言不需要解析filterItem就是subscribedLink if(subscribedLink == null) subscribedLink = filterItem; // subscribedLink 这里是filterItem用于filter source的元素实体,下面开始过滤 source = source.filter(resultSubscribed=>! resultSubscribed.includes(subscribedLink)); } return source; }, getTisHubAllTis(filterList = []) { return new Promise((resolve,reject)=>{ let openIssuesTisPromise = this.getOpenIssuesTis(); let result = []; return Promise.all([ this.getOpenIssuesTis(), this.getClosedIssuesTis() ]).then(values=>{ for(let value of values) { if(value == null ) continue; for(let tisListObj of value) { if(tisListObj != null ) result.push(...tisListObj.tisList) } } // 过滤并提交结果 resolve(this.tisFilter(result,filterList)); }) }) }, // {keyword,state} .其中state {open, closed, all} getTisForIssues(params = {}) { return new Promise((resolve,reject)=>{ GithubAPI.getTisForIssues(params).then(response=>{ if(response != null && Array.isArray(response)) { resolve(response.map(obj=>{return { owner: obj.user.login, ownerProfile: obj.user.html_url, title: obj.title, tisList: parseTis(obj.body), status: obj.state }})) } }).catch(error=>resolve([])); }) }, getOpenIssuesTis(params = {}) { return this.getTisForIssues({state: "open",...params}); }, getClosedIssuesTis(params = {}) { return this.getTisForIssues({state: "closed",...params}); }, tisListToTisText(tisList) { let text = ""; for(let tis of tisList) text += tis.tisList; return text; } } // 全局注册表 let ERROR = { tell(info) { console.error("ERROR " + info) } } let registry = { view: { viewVisibilityController: () => { ERROR.tell("视图未初始化,但你使用了它的未初始化的注册表信息!") }, viewDocument: null, // 视图挂载后有值 element: null, // 存放着视图的关键元素对象 视图挂载后有值 tis: { beginTis(msg) { if(msg == null || msg.length === 0) return; const tisDocument = document.querySelector("#my_search_box > #tis"); tisDocument.innerHTML = msg; tisDocument.display = "block"; console.log("设置结束") return ()=>{ tisDocument.innerHTML = ""; // 置空消息内容 tisDocument.display = "none"; // 让tis不可见 } } }, setButtonVisibility: () => { ERROR.tell("按钮未初始化!") }, titleTagHandler: { handlers: [], // 标题tag处理器 execute: function (title) { // 去掉标题内容只剩下tags let arr = captureRegEx(/(\[.*\])/gm,title) if(arr == null || arr[0] == null || arr[0][0] == null) return ""; let tagStr = arr[0][0]; for(let titleTagHandler of this.handlers) { let result = titleTagHandler(tagStr.trim()); if(result != -1) return result; } return tagStr; } }, viewFirstShowEventListener: [], viewHideEventAfterListener: [], // 在查看详情内容时,返回后事件 itemDetailBackAfterEventListener: [ // 简讯内容隐藏-确定存在触发的事件 - 脚本环境变量置空 () => registry.script.clearMSSE() ], // 视图延时隐藏时间,避免点击右边logo,还没显示就隐藏了 delayedHideTime: 100, initialized: false, textView: { cssFillPrefix(css = "", prefix = "") { const cssBlocks = css.split('}'); let outputCSS = ''; for (let block of cssBlocks) { let blockLines = block.split('\n'); let blockOutput = ''; for (let line of blockLines) { // 判断行末是否以 `{` 或 `,` 结尾且行首不能有空格 if ((line.trim().endsWith('{') || line.trim().endsWith(',')) && !line.startsWith(' ') && !line.trim().startsWith('@')) { blockOutput += `${prefix} ${line.trim()}`; // 在当前行前加上前缀 } else { blockOutput += line; // 其他行保持原样 } blockOutput += '\n'; // 添加换行符,用于分隔CSS内容的各行 } if (blockOutput.trim() !== '') { outputCSS += blockOutput; outputCSS += '}\n'; // 只有在当前块不为空时添加闭合大括号 } } return outputCSS; }, show(html,css = "",js = "") { const MS_BODY_ID = "ms-page-body"; html = `` + `
${html}
` // 这里在函数内执行js是为了在同一页面未刷新下可多次执行该js不执行,也就是对变量/函数等进行隔离 +`` let my_search_box = $(registry.view.viewDocument); // 视图还没有初始化 if(my_search_box == null) return; let matchResult = registry.view.element.matchResult; let textView = registry.view.element.textView textView.html(html); /*使用code代码块样式*/ document.querySelectorAll('#text_show pre code').forEach((el) => { // 这里没有错,发警告不用理 hljs.highlightElement(el); }); matchResult.css({ "display": "none" }) textView.css({ "display":"block" }) waitViewRenderingComplete(() => { const currentKey = registry.searchData.searchHistory.currentKeyword(); if(currentKey.trim().length > 3) scrollToText(currentKey,0,"#ms-page-body"); }); } }, // 搜索框logo控制 logo: { // logo src值 originalLogoImgSrc: null, // logo按钮是否按下状态 isLogoButtonPressedRef: ref(false), getLogoImg: function () { let viewDocument = registry.view.viewDocument; if(viewDocument == null ) return null; let currentLogoImg = registry.view.element.logoButton.find("img"); if(this.originalLogoImgSrc == null) this.originalLogoImgSrc = currentLogoImg.attr("src"); return currentLogoImg; }, change: function (imgResource) { let logoImg = this.getLogoImg(); if(imgResource == null || logoImg == null ) return; logoImg.attr("src",imgResource) }, reset: function() { let logoImg = this.getLogoImg(); if (logoImg == null ) return; logoImg.attr("src",this.originalLogoImgSrc) } }, modeEnum: { UN_INIT: -2, // 未初始化 HIDE: -1, // 隐藏 WAIT_SEARCH: 0, // 待搜索模式 SHOW_RESULT: 1, // 结果显示模式 SHOW_ITEM_DETAIL: 2 // 查看项详情 (简述内容查看/脚本页) }, seeNowMode() { if(this.viewDocument == null) return this.modeEnum.UN_INIT; if(this.element.textView.css('display')!== "none") return this.modeEnum.SHOW_ITEM_DETAIL; if(this.element.matchResult.css('display') !== "none") return this.modeEnum.SHOW_RESULT; return this.viewDocument.style.display !== "none" ? this.modeEnum.WAIT_SEARCH : this.modeEnum.HIDE; } }, other: { UPDATE_CDNS_CACHE_KEY: "UPDATE_CDNS_CACHE_KEY" }, searchData: { // 处理的历史 processHistory: [], // 用于数据显示后,数据又更新了 version: 0, // 全局搜索数据 data: [], // 数据更新后有效时长 effectiveDuration: 1000*60*60*12, // 数据设置到全局中的时间 dataMountTime: null, clearData() { this.data = []; this.dataMountTime = null; }, setData(data) { if(data == null || data.length == 0) return; this.data = data; this.dataMountTime = Date.now(); }, getData() { let dataPackage = cache.get(registry.searchData.SEARCH_DATA_KEY); if(dataPackage == null || dataPackage.data == null) return this.data; // 缓存信息不为空,深入判断是否使用缓存的数据 let updateDataTime = dataPackage.expire - this.effectiveDuration; // 如果数据在挂载后面已经更新了,重新加载数据到全局中 // 全局data (即this.data)与dataMountTime是同时设置的,两者理论上是必须同时有值的 if(this.data == null || updateDataTime > this.dataMountTime) { console.logout("== 数据未加载或已检查到在其它页面已重新更新数据 ==") this.setData(dataPackage.data); } return this.data; }, // 根据keys(由idFun决定)从data中匹配items matchItemsByKeys: function (keys = []) { let that = this; if(keys.length == 0) return []; // 有keys转items let items = keys.map(key=>{ for(let item of that.data) { if(that.idFun(item) == key) return item; } return null; }) // 对数组keys去空,注意此时keys已经是items了 return items.filter(item => item != null); }, specialKeyword: { // 特殊的keyword new: "", history: "", highFrequency: "" }, // 决定着数据是否要再次初始化 isDataInitialized: false, // 从可自定义搜索数据中根据title与desc进行数据匹配 findSearchDataItem: function (title = "",desc = "",matchData) { if(matchData == null ) matchData = this.data; for(let item of matchData) { if( (item.title.includes(title) || title.includes(item.title) ) && ( item.desc.includes(desc) || desc.includes(item.desc) )) return item; } return null; }, // 数组差异-获取不同的元素比较的基值 idFun(item) { // 自定义比较 if(item == null || !( item instanceof Object && item.title != null)) return null; return item.title.replace(/\[.*\]/,"").trim()+(""+item.desc).trim(); }, // 旧的新数据缓存KEY OLD_SEARCH_DATA_KEY: "OLD_SEARCH_DATAS_KEY", // 标签数据缓存KEY DATA_ITEM_TAGS_CACHE_KEY: "DATA_ITEM_TAGS_CACHE_KEY", // 用户维护的不关注标签列表,缓存KEY USER_UNFOLLOW_LIST_CACHE_KEY: "USER_UNFOLLOW_LIST_CACHE_KEY", // 用户安装tishub订阅的缓存 USE_INSTALL_TISHUB_CACHE_KEY: "USE_INSTALL_TISHUB_CACHE_KEY", // 默认用户不关注标签 USER_DEFAULT_UNFOLLOW: ["成人内容","Adults only"], // 已经清理了用户不关注的与隐藏的标签,这是用户应真正搜索的数据 CLEANED_SEARCH_DATA_CACHE_KEY: "CLEANED_SEARCH_DATA_CACHE_KEY", subscribeKey: "subscribeKey", showSize: 15, isSearchAll: false, searchEven: { event:{}, // 搜索状态,失去焦点隐藏的一要素 isSearching:false, async send(search,rawKeyword) { try { // 标记为搜索进行中 this.isSearching = true; for(let subscriptionRegular of Object.keys(this.event)) { const regex = new RegExp(subscriptionRegular,"i"); // 将正则字符串转换为正则表达式对象 if(regex.test(rawKeyword) && typeof this.event[subscriptionRegular] == "function" ) { return this.event[subscriptionRegular](search,rawKeyword); } } return await search(rawKeyword); }finally { // 标记为搜索结束 this.isSearching = false; } } }, // 新数据设置的过期天数 NEW_DATA_EXPIRE_DAY_NUM:7, // 搜索逻辑,可用来手动触发搜索 triggerSearchHandle: function (keyword){ // 获取input元素 const inputEl = registry.view.element?.input; // 如果视图还没有初始化,触发搜索无效 if(! inputEl) return; if(keyword == null) { keyword = inputEl.val() ?? "" }else { inputEl.val(keyword); } // 手动触发input事件 inputEl[0].dispatchEvent(new Event("input", { bubbles: true })); // 维护全局搜索keyword this.keyword = keyword; }, // 数据改变事件 dataChangeEventListener: [], // 缓存被删除事件 dataCacheRemoveEventListener:[], onSearch: [], // 新数据块处理完成事件 // 更新搜索数据的责任链 USDRC: getResponsibilityChain(), onNewDataBlockHandleAfter: [], // 新数据的tag NEW_ITEMS_TAG: "[新]", // 搜索的keyword keyword: "", // 搜索的附加文件 searchForFile: { files: [], refreshFileListView() { registry.view.element.files.html('') for(let [index,file] of this.files.entries()) { const reader = new FileReader(); reader.onload = function (e) { // 创建 img 元素并设置 src 为文件的 Base64 数据 registry.view.element.files.append(`
`); }; reader.readAsDataURL(file); } }, push(file) { this.files.push(file); this.refreshFileListView(); }, delete() { this.files.pop(); this.refreshFileListView(); }, clear() { const trashCan = this.files; this.files = []; this.refreshFileListView(); return trashCan; }, start() { registry.view.element.input.on('paste', event => { // 希望input外层元素无法感知到paste的动作 event.stopPropagation(); // 获取粘贴事件对象中的剪贴板数据 const clipboardData = event.originalEvent.clipboardData || window.clipboardData; if (clipboardData) { const items = clipboardData.items; for (let i = 0; i < items.length; i++) { const file = items[i].getAsFile(); if (! file.type.startsWith('text/')) this.push(file); } } }); registry.view.viewHideEventAfterListener.push(() => this.clear()); } }, // 持久化Key SEARCH_DATA_KEY: "SEARCH_DATA_KEY", SEARCH_NEW_ITEMS_KEY:"SEARCH_NEW_ITEMS_KEY", // 搜索搜索出来的数据 searchData: [], pos: 0, clearUrlSearchTemplate(url) { return url.replace(/\[\[[^\[\]]*\]\]/gm,""); }, faviconSources: [ // favicon来源:https://api.cxr.cool/ // "https://favicon.yandex.net/favicon/${domain}", 淘汰原因:当获取不到favicon时不报错,而是显示空白图标 // "https://api.cxr.cool/ico/?url=${domain}", 淘汰原因:慢 // "https://api.vvhan.com/api/ico?url=${domain}", 淘汰原因:快,但存在很多网站的图标无法获取 // "https://statistical-apricot-seahorse.faviconkit.com/${domain}/32", 淘汰原因:有些图标无法获取,但会根据网站生成其网站单字符图标 // "https://favicons.fuzqing.workers.dev/api/getFavicon?url=${rootUrl}", 淘汰原因:快,但存在很多网站的图标无法获取 // "https://tools.ly522.com/ico/favicon.php?url=${rootUrl}", 淘汰原因:废的,10个8个获取不到 "https://api.iowen.cn/favicon/${domain}.png", // 神 "https://api.xinac.net/icon/?url=${rootUrl}", // 可选:但有点慢 // "https://favicon.qqsuu.cn/${rootUrl}", 淘汰原因:外国站点不行 // "https://api.uomg.com/api/get.favicon?url=${rootUrl}", 淘汰原因:外国站点不行 // "https://api.vvhan.com/api/ico?url=${domain}", // 淘汰原因:存在很多网站的图标无法获取 // "https://api.15777.cn/get.php?url=${rootUrl}",// 淘汰原因:存在很多网站的图标无法获取 "https://ico.txmulu.com/${domain}", // 可选,但有些站点不行 // "https://api.cxr.cool/ico/?url=${domain}", // 很多网站获取不到,国外站点不行 "${rootUrl}/favicon.ico", // 永久有效的兜底 ], CACHE_FAVICON_SOURCE_KEY: "CACHE_FAVICON_SOURCE_KEY", // 可见远程源 CACHE_FAVICON_SOURCE_TIMEOUT: 1000*60*60*4, // 4个小时重新检测一下favicon源/过期时间,只会在呼出搜索框后检查 getFaviconAPI: (function(){ let defaultFaviconUrlTemplate = "${rootUrl}/favicon.ico"; let faviconUrlTemplate = defaultFaviconUrlTemplate; let isRemoteTemplate = false; // 查看是否已经检查模板 function checkTemplateAndUpdateTemplate() { let faviconSourceCache = cache.get(registry.searchData.CACHE_FAVICON_SOURCE_KEY); if( !isRemoteTemplate && faviconSourceCache != null && faviconSourceCache.sourceTemplate != null ) { faviconUrlTemplate = faviconSourceCache.sourceTemplate; // 设置已经是远程Favicon模板 isRemoteTemplate = true; } } return function(url,isStandby = false) { checkTemplateAndUpdateTemplate(); let useFaviconUrlTemplate = faviconUrlTemplate; // 如果是要获取备用favicon,那直接使用上面定义的faviconUrlTemplate if(isStandby) useFaviconUrlTemplate = defaultFaviconUrlTemplate; // 去掉资源的“可搜索”模板,才是真正的URL url = registry.searchData.clearUrlSearchTemplate(url); // 将资源url放到获取favicon的源模板中 let urlObj = parseUrl(url) return useFaviconUrlTemplate.fillByObj(urlObj); } })(), tmpVar: null, // 用于防抖 searchPlaceholder(target = "SELECT",placeholder,duration = 1200) { // 全部的输入提示 let inputDescs = ["我的搜索"]; // 当前应用“输入提示” let inputDesc = inputDescs[Math.floor(Math.random()*inputDescs.length)]; if(target == "UPDATE") { const inputEl = registry.view.element?.input; // 如果视图还没挂载,无需显示 if(! inputEl) return; if(this.tmpVar != null) { clearTimeout(this.tmpVar); } this.tmpVar = setTimeout(()=>{ inputEl.attr("placeholder",this.searchPlaceholder()); },duration) let updateResult = placeholder==null?`🔁 数据库更新到 ${this.data==null?0:this.data.length}条`:placeholder; inputEl.attr("placeholder",updateResult); return updateResult; } return inputDesc; }, // 存储着text转pinyin的历史 registry.searchData.subSearch.isSubSearchMode TEXT_PINYIN_KEY: "TEXT_PINYIN_MAP", // 默认数据不应初始化,不然太占内存了,只用调用了toPinyin才会初始化 getGlobalTextPinyinMap() getGlobalTextPinyinMap: (function() { let textPinyinMap = null; return function (){ if(textPinyinMap != null) return textPinyinMap; return (textPinyinMap = cache.jGet("TEXT_PINYIN_MAP")??{}); } })(), subSearch: { searchBoundary: " : ", isEnteredSubSearchMode: false, // 是否已经进入子搜索模式 // 不传参数是看当前是否为子搜索模式 , [0] 是最近一个 isSubSearchMode(by = undefined) { let byKeyword = typeof(by) === 'string' ? by // by就是keyword : (by === undefined ? registry.searchData.searchHistory.currentKeyword() : registry.searchData.searchHistory.history[by]); // by是index return byKeyword && byKeyword.includes(this.searchBoundary); }, // 获取父级(根keyword) getParentKeyword(keyword) { // 如果没有传入使用搜索框的value if(! keyword) keyword = registry.searchData.searchHistory.currentKeyword(); return (keyword || "").split(this.searchBoundary)[0].trim() }, // 获取子搜索keyword getSubSearchKeyword(keyword) { // 如果没有传入使用搜索框的value if(! keyword) keyword = registry.searchData.searchHistory.currentKeyword(); let _arr = (keyword || "").split(this.searchBoundary); if( _arr.length < 2 ) return undefined; return _arr[1].trim(); } }, searchHistory: { history: [], // 新,旧... add(keyword) { if(! keyword) return; // 维护isEnteredSubSearchMode变量状态(进入与退出) const searchBoundary = registry.searchData.subSearch.searchBoundary if(keyword !== searchBoundary && keyword.endsWith(searchBoundary)) { registry.searchData.subSearch.isEnteredSubSearchMode = true; console.logout("进入了子搜索") }else if(registry.searchData.subSearch.isEnteredSubSearchMode && !keyword.includes(searchBoundary)){ registry.searchData.subSearch.isEnteredSubSearchMode = false; console.logout("退出了子搜索") } // 加入到历史 const _history = this.history; _history.unshift(keyword); _history.slice(10); // 不能超过10个元素 }, currentKeyword() { return registry.view.element.input.val() }, // 当前keyword "123 : 哈哈" 与 最近"123 : 嘻嘻" 则返回true,即看左边 seeCurrentEqualsLastByRealKeyword() { // 上一次真实搜索keyword === 当前真实搜索keyword return registry.searchData.subSearch.getParentKeyword(this.history[0]) === registry.searchData.subSearch.getParentKeyword(this.currentKeyword()); } }, searchProTag: "[可搜索]", links: { stringifyForSearch(links) { if(links == null) return '' links = links.filter(link => link != null); return links.reduce((acc, cur) => acc + `${cur.text}${cur.url}${cur.title}`, ''); } } }, script: { // MSSE默认值/模板 MS_SCRIPT_ENV_TEMPLATE: { event: { // 发送sub keyword事件 sendListener: [] // 脚本页监听IPush事件 }, // 让脚本页可使用cache,为避免冲突,加前缀于区别 cache: {...cache, prefix: "MS_SCRIPT_CACHE:"}, // 让脚本页面获取脚本项数据 getSearchDB() { return [...registry.searchData.getData()] }, // 让脚本页面获取选择的文本 getSelectedText, // 挂载markdown md2html, // 挂载http request对象 request }, // 当值为undefined时表示会话未开始 SESSION_MS_SCRIPT_ENV: undefined, openSessionForMSSE() { return (this.SESSION_MS_SCRIPT_ENV = (actualWindow.MS_SCRIPT_ENV = this.MS_SCRIPT_ENV_TEMPLATE)); }, clearMSSE() { this.SESSION_MS_SCRIPT_ENV = undefined; }, // 这个函数会在脚本页渲染完成后调用 tryRunTextViewHandler() { const input = registry.view.element.input; const rawKeyword = input.val(); // 如果还在显示搜索的数据项 执行失败返回false if(registry.view.seeNowMode() === registry.view.modeEnum.SHOW_ITEM_DETAIL) { // 当msg不为空发送msg消息到脚本 const subKeyword = registry.searchData.subSearch.getSubSearchKeyword() if( subKeyword == undefined && registry.searchData.searchForFile.files.length === 0) return; // 通知脚本回车send事件 const msg = subKeyword; // actualWindow是页面真实windows对象 this.SESSION_MS_SCRIPT_ENV.event.sendListener.forEach(listener=>listener(msg,registry.searchData.searchForFile.clear())) // clear : 清理掉send msg内容 input.val(rawKeyword.replace(msg,"")) return true; } return false; } } } let dao = {} // registry.registry.viewDocument // 页面文本选择器 function getSelectedText(tis = '请选择页面文本') { function createTipElement() { const tipElement = document.createElement('p'); tipElement.textContent = tis; // 设置行内样式 tipElement.style.position = 'fixed'; tipElement.style.top = '0'; tipElement.style.left = '50%'; tipElement.style.transform = 'translateX(-50%)'; tipElement.style.backgroundColor = 'black'; tipElement.style.color = 'white'; tipElement.style.padding = '10px 20px'; tipElement.style.fontSize = '16px'; tipElement.style.zIndex = '9999'; tipElement.style.borderRadius = '5px'; document.body.appendChild(tipElement); return tipElement; } const view = registry.view.viewDocument; if(view == null) throw new Error('调用页面文本选择器异常,原因:视图隐藏不允许!') let tipElement = createTipElement(); return new Promise((resolve) => { // 先隐藏搜索视图 view.style.display = "none"; // 监听鼠标抬起事件(用户结束选择) const onMouseUp = () => { const selectedText = window.getSelection().toString().trim(); // 确保用户已选择文本 if (selectedText) { tipElement?.remove(); view.style.display = "block"; resolve(selectedText); document.removeEventListener('mouseup', onMouseUp); } }; // 监听鼠标抬起事件 document.addEventListener('mouseup', onMouseUp); }); } // 网页脚本自动执行函数 let autoRunStringScript = { cacheKey : "autoRunStringScriptKey", getData() { let scripts = cache.get(this.cacheKey)??{}; let keys = Object.keys(scripts); for(let key of keys) { let time = scripts[key].timeout; if(Date.now() > time) delete scripts[key]; } cache.set(this.cacheKey,scripts); return scripts; }, add(target,funStr,effectiveTime = 5000) { if(target == null || ! target.trim().startsWith("http")) return; let data = this.getData(); data[target.trim()] = { timeout: Date.now()+effectiveTime, handle: funStr } cache.set(this.cacheKey,data); }, run() { let currentPageUrl = document.URL; let data = this.getData(); let keys = Object.keys(data); let targetObj = null; for(let key of keys) { if(key.startsWith(currentPageUrl) || currentPageUrl.startsWith(key)) targetObj = data[key]; } if(targetObj != null) { // 从data中失去,再执行 delete data[currentPageUrl]; let handle = targetObj.handle; if(handle == null) return; new Function('$',handle)($); } } } // 页面加载执行 autoRunStringScript.run(); // 添加页面模拟脚本 function addPageSimulatorScript(url,scriptStr) { scriptStr = `function exector(handle) { function selector(select, all = false) { return all ? document.querySelectorAll(select) : document.querySelector(select); } function clicker(select) { let element = selector(select); if (element != null) element.click(); } function scroller(selector = 'body', topOffset = null, timeConsuming = 2000) { return new Promise((resolove,reject)=>{ var containerElement = $(selector); if (containerElement.length > 0) { if (topOffset !== null) { $('html, body').animate({ scrollTop: containerElement.offset().top + topOffset }, timeConsuming); } else { $('html, body').animate({ scrollTop: containerElement.offset().top }, timeConsuming); } } else { console.error('找不到指定的元素'); } setTimeout(()=>{resolove(true)},timeConsuming) }) } function annotator(select, styleStr = "border:5px solid red;") { let element = selector(select); if (element == null) return; element.style = styleStr; } handle({ click: clicker, roll: scroller, dimension: annotator }); } window.onload = function () { exector(${scriptStr}) }`; autoRunStringScript.add(url,scriptStr,6000); } // 判断是否只是url且不应该是URL文本 (用于查看类型) function isUrl(resource) { // 如果为空或不是字符串,就不是url if(resource == null || typeof resource != "string" ) return false; // resource是字符串类型 resource = resource.trim().split("#")[0]; // 不能存在换行符,如果存在不满足 if(resource.indexOf("\n") != -1 ) return false; // 被“空白符”切割后只能有一个元素 if(resource.split(/\s+/).length != 1) return false; // 如果不满足url,返回false return isHttpUrl(resource); } /*cache.remove(registry.searchData.SEARCH_DATA_KEY); cache.remove(registry.searchData.SEARCH_DATA_KEY+"2"); cache.remove(registry.searchData.SEARCH_NEW_ITEMS_KEY); */ // 设置远程可用Favicon源 let setFaviconSource = function () { function startTestFaviconSources(sources,pos,setFaviconUrlTemplate) { if(pos > sources.length - 1) return; console.logout(`${pos}/${sources.length-1}: 正在测试 `+sources[pos]) checkUsability(sources[pos]).then(function(result) { console.logout("使用的源:"+ sources[pos]) setFaviconUrlTemplate(result); }).catch(function() { startTestFaviconSources(sources,++pos,setFaviconUrlTemplate) }); } let cacheFaviconSourceData = cache.get(registry.searchData.CACHE_FAVICON_SOURCE_KEY); let currentTime = new Date().getTime(); let timeout = registry.searchData.CACHE_FAVICON_SOURCE_TIMEOUT; let faviconSources = registry.searchData.faviconSources; // 生成favicon源镜像 function currentSourceArraySnapshot() { return JSON.stringify(faviconSources); } if(cacheFaviconSourceData == null || currentTime - cacheFaviconSourceData.updateTime > timeout || cacheFaviconSourceData.sourceArraySnapshot != currentSourceArraySnapshot()) { if(cacheFaviconSourceData != null) { console.logout(`==之前检查的已超时或源发现了修改,重新设置Favicon源==`); } let pos = 0; let promise = null; function setFaviconUrlTemplate(source = null) { console.logout("Test compled, set source! "+source) if(source != null) { cache.set(registry.searchData.CACHE_FAVICON_SOURCE_KEY, { updateTime: new Date().getTime(), sourceTemplate: source, sourceArraySnapshot: currentSourceArraySnapshot() }) } } // 去测试index=0的源, 当失败,会向后继续测试 if(faviconSources.length < 1) return; startTestFaviconSources(faviconSources,0,setFaviconUrlTemplate); }else { console.logout(`Favicon源${(timeout - (currentTime - cacheFaviconSourceData.updateTime))/1000}s后测试`); } } // 判断是否要执行设置源,如果之前没有设置过的话就要设置,而不是通过事件触发 if(cache.get(registry.searchData.CACHE_FAVICON_SOURCE_KEY) == null ) setTimeout(()=>{setFaviconSource();},2000); // 添加事件(视图在页面中初次显示时) registry.view.viewFirstShowEventListener.push(setFaviconSource); // 【函数库】 // 加载样式 function loadStyleString(css) { var style = document.createElement("style"); style.type = "text/css"; try { style.appendChild(document.createTextNode(css)); } catch(ex) { style.styleSheet.cssText = css; } var head = document.getElementsByTagName('head')[0]; head.appendChild(style); return style; } // 加载html function loadHtmlString(html) { // 创建一个新的 div 元素 var newDiv = document.createElement("div"); // 设置新的 div 的内容为要追加的 HTML 字符串 newDiv.innerHTML = html; // 将新的 div 追加到 body 的末尾 document.body.appendChild(newDiv); return newDiv; } // Div方式的Page页(比如构建配置面板视图) function DivPage(cssStr,htmlStr,handle) { let style = loadStyleString(cssStr); let div = loadHtmlString(htmlStr); function selector(select,isAll = false) { if(isAll) { return div.querySelectorAll(select); }else { return div.querySelector(select); } } function remove() { div.remove(); style.remove(); } handle(selector,remove); } // 异步函数 function asyncExecFun(fun,time = 20) { setTimeout(()=>{ fun(); },time) } // 同步执行函数 let syncActuator = function () { return (function () { let queue = []; let vote = 0; let timer = null; // 确保定时器已经在运行 function ensureTimerRuning() { if (timer != null) return; timer = setInterval(async () => { let taskItem = queue.pop(); if (taskItem != null) { taskItem.active = true; await taskItem.task; // 任务执行完,消耗一票 vote--; if (vote <= 0) { clearInterval(timer); timer = null; } } }, 100); } return function (handleFun, args, that) { // 让票加一 vote++; // 确保定时器运行 ensureTimerRuning(); let taskItem = { active: false, task: null } taskItem.task = new Promise((resolve, reject) => { let timer = null; timer = setInterval(async () => { if (taskItem.active) { await resolve(handleFun.apply(that ?? window, args)); clearInterval(timer); } }, 30) }) queue.unshift(taskItem) return taskItem.task; } })() } // 全页面“询问”函数 function askIsExpiredByTopic(topic,validTime=10*1000) { let currentTime = new Date().getTime(); let lastTime = cache.get(topic); let isExpired = lastTime == null || lastTime + validTime < currentTime; if(isExpired) { // 获取到资格,需要标记 cache.set(topic,currentTime); } return isExpired; } // 移除数组中重复元素的函数 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); } // 【追加原型函数】 // 往字符原型中添加新的方法 matchFetch String.prototype.matchFetch=function (regex,callback) { let str = this; // Alternative syntax using RegExp constructor // const regex = new RegExp('\\[\\[[^\\[\\]]*\\]\\]', 'gm') let m; let length = 0; while ((m = regex.exec(str)) !== null) { // 这对于避免零宽度匹配的无限循环是必要的 if (m.index === regex.lastIndex) { regex.lastIndex++; } // 结果可以通过`m变量`访问。 m.forEach((match, groupIndex) => { length++; callback(match, groupIndex); }); } return length; }; // 往字符原型中添加新的方法 matchFetch String.prototype.fillByObj=function (obj) { if(obj == null ) return null; let template = this; let resultUrl = template; for(let key of Object.keys(obj)) { let regexStr = `\\$\\s*?{[^{}]*${key}[^{}]*}`; resultUrl = resultUrl.replace(new RegExp(regexStr),obj[key]); } if(/\$.*?{.*?}/.test(resultUrl)) return null; return resultUrl; } // 比较两个数组是否相等(顺序不相同不影响) function isArraysEqual (arr1,arr2) { if( arr2 == null || arr1.length != arr2.length ) return false; for(let arr1Item of arr1) { let f = false; for(let arr2Item of arr2) { if(arr1Item == arr2Item ) { f = true; break; } } if(! f) return false; } return true; } function compareArrayDiff (arr1, arr2, idFun = () => null,diffRange = 3) { // diffRange值:“1”是左边多的,“2”是右边数组多的,3是左右合并,0是相同的部分,30是两个数组去重的 function hashString(obj) { let str = JSON.stringify(obj); let hash = 0; [...str].forEach((char) => { hash += char.charCodeAt(0); }); return "" + hash; } if (arr2 == null || arr2.length == 0) return arr1; // arr1与arr2都为数组对象 // 将arr1生成模板 let template = {}; for (let item of arr1) { let itemHash = hashString(idFun(item) ?? item); if (template[itemHash] == null) template[itemHash] = []; template[itemHash].push(item); } let leftDiff = []; let rightDiff = []; let overlap = []; // arr2根据arr1的模板进行比对 for (let item of arr2) { let itemHash = hashString(idFun(item) ?? item); let hitArr = template[itemHash]; let item2Json = idFun(item) ?? JSON.stringify(item); if (hitArr != null) { // 模板中存在 for (let hitIndex in hitArr) { let hashItem = hitArr[hitIndex]; // 判断冲突是否真的相同 let item1Json = idFun(hashItem) ?? JSON.stringify(hashItem); if (item1Json == item2Json) { // 命中-将arr1命中的删除 delete hitArr.splice(hitIndex, 1); overlap.push( {...item, ...hashItem} ); break; } } } else { // 模板不存在,是差异项 rightDiff.push(item); } } // 将模板中未命中的收集 for (let templateKey in template) { let templateValue = template[templateKey]; //templateValue 是数组 if (templateValue == null || !(templateValue instanceof Array)) continue; for (let templateValueItem of templateValue) { leftDiff.push(templateValueItem); } } // 根据参数,返回指定的数据 switch (diffRange) { case 0: return overlap; break; case 1: return leftDiff; break; case 2: return rightDiff; break; case 3: return [...leftDiff, ...rightDiff]; break; case 30: return [...leftDiff, ...rightDiff, ...overlap]; } } // 保证replaceAll方法替换后也可以正常 String.prototype.toReplaceAll = function(str1,str2) { return this.split(str1).join(str2); } // 向原型中添加方法:文字转拼音 String.prototype.toPinyin = function (isOnlyFomCacheFind= false,options = { toneType: 'none', type: 'array' }) { let textPinyinMap = registry.searchData.getGlobalTextPinyinMap(); // 查看字典中是否存在 if(textPinyinMap[this] != null) { // console.logout("命中了") return textPinyinMap[this]; } // 如果 isOnlyFomCacheFind = true,那返回原数据 if(isOnlyFomCacheFind) return null; // console.logout("字典没有,将进行转拼音",Object.keys(textPinyinMap).length) let {pinyin} = pinyinPro; let text = this; let space = "" let spaceChar = " "; text = text.toReplaceAll(spaceChar,space) let pinyinArr = pinyin(text,options); // 保存到全局字典对象 ( 会话级别 ) textPinyinMap[this] = pinyinArr.join("").toReplaceAll(space,spaceChar).toUpperCase(); return textPinyinMap[this]; } // 加载全局样式 loadStyleString(` /*搜索视图样式*/ #searchBox { height: 45px; background: #ffffff; padding: 0 10px; box-sizing: border-box; z-index: 10001; position: relative; display: flex; align-items: center; flex-wrap: nowrap; } #searchBox #ms-input-files { display: flex; flex-wrap: nowrap; align-items: center; height: 100%; } #ms-input-files .ms-input-file { height: 70%; display: flex; align-items: center; margin-right: 2px; } #ms-input-files .ms-input-file img { height: 100%; box-sizing: border-box; padding: 3px; background: #d3d3d3; } #my_search_input { text-align: left; width: 100%; height: 100%; border: none; outline: none; font-size: 15px; background: #fff; padding: 0px; box-sizing: border-box; color: rgba(0, 0, 0, .87); font-weight: 400; margin: 0px; } #matchResult { display: none; } #matchResult > ol { margin: 0px; padding: 0px 15px 5px; } #text_show { display: none; width: 100%; box-sizing: border-box; padding: 5px 10px 7px; font-size: 15px; line-height: 25px; max-height: 450px; overflow: auto; text-align: left; color: #000000; user-select: text !important; /* 允许用户选中复制 */ } #text_show img { width: 100%; } #text_show .copy-btn { position: absolute; top: 8px; right: 8px; background: #7f7f7fa1; color: white; border: none; padding: 3px 10px; font-size: 12px; cursor: pointer; border-radius: 3px; opacity: 0.8; transition: opacity 0.3s; } #text_show .copy-btn:hover { opacity: 1; } /*定义字体*/ @font-face { font-family: 'HarmonyOS'; src: url('https://s1.hdslb.com/bfs/static/jinkela/long/font/HarmonyOS_Medium.a1.woff2'); } #my_search_view { font-family: 'HarmonyOS', sans-serif !important; } .searchItem { background-image: url(); background-size: 100% 100%; background-clip: content-box; background-origin: content-box; } #my_search_input { animation-duration: 1s; animation-name: my_search_view; outline: none; border: none; box-shadow: none; } #my_search_input:focus{ outline: none; border: none; box-shadow: none; } .resultItem { animation-duration: 0.5s; animation-name: resultItem; } .resultItem .enter_main_link{ display: flex !important ; justify-content: start; align-items: center; flex-grow:3; } /*关联图标样式*/ .resultItem .vassal { /*对下面的svg位置进行调整*/ display: flex !important; align-items: center; flex-shrink:0; margin-right:2px; } .related-links { margin: 0 3px; display: flex; gap: 4px; /* 增加链接之间的间距 */ } .related-links > a { white-space: nowrap; line-height: 16px; font-size: 12px; padding: 3px 10px; background: #f0f4ff; color: #3578FE; text-decoration: none; transition: all 0.3s ease; } .related-links > a:hover { background: #e6f0ff; /* 悬停时背景颜色变浅 */ border-color: #cce1ff; /* 改变边框颜色 */ color: #255ec8; /* 深蓝色字体加深 */ box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); /* 增加阴影效果 */ } .resultItem svg{ width: 16px; height:16px; } @-webkit-keyframes my_search_view { 0% { width: 0px; } 50% { width: 50%; } 100% { width: 100%; } } @-webkit-keyframes resultItem { 0% { opacity: 0; } 40% { opacity: 0.6; } 50% { opacity: 0.7; } 60% { opacity: 0.8; } 100% { opacity: 1; } } /*简述超链接样式*/ #text_show a { color: #1a0dab !important; text-decoration:none; } /*自定义markdown的html样式*/ #text_show>p>code { padding: 2px 0.4em; font-size: 95%; background-color: rgba(188, 188, 188, 0.2); border-radius: 5px; line-height: normal; font-family: SFMono-Regular,Consolas,'Liberation Mono',Menlo,monospace; color: #558eda; } #my_search_input::placeholder { color: #757575; } /*简述文本颜色为统一*/ #text_show p { color: #202122; } /*让简述内容的li标签不受页面样式影响*/ #text_show > ul > li { list-style-type: disc !important; } #text_show > ul > li > ul > li { list-style-type: circle !important; } #text_show > ol > li { list-style-type: decimal !important; } /*当视图大于等于1400.1px时*/ @media (min-width: 1400.1px) { #my_search_box { left: 24%; right:24%; } } /*当视图小于等于1400px时*/ @media (max-width: 1400px) { #my_search_box { left: 20%; right:20%; } } /*当视图小于等于1200px时*/ @media (max-width: 1200px) { #my_search_box { left: 15%; right:15%; } } /*当视图小于等于800px时*/ @media (max-width: 800px) { #my_search_box { left: 10%; right:10%; } } /*输入框右边按钮*/ #logoButton { position: absolute; font-size: 12px; right: 5px; padding: 0px; border: none; display: block; background: rgba(255, 255, 255, 0); margin: 0px 7px 0px 0px; cursor: pointer !important; outline: none; } #logoButton:active { opacity: 0.4; } #logoButton img { display: block; width: 25px; } /*代码颜色*/ #text_show code,#text_show pre{ color:#5f6368; } /* 滚动条整体宽度 */ #text_show::-webkit-scrollbar, #text_show pre::-webkit-scrollbar { -webkit-appearance: none; width: 5px; height: 5px; } /* 滚动条滑槽样式 */ #text_show::-webkit-scrollbar-track, #text_show pre::-webkit-scrollbar-track { background-color: #f1f1f1; } /* 滚动条样式 */ #text_show::-webkit-scrollbar-thumb, #text_show pre::-webkit-scrollbar-thumb { background-color: #c1c1c1; } #text_show::-webkit-scrollbar-thumb:hover, #text_show pre::-webkit-scrollbar-thumb:hover { background-color: #a8a8a8; } #text_show::-webkit-scrollbar-thumb:active, #text_show pre::-webkit-scrollbar-thumb:active { background-color: #a8a8a8; } /*结果项样式*/ #matchResult li { line-height: 30.2px; height: 30.2px; color: #0088cc; list-style: none; width: 100%; padding: 0.5px; display: flex; justify-content: space-between; align-items: center; margin: 0.5px 0 !important; } #matchResult li > a { display: inline-block; font-size: 15.5px; text-decoration: none; text-align: left; cursor: pointer; font-weight: 400; background: rgb(255 255 255 / 0%); overflow: hidden; text-overflow: ellipsis; white-space: nowrap; margin: 0 3px; } #matchResult .flag { color: #fff; height: 21px; line-height: 21px; font-size: 10px; padding: 0px 6px; border-radius: 5px; font-weight: 600; box-sizing: border-box; margin-right: 3.5px; box-shadow: rgba(0, 0, 0, 0.3) 0px 0px 0.5px; } #matchResult .item_title { color: #1a0dab; } #matchResult .obsolete { text-decoration:line-through; color:#a8a8a8; } #matchResult .item_desc { color: #474747; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; } #matchResult img { display: inline-block; width: 24px; height: 24px; margin: 0 6px 0 3px; box-shadow: 0 0 2px rgba(0, 0, 0, 0.5); border-radius: 30%; box-sizing: border-box; padding: 3px; flex-shrink: 0; /* 当容量不够时,不压缩图片的大小 */ } #my_search_box { position: fixed;top:50px; border:2px solid #cecece;z-index:2147383656; background: #ffffff; } #my_search_box > #tis { position: absolute; left: 5px; top: -20px; font-size: 12px; color: #d5a436; font-weight: bold; } `) //防抖函数模板 function debounce(fun, wait) { let timer = null; return function (...args) { // 清除原来的定时器 if (timer) clearTimeout(timer) // 开启一个新的定时器 timer = setTimeout(() => { fun.apply(this, args) }, wait) } } // 判断是否为指定指令 function isInstructions(cmd) { let searchInputDocument = registry.view.element.input; if(searchInputDocument == null) return false; let regexString = "^\\s*:" + cmd + "\\s*$"; let regex = new RegExp(regexString,"i"); return regex.test(searchInputDocument.val()); } // 获取一个同步执行器实例 let pinyinActuator = syncActuator(); // 向数据项中加入拼音项 如:title加了titlePinyin, desc加了descPinyin function genDataItemPinyin(threadHandleItems){ let textPinyinMap = registry.searchData.getGlobalTextPinyinMap(); // console.logout("分配的预热item:",threadHandleItems) pinyinActuator(()=>{ if(threadHandleItems.length < 1) return; for(let item of threadHandleItems) { // 查看字典是否存在,只有没有预热过再预热 if( textPinyinMap[threadHandleItems.title] != null ) continue; item.title.toPinyin(); item.desc.toPinyin(); } // 持久化-textPinyinMap字典 (这里需要判断是否值已经被初始化) if(textPinyinMap != null ) { cache.jSet(registry.searchData.TEXT_PINYIN_KEY,textPinyinMap); } }); } // 当页面加载完成时触发-转拼音库操作 const refresh = debounce(()=>{ console.logout("==pinyin word==") let threadHandleItemSize = 100; let threadHandleItems = []; let currentSize = 0; let data = registry.searchData.getData(); for(let item of data) { // 加入处理容器中 threadHandleItems.push(item); currentSize++; // 判断是否已满 if(currentSize >= threadHandleItemSize || data[data.length-1] == item ) { // 已满-去操作 genDataItemPinyin(threadHandleItems); // 重置数据 currentSize = 0; threadHandleItems = []; } } }, 2000) registry.searchData.dataChangeEventListener.push(refresh); // 实现模块一:使用快捷键触发指定事件 function triggerAndEvent(goKeys = "ctrl+alt+s", fun, isKeyCode = false) { // 监听键盘按下事件 let handle = function (event) { let isCtrl = goKeys.indexOf("ctrl") >= 0; let isAlt = goKeys.indexOf("alt") >= 0; let lastKey = goKeys.replace("alt", "").replace("ctrl", "").replace(/\++/gm,"").trim(); // 判断 Ctrl+S if (event.ctrlKey != isCtrl || event.altKey != isAlt) return; if (!isKeyCode) { // 查看 lastKey == 按下的key if (lastKey.toUpperCase() == event.key.toUpperCase()) fun(); } else { // 查看 lastKey == event.keyCode if (lastKey == event.keyCode) fun(); } } // 如果使用 document.onkeydown 这种,只能有一个监听者 $(document).keyup(handle); } function codeCopyMount(elementSelector) { document.querySelectorAll(`${elementSelector} pre code`).forEach((codeBlock) => { // 创建复制按钮 const copyButton = document.createElement("button"); copyButton.innerText = "复制"; copyButton.classList.add("copy-btn"); // 复制代码逻辑 copyButton.addEventListener("click", () => { const text = codeBlock.innerText || codeBlock.textContent; navigator.clipboard.writeText(text).then(() => { copyButton.innerText = "已复制"; setTimeout(() => (copyButton.innerText = "复制"), 2000); }).catch(err => { console.error("复制失败:", err); }); }); // 让
 相对定位,以便按钮放在右上角
            const pre = codeBlock.parentElement;
            pre.style.position = "relative";

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

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

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

                        // 如果是最后一行,打包
                        let nextLine = lines[i+1];
                        if(i === lines.length-1 || ( nextLine != null && nextLine.indexOf("# ") == 0 )) {
                           // 加入resource,最后一项
                           current_build_search_item.resource = current_build_search_item_resource;
                           if(! isBlank(current_build_search_item_vassal)) {
                              current_build_search_item.vassal = current_build_search_item_vassal;
                           }
                           if(current_build_search_item_links.length > 0) {
                              current_build_search_item.links = current_build_search_item_links;
                           }
                           // 打包装箱
                           search_data_lines.push(current_build_search_item);
                           // 重置资源
                           appendTarget = "resource"
                           current_build_search_item_resource = "";
                           current_build_search_item_vassal = "";
                           current_build_search_item_links = [];
                        }
                    }
                    // 添加种类
                    for(let line of search_data_lines) {
                       line.type = type;
                    }
                    return search_data_lines;
             }
           
           
             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;
             }
          
        ` + getSubscribe();
        return new Promise(async (resolve,reject)=>{
            // 这里请求tishub datasources
            // [ {name: "官方订阅",body: "",status: ""} ] // status: disable enable
            const installHubTisList = cache.get(registry.searchData.USE_INSTALL_TISHUB_CACHE_KEY) || [];
            const installDataSources = installHubTisList.map(installTis => `${installTis.body}`).join("\n");
            resolve(installDataSources+localDataSources);
        })
    }


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

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

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

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

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

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

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

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

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

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

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


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

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

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

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

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

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

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

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

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

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

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

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

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

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

    // 模块二
    registry.view.viewVisibilityController = (function() {
        // 整个视图对象
        let viewDocument = null;
        let initView = function () {
            // 初始化视图
            let view = document.createElement("div")
            view.id = "my_search_box";
            let menu_icon = "";
            const matchResultDocumentId = "matchResult", textViewDocumentId = "text_show",searchInputDocumentId = "my_search_input",matchItemsId = "matchItems",searchBoxId = "searchBox",logoButtonId = "logoButton",msInputFilesId = "ms-input-files";
            view.innerHTML = (`
             
`) // 挂载到文档中 document.body.appendChild(view) // 整个视图对象放在组件全局中/注册表中 viewDocument = registry.view.viewDocument = view; // 想要追加请看下面registry.view.element是否已经包含,没有在那下面追加即可~ // 搜索框对象 let searchInputDocument = $(document.getElementById(searchInputDocumentId)), matchItems = $(document.getElementById(matchItemsId)), searchBox = $(document.getElementById(searchBoxId)), logoButton = $(document.getElementById(logoButtonId)), textView = $(document.getElementById(textViewDocumentId)), matchResult = $(document.getElementById(matchResultDocumentId)), msInputFiles = $(document.getElementById(msInputFilesId)); // 将视图对象放到注册表中 registry.view.element = { input: searchInputDocument, logoButton, matchItems, searchBox, matchResult, textView, files: msInputFiles } // 开启files 粘贴事件监听 registry.searchData.searchForFile.start(); // 菜单函数(点击输入框右边按钮时会调用) function onClickLogo() { // alert("小彩蛋:可以搜索一下“系统项”了解脚本基本使用哦~"); // 调用手动触发搜索函数,如果已经搜索过,搜索空串(清理) let keyword = "[系统项]"; registry.searchData.triggerSearchHandle(searchInputDocument.val()==keyword?'':keyword); // 重新聚焦搜索框 registry.view.element.input.focus() } const isLogoButtonPressedRef = registry.view.logo.isLogoButtonPressedRef; // 按下按钮时设置变量为 true logoButton.on('mousedown', function() { isLogoButtonPressedRef.value = true; }); // 按钮弹起时设置变量为 false,并让输入框聚焦 logoButton.on('mouseup', function() { isLogoButtonPressedRef.value = false; onClickLogo() // 触发logo点击事件 searchInputDocument.focus(); // 输入框聚焦 }); // 防止鼠标拖出按钮后弹起无法触发 mouseup logoButton.on('mouseleave', function() { if (isLogoButtonPressedRef.value) { isLogoButtonPressedRef.value = false; searchInputDocument.focus(); // 输入框聚焦 } }); // 设置视图已经初始化 registry.view.initialized = true; // 在搜索的结果集中上下选择移动然后回车(相当点击) searchInputDocument.keyup(function(event){ let keyword = $(event.target).val().trim(); // 当不为空时,放到全局keyword中 if(keyword) { registry.searchData.keyword = event.target.value; } // 处理keyword中的":"字符 if(keyword.endsWith("::") || keyword.endsWith("::")) { keyword = keyword.replace(/::|::/,registry.searchData.subSearch.searchBoundary).replace(/\s+/," "); // 每次要形成一个" : "的时候去掉重复的" : : " -> " : " keyword = keyword.replace(/((\s{1,2}:)+ )/,registry.searchData.subSearch.searchBoundary); $(event.target).val(keyword.toUpperCase()); } }); // searchInputDocument.keydown:这个监听用来处理其它键(非上下选择)的。 searchInputDocument.keydown(function (event){ // 阻止键盘事件冒泡 | 阻止输入框外监听到按钮,应只作用于该输入框 event.stopPropagation(); // 判断一个输入框的东西,如果如果按下的是删除,判断一下是不是"搜索模式" let keyword = $(event.target).val(); let input = event.target; if(event.key == "Backspace" ) { // 按的是删除键-块删除 if(keyword.endsWith(registry.searchData.subSearch.searchBoundary)) { // 取消默认事件-删除 event.preventDefault(); return; }else if(/^\s*[\[<][^\[\]<>]*[\]>]\s*$/.test( keyword )) { // 如果输入框只有[xxx]或那就清空掉输入框 searchInputDocument.val('') // keyword重置为空字符后触发搜索 registry.searchData.triggerSearchHandle(); event.preventDefault(); return; }else if(keyword === ""){ registry.searchData.searchForFile.delete(); } }else if ( ! event.shiftKey && event.keyCode === 9 ) { // Tab键 if(! registry.searchData.subSearch.isSubSearchMode()) { // 转大写 event.target.value = event.target.value.toUpperCase() // 添加搜索pro模式分隔符 event.target.value += registry.searchData.subSearch.searchBoundary // 阻止默认行为,避免跳转到下一个元素 registry.searchData.triggerSearchHandle(); } event.preventDefault(); }else if (event.shiftKey && event.keyCode === 9 ) { // 按下shift + tab键时取消搜索模式 if(registry.searchData.subSearch.isSubSearchMode()) { // 在这里编写按下shift+tab键时要执行的代码 let input = event.target; input.value = input.value.split(registry.searchData.subSearch.searchBoundary)[0] event.target.value = event.target.value.toLowerCase(); // 手动触发输入事件 input.dispatchEvent(new Event("input", { bubbles: true })); } event.preventDefault(); } }) // searchInputDocument.keydown:这个监听用来处理上下选择范围的操作 searchInputDocument.keydown(function (event){ let e = event || window.event; if(e && e.keyCode!=38 && e.keyCode!=40 && e.keyCode!=13) return; if(e && e.keyCode==38){ // 上 registry.searchData.pos --; } if(e && e.keyCode==40){ //下 registry.searchData.pos ++; } // 如果是回车 && registry.searchData.pos == 0 时,设置 registry.searchData.pos = 1 (这样是为了搜索后回车相当于点击第一个) if(e && e.keyCode==13 && registry.searchData.pos == 0){ // 回车选择的元素 // 如果当前是在搜索中就忽略回车这个操作 if(registry.searchData.searchEven.isSearching) return; registry.searchData.pos = 1; } // 当指针位置越出时,位置重定向 if(registry.searchData.pos < 1 || registry.searchData.pos > registry.searchData.searchData.length ) { if(registry.searchData.pos < 1) { // 回到最后一个 registry.searchData.pos = registry.searchData.searchData.length; }else { // 回到第一个 registry.searchData.pos = 1; } } // 设置显示样式 let activeItem = $(registry.view.element.matchItems.find('li')[registry.searchData.pos-1]); // 设置活跃背景颜色 let activeBackgroundColor = "#dee2e6"; activeItem.css({ "background":activeBackgroundColor }) // 设置其它子元素背景为默认统一背景 activeItem.siblings().css({ "background":"#fff" }) // 看是不是item detail内容显示中,如果是回车发送send事件,否则才是结果集显示的回车选择 if(e && e.keyCode==13 && activeItem.find("a").length > 0 && !registry.script.tryRunTextViewHandler()){ // 回车 // 点击当前活跃的项,点击 activeItem.find("a")[0].click(); } // 取消冒泡 e.stopPropagation(); // 取消默认事件 e.preventDefault(); }); // 将输入框的控制按钮设置可见性函数公开放注册表中 registry.view.setButtonVisibility = function (buttonVisibility = false) { // registry.view.setButtonVisibility logoButton.css({ "display": buttonVisibility?"block":"none" }) } // 高权重项特殊搜索关键词直达 registry.searchData.searchEven.event[registry.searchData.specialKeyword.highFrequency] = function(search,rawKeyword) { return DataWeightScorer.highFrequency(45); } // 历史记录特殊搜索关键词直达 registry.searchData.searchEven.event[registry.searchData.specialKeyword.history] = function(search,rawKeyword) { return SelectHistoryRecorder.history(15); } // 向搜索事件(只会触发一个)中添加一个“NEW”搜索关键词 registry.searchData.searchEven.event["new|"+registry.searchData.specialKeyword.new] = function(search,rawKeyword) { let showNewData = null; let activeSearchData = registry.searchData.getData(); // 如果当前注册表中全局搜索数据为空,使用缓存的数据 if(activeSearchData == null ) { let cacheAllSearchData = cache.get(registry.searchData.SEARCH_DATA_KEY); if(cacheAllSearchData != null && cacheAllSearchData.data != null) activeSearchData = cacheAllSearchData.data; } // 如果最新数据都没有,使用旧数据(上一次) if(activeSearchData == null ) { let oldCacheAllSearchData = cache.get(registry.searchData.OLD_SEARCH_DATA_KEY); if(oldCacheAllSearchData != null) activeSearchData = oldCacheAllSearchData; } // 只展示 newItems 数据中data也存在的项 let newItems = cache.get(registry.searchData.SEARCH_NEW_ITEMS_KEY)??[]; if(newItems.length > 0 && activeSearchData.length > 0) { // 返回的showNewData是左边的(activeSearchData),而不是右边的(newItems),但newItems多出来 的属性也会合并到activeSearchData的item showNewData = compareArrayDiff(activeSearchData,newItems,registry.searchData.idFun,0) } if(showNewData == null) return []; // 对数据进行排序 showNewData.sort(function(item1, item2){return item2.expires - item1.expires}); showNewData.map((item,index)=>{ let dayNumber = registry.searchData.NEW_DATA_EXPIRE_DAY_NUM; // 去掉[新] 再都加[新],使得就算没有也在显示时也是有新标签的 item.title = registry.searchData.NEW_ITEMS_TAG+item.title.toReplaceAll(registry.searchData.NEW_ITEMS_TAG,"") // 添加“几天前” item.title = item.title + " | " + Math.floor( (Date.now() - (item.expires - 1000*60*60*24*dayNumber) )/(1000*60*60*24) )+"天前"; //toDateString return item; }) // 将最新的一条由“新”改为“最新一条” showNewData[0].title = showNewData[0].title.toReplaceAll(registry.searchData.NEW_ITEMS_TAG,"[最新一条]") return showNewData; } // 可填充搜索模式优先路由(key是正则字符串,value为字符串类型是转发,如果是函数,是自定义搜索逻辑) const searchableSpecialRouting = { "^\\s*$": "问AI", "^问AI$": async (search,rawKeyword,keywordForFill0)=>{ return await search(keywordForFill0,{isAccurateSearch : true}); } } // 返回undfind表示没有定义匹配对应的SpecialRouting,执行通用路由 | null表示跳过 | 返回数组表示SpecialRouting执行搜索得到的结果 const searchableSpecialRoutingHandler = async function(search,rawKeyword){ const keywordForFill0 = registry.searchData.subSearch.getParentKeyword(rawKeyword); for(let key of Object.keys(searchableSpecialRouting)) { if(isMatch(key,keywordForFill0)) { const value = searchableSpecialRouting[key]; if(typeof value === "string") { registry.searchData.triggerSearchHandle(value+registry.searchData.subSearch.searchBoundary) return []; } if(typeof value === "function") return await value(search,rawKeyword,keywordForFill0); } } // 表示没有匹配到SpecialRouting return undefined; } registry.searchData.searchEven.event[".*"+registry.searchData.subSearch.searchBoundary+".*"] = async function(search,rawKeyword) { const specialRoutinResult = await searchableSpecialRoutingHandler(search,rawKeyword) // 当没有优先Result, 只搜索“可搜索”项 return Array.isArray(specialRoutinResult) ? specialRoutinResult : await search(`${registry.searchData.searchProTag} ${registry.searchData.subSearch.getParentKeyword()}`); } // 搜索AOP async function searchAOP(search,rawKeyword) { // 转发到对应的AOP处理器中(keyword规则订阅者) let data = registry.searchData.getData(); console.log("搜索data:",data) return await registry.searchData.searchEven.send(search,rawKeyword); } function searchUnitHandler(beforeData = [],keyword = "") { // 触发搜索事件 for(let e of registry.searchData.onSearch) e(keyword); // 如果没有搜索内容,返回空数据 keyword = keyword.trim().toUpperCase(); if(keyword == "" || registry.searchData.getData().length == 0 ) return []; // 切割搜索内容以空格隔开,得到多个 keyword let searchUnits = keyword.split(/\s+/); // 弹出一个 keyword keyword = searchUnits.pop(); // 本次搜索的总数据容器 let searchResultData = []; let searchLevelData = [ [],[],[] // 分别是匹配标题/desc/url 的结果 ] // 数据出来的总数据 //let searchData = [] // 前置处理函数,这里使用观察者模式 // searchPreFun(keyword); // 搜索操作 // 为实现当关键词只有一位时,不使用转拼音搜索,后面搜索涉及到的转拼音操作要使用它,而不是直接调用toPinyin function getPinyinByKeyword(str,isOnlyFomCacheFind=false) { if(registry.searchData.keyword.length > 1 ) return str.toPinyin(isOnlyFomCacheFind)??""; return str??""; } let pinyinKeyword = getPinyinByKeyword(keyword); let searchBegin = Date.now() for (let dataItem of beforeData) { /* 取消注释会导致虽然是15条,但有些匹配度高的依然不能匹配 // 如果已达到搜索要显示的条数,则不再搜索 && 已经是本次最后一次过滤了 => 就不要扫描全部数据了,只搜出15条即可 let currentMeetConditionItemSize = searchLevelData[0].length + searchLevelData[1].length + searchLevelData[2].length; if(currentMeetConditionItemSize >= registry.searchData.showSize && searchUnits.length == 0 && registry.searchData.subSearch.isSubSearchMode() ) break; */ // 将数据放在指定搜索层级数据上DeepSeek if(dataItem.title.includes("DeepSeek")) debugger; if ( (( getPinyinByKeyword(dataItem.title,true).includes(pinyinKeyword) || dataItem.title.toUpperCase().includes(keyword) ) && searchLevelData[0].push(dataItem) ) || (( getPinyinByKeyword(dataItem.desc,true).includes(pinyinKeyword) || dataItem.desc.toUpperCase().includes(keyword)) && searchLevelData[1].push(dataItem) ) || ( `${registry.searchData.links.stringifyForSearch(dataItem.links)}${dataItem.resource}${dataItem.vassal}`.substring(0, 4096).toUpperCase().includes(keyword) && searchLevelData[2].push(dataItem) ) ) {} // 若满足条件的数据对象则会添加到对应的盒子中 } let searchEnd = Date.now(); console.logout("常规搜索主逻辑耗时:"+(searchEnd - searchBegin ) +"ms"); // 将上面层级数据进行权重排序然后放在总容器中 searchResultData.push(...DataWeightScorer.sort(searchLevelData[0],registry.searchData.idFun)); searchResultData.push(...DataWeightScorer.sort(searchLevelData[1],registry.searchData.idFun)); searchResultData.push(...DataWeightScorer.sort(searchLevelData[2],registry.searchData.idFun)); if(searchUnits.length > 0 && searchUnits[searchUnits.length-1].trim() != registry.searchData.subSearch.searchBoundary.trim()) { // 递归搜索 searchResultData = searchUnitHandler(searchResultData,searchUnits.join(" ")); } return searchResultData; } // ==标题tag处理== // 1、标题tag颜色选择器 function titleTagColorMatchHandler(tagValue) { let vcObj = { "系统项":"background:rgb(0,210,13);", "非最佳":"background:#fbbc05;", "推荐":"background:#ea4335;", "装机必备":"background:#9933E5;", "好物":"background:rgb(247,61,3);", "安卓应用":"background:#73bb56;", "Adults only": "background:rgb(244,201,13);", "可搜索":"background:#4c89fb;border-radius:0px !important;", "新":"background:#f70000;", "最新一条":"background:#f70000;", "精选好课":"background:#221109;color:#fccd64 !important;" }; let resultTagColor = "background:#5eb95e;"; Object.getOwnPropertyNames(vcObj).forEach(function(key){ if(key == tagValue) { resultTagColor = vcObj[key]; } }); return resultTagColor; } // 2、标题内容处理程序 function titleTagHandler(title) { if(!(/[\[]?/.test(title) && /[\]]?/.test(title))) return -1; // 格式是:[tag]title 这种的 const regex = /(\[[^\[\]]*\])/gm; let m; let resultTitle = title; while ((m = regex.exec(title)) !== null) { // 这对于避免零宽度匹配的无限循环是必要的 if (m.index === regex.lastIndex) { regex.lastIndex++; } let tag = m[0]; if(tag == null || tag.length == 0) return -1; let tagCore = tag.substring(1,tag.length - 1); // 正确提取 resultTitle = resultTitle.toReplaceAll(tag,`${tagCore}`); } return resultTitle; } // 3、添加标题处理器 titleTagHandler registry.view.titleTagHandler.handlers.push(titleTagHandler) // 给输入框加事件 // 执行 debounce 函数返回新函数 let handler = async function (e) { // 搜索使用的数据版本 let version = registry.searchData.version; let rawKeyword = e.target.value; // 在本次搜索加入到历史前检查(如果之前是子搜索模式且现在还是子搜索模式那就跳过搜索,因为内是子搜索内容被修改不进行搜索) if(registry.searchData.subSearch.isEnteredSubSearchMode && registry.searchData.subSearch.isSubSearchMode() && registry.searchData.searchHistory.seeCurrentEqualsLastByRealKeyword()) return; // 添加到搜索历史(维护这个历史有用是为了子搜索模式的“进”-“出”) registry.searchData.searchHistory.add(rawKeyword) // 字符串重叠匹配度搜索(类AI搜索) async function stringOverlapMatchingDegreeSearch(rawKeyword) { const endTis = registry.view.tis.beginTis("(;`O´)o 匹配度模式搜索中...") // 这里为什么要用异步,不果不会那上面设置的tis会得不到渲染,先保证上面已经渲染完成再执行下面函数 return await new Promise((resolve,reject)=>{ waitViewRenderingComplete(() => { try { // 搜索逻辑开始 // `registry.searchData.getData()`会被排序desc // 为什么需要拷贝data,因为全局的搜索位置不能改变!! const searchBegin = Date.now(); let searchResult = overlapMatchingDegreeForObjectArray(rawKeyword.toUpperCase(),[...registry.searchData.getData()], (item)=>{ const str2ScopeMap = {} const { tags , cleaned } = extractTagsAndCleanContent(`${item.title}`); str2ScopeMap[cleaned.toUpperCase()] = 4; str2ScopeMap[`${item.describe}${tags.join()}`.toUpperCase()] = 2; str2ScopeMap[`${item.links && registry.searchData.links.stringifyForSearch(item.links)}${item.resource}${item.vassal}`.substring(0, 4096).toUpperCase()] = 1; return str2ScopeMap; },"desc",{sort:"desc",onlyHasScope:true}); const searchEnd = Date.now(); console.log("启动类AI搜索结果 :",searchResult) console.logout("类AI搜索主逻辑耗时:"+(searchEnd - searchBegin ) +"ms"); resolve(searchResult) }catch (e) { console.error("类AI搜索异常!",e) resolve([]) }finally { endTis() } }) }) } // 常规方式搜索(搜索逻辑入口) async function search(rawKeyword,{isAccurateSearch = false} = {}) { let processedKeyword = rawKeyword.trim().split(/\s+/).reverse().join(" "); version = registry.searchData.version; // 常规搜索 let searchResult = searchUnitHandler(registry.searchData.getData(),processedKeyword); // 如果常规搜索不到使用类AI搜索(不能是精确搜索 && 常规搜索没有结果 && 搜索keyword不为空串) if(!isAccurateSearch && (searchResult == null || searchResult.length === 0) && `${rawKeyword}`.trim().length > 0 ) { searchResult = await stringOverlapMatchingDegreeSearch(rawKeyword) } return searchResult; } // 搜索AOP或说搜索代理 // 递归搜索,根据空字符切换出来的多个keyword // let searchResultData = searchUnitHandler(registry.searchData.data,key) let searchResultData = await searchAOP(search,rawKeyword); // 如果搜索的内容无效,跳过内容的显示 if(searchResultData == null) return; // 放到视图上 // 置空内容 matchItems.html("") // 最多显示条数 let show_item_number = registry.searchData.showSize ; function getFaviconImgHtml(searchResultItem) { if(searchResultItem == null) return null; let resource = searchResultItem.resource.trim(); let customIcon = null; if(searchResultItem.icon != null) { customIcon = searchResultItem.icon; }else { let type = searchResultItem.type; // 如果不是url,那其它类型就需要自定义图标 let typesAndImg = { "sketch":"", "script":"" } // url与sketch类型可互转,主要看resource type = (type == "url" || type == "sketch")?(isUrl(resource)?"url":"sketch"):type; if(type != "url") customIcon = typesAndImg[type]; } if(customIcon != null) { return `` }else { return `` } } // 标题内容处理器 function titleContentHandler(title) { // 对标题去掉所有tag const { cleaned } = extractTagsAndCleanContent(title) title = cleaned // 如果带#将加上删除线,通过加obsolete类名方式 return `${title.replace(/^#/,"")}`; } let matchItemsHtml = ""; // 真正渲染到列表的数据项 let searchData = [] for(let searchResultItem of searchResultData ) { // 限制条数 if(show_item_number-- <= 0 && !registry.searchData.isSearchAll) { break; } // 显示时清理标签-虽然在加载数据时已经清理了,但这是后备方案 // clearHideTag(searchResultItem); // 将数据放入局部容器中 searchData.push(searchResultItem) let isSketch = !isUrl(searchResultItem.resource);// searchResultItem.resource.trim().toUpperCase().indexOf("HTTP") != 0; let vassalSvg = ``; // 构建快捷link html function buildRelatedLinksHtml(links) { if (links == null || links.length === 0) return ''; let html = `'; return html; } // 将符合的数据装载到视图 let item = `
  • ${getFaviconImgHtml(searchResultItem)} ${registry.view.titleTagHandler.execute(clearHideTagForTitle(searchResultItem.title))}${titleContentHandler(searchResultItem.title)} (${searchResultItem.desc}) ${buildRelatedLinksHtml(searchResultItem.links)} ${searchResultItem.vassal !=null?''+vassalSvg+'':''}
  • ` matchItemsHtml += item; } matchItems.html(matchItemsHtml); let loadErrorTagIcon = ""; // 给刚才添加的img添加事件 for(let imgObj of registry.view.element.matchItems.find('img')) { // 加载完成事件,去除加载背景 imgObj.onload = function(e) { $(e.target).css({ "background": "#fff" }) } // 加载失败,设置自定义失败的本地图片 imgObj.onerror = function(e,a,b,c) { let currentErrorImg = $(e.target); let standbyFaviconAttr = "standbyFavicon"; let standbyFavicon = currentErrorImg.attr(standbyFaviconAttr); if(standbyFavicon != null) { // 如果备用favicon使用 currentErrorImg.attr("src",standbyFavicon) currentErrorImg.removeAttr(standbyFaviconAttr) }else { // 如果备用favicon直接使用加载失败图标base64 currentErrorImg.attr("src",loadErrorTagIcon) } } } // 隐藏文本显示视图 textView.css({ "display":"none" }) // 让搜索结果显示 let matchResultDisplay = "block"; if(searchResultData.length < 1) matchResultDisplay="none"; matchResult.css({ "display":matchResultDisplay, "overflow":"hidden" }) // 将显示搜索的数据放入全局容器中 registry.searchData.searchData = searchData; // 指令归位(置零) registry.searchData.pos = 0; } registry.view.element.matchItems.on("click","li > a",function(e) { let targetObj = e.target; // 如果当前标签是svg标签,那委托给父节点 while ( targetObj != null && !/^(a|A)$/.test(targetObj.tagName)) { targetObj = targetObj.parentNode } // 取消默认事件,全部都是手动操作 e.preventDefault(); // 取消冒泡 window.event? window.event.cancelBubble = true : e.stopPropagation(); // 设置为阅读模式 // $("#my_search_input").val(":read"); // 获取当前结果在搜索数组中的索引 let dataIndex = parseInt($(targetObj).attr("index")); let dataVersion = parseInt($(targetObj).attr("version")); let currentSearchDataVersion = registry.searchData.version; let itemData = registry.searchData.getData()[dataIndex]; if(itemData == null || dataVersion != currentSearchDataVersion ) { console.log("后备方案(没有找到了?"+(itemData == null)+",数据版本改变了?"+(dataVersion != currentSearchDataVersion)+")") // 索引出现问题-启动后备方案-全局搜索 let title = $(targetObj).parent().find(".item_title").text(); let desc = $(targetObj).parent().find(".item_desc").text(); // 从全局数据中根据title与desc进行匹配 itemData = registry.searchData.findSearchDataItem(title,desc) // 从历史数据中找,根据title与desc进行匹配 if(itemData == null) itemData = registry.searchData.findSearchDataItem(title,desc,SelectHistoryRecorder.history) } // 给选择的item加分,便于后面调整排序 (这里的idFun使用注册表中已经有的,也是我们确认item唯一的函数) if(itemData != null) DataWeightScorer.select(itemData,registry.searchData.idFun); // 记录选择的item项 SelectHistoryRecorder.select(itemData,registry.searchData.idFun); // === 如果是简述搜索信息,那就取消a标签的默认跳转事件=== let hasVassal = $(targetObj).attr("vassal") != null; // 初始化textView注册表中的对象 function showTextPage(title,desc,body) { registry.view.textView.show(`标题:${title}
    描述:${desc}
    简述内容:
    ${md2html(body)} `) } if(hasVassal) { showTextPage(itemData.title,"主项的相关/附加内容",itemData.vassal); // 挂载一键code复制 codeCopyMount("#text_show"); return; }else if(itemData.type == "script"){ // 是脚本,执行脚本 let callBeforeParse = new CallBeforeParse(); let jscript = ( itemData.resourceObj == null || itemData.resourceObj.script == null ) ?"function (obj) {alert('- _ - 脚本异常!')}":itemData.resourceObj.script; // 调用里面的函数,传入注册表对象 // 打开网址函数 function open(url) { let openUrl = url; return { simulator(operate = (click, roll, dimension) => {}) { // 模拟器 if(openUrl == null || operate == null || typeof operate != 'function') return; let pageSimulatorScript = operate.toString(); addPageSimulatorScript(openUrl,pageSimulatorScript); // 保存模拟操作,模拟脚本将在指定时间内打开指定网址有效 window.open(openUrl); // 打开网址 return this; } } } let view = { beforeCallback: null, afterCallback: null, mountBefore(handle) { this.beforeCallback = handle; return this; }, mountAfter(handle) { this.afterCallback = handle; return this; }, // mount是脚本项-脚本js调用 mount() { // 看脚本js是否给beforeCallback ,如果有在此执行 if(this.beforeCallback != null) this.beforeCallback(); // 挂载MS_SCRIPT_ENV 实现系统脚本API到视图 registry.script.openSessionForMSSE(); // 挂载视图 let viewHtml = itemData.resourceObj['view:html']; let viewCss = itemData.resourceObj['view:css']; let viewJs = itemData.resourceObj['view:js']; registry.view.textView.show(viewHtml,viewCss,viewJs); // wait view complate alfter ... waitViewRenderingComplete(()=>{ registry.script.tryRunTextViewHandler(); // 看脚本js是否给afterCallback ,如果有在此执行 if(this.afterCallback != null) this.afterCallback(); }) } } // 设置logo为运行图标 registry.view.logo.change("") try { Function('obj',`(${jscript})(obj)`)({registry,cache,$,open,view}) } catch (error) { setTimeout(()=>{alert("Ծ‸Ծ 你选择的是脚本项,而当前页面安全策略不允许此操作所依赖的函数!这种情况是极少数的,请换个页面试试!")},20) console.logout("脚本执行失败!",error); } // logo图标还原 setTimeout(()=>{registry.view.logo.reset();},200) return; }else if(! isUrl(itemData.resource)) { showTextPage(itemData.title,itemData.desc,itemData.resource) return; } // 隐藏视图 registry.view.viewVisibilityController(false) const initUrl = itemData.resource;//$(targetObj).attr("href"); // 不作改变的URL let url = initUrl; // 进行修改,形成要跳转的真正url let temNum = url.matchFetch(/\[\[[^\[\]]*\]\]/gm, function (matchStr,index) { // temNum是url中有几个 "[[...]]", 得到后,就已经得到解析了 let templateStr = matchStr; // 使用全局的keyword, 构造出真正的keyword let keyword = registry.searchData.keyword.split(":").reverse(); keyword.pop(); keyword = keyword.reverse().join(":").trim(); let parseAfterStr = matchStr.replace(/{keyword}/g,keyword).replace(/\[\[+|\]\]+/g,""); url = url.replace(templateStr,parseAfterStr); }); // 如果搜索的真正keyword为空字符串,则去掉模板跳转 if( registry.searchData.keyword.split(registry.searchData.subSearch.searchBoundary).length < 2 || registry.searchData.keyword.split(registry.searchData.subSearch.searchBoundary)[1].trim() == "" ) { url = registry.searchData.clearUrlSearchTemplate(initUrl); } // 跳转(url如果有模板,可能已经去掉模板,取决于是“搜索模式”) window.open(url); }) //registry.searchData.searchHandle = handler; const refresh = debounce(handler, 300) // 第一次触发 scroll 执行一次 fn,后续只有在停止滑动 1 秒后才执行函数 fn searchInputDocument.on('input', refresh) } function ensureViewHide() { // 隐藏视图 // 如果视图还没有初始化,直接退出 if (! registry.view.initialized) return; // 如果正在查看查看“简讯”,先退出简讯 const nowMode = registry.view.seeNowMode(); if(nowMode === registry.view.modeEnum.SHOW_ITEM_DETAIL) { // 让简讯隐藏 registry.view.element.textView.css({"display":"none"}) // 让搜索结果显示 registry.view.element.matchResult.css({ display:"block",overflow: "hidden" }) // 通知简讯back事件 registry.view.itemDetailBackAfterEventListener.forEach(listener=>listener()) return; } // 让视图隐藏 viewDocument.style.display = "none"; // 将输入框内容置空,在置空前将值备份,好让未好得及的操作它 registry.view.element.input.val("") // 将之前搜索结果置空 registry.view.element.matchItems.html("") // 隐藏文本显示视图 registry.view.element.textView.css({ "display":"none" }) // 让搜索结果隐藏 registry.view.element.matchResult.css({ "display":"none" }) // 视图隐藏-清理旧数据 registry.searchData.clearData(); // 触发视图隐藏事件 registry.view.viewHideEventAfterListener.forEach(fun=>fun()); } function showView() { // 让视图可见 viewDocument.style.display = "block"; //聚焦 registry.view.element.input.focus() // 当输入框失去焦点时,隐藏视图 registry.view.element.input.blur(function() { const isLogoButtonPressedRef = registry.view.logo.isLogoButtonPressedRef if(isLogoButtonPressedRef.value) { console.logout("隐藏跳过,因为isLogoButtonPressedRef") return }; setTimeout(function(){ const isDebuging = isInstructions("debug"); const isSearching = registry.searchData.searchEven.isSearching; // 当前视图是否在展示数据,如搜索结果,简述内容?如果在展示不隐藏 let isWaitSearch = registry.view.seeNowMode() === registry.view.modeEnum.WAIT_SEARCH; if(isDebuging || isSearching || !isWaitSearch || isLogoButtonPressedRef.value) { console.logout("隐藏跳过,条件列表不满足!") return }; registry.view.viewVisibilityController(false); },registry.view.delayedHideTime) }); } // 返回给外界控制视图显示与隐藏 return function (isSetViewVisibility) { if (isSetViewVisibility) { // 让视图可见 >>> // 如果还没初始化先初始化 // 初始化数据 initData(); if (! registry.view.initialized) { // 初始化视图 initView(); // 初始化数据 // initData(); } // 让视图可见 showView(); } else { // 隐藏视图 >>> ensureViewHide(); } } })(); // 触发策略——快捷键 let useKeyTrigger = function (viewVisibilityController) { let isFirstShow = true; // 将视图与触发策略绑定 function showFun() { // 让视图可见 viewVisibilityController(true); // 触发视图首次显示事件 if(isFirstShow) { for(let e of registry.view.viewFirstShowEventListener) e(); isFirstShow = false; } } window.addEventListener('message', event => { // console.log("父容器接收到了信息~~") if(event.data == MY_SEARCH_SCRIPT_VIEW_SHOW_EVENT) { showFun() // 接收显示呼出搜索框 } }); triggerAndEvent("ctrl+alt+s", showFun) triggerAndEvent("Escape", function () { // 如果视图还没有初始化,就跳过 if(registry.view.viewDocument == null ) return; // 让视图不可见 viewVisibilityController(false); }) } // 触发策略组 let trigger_group = [useKeyTrigger]; // 初始化入选的触发策略 (function () { for (let trigger of trigger_group) { trigger(registry.view.viewVisibilityController); } })(); // 打开视图进行配置 // 显示配置视图 // 是否显示进度 - 进度控制 function clearCache() { cache.remove(registry.searchData.SEARCH_DATA_KEY); // 如果处于debug模式,也清理其它的 if(isInstructions("debug")) { cache.remove(registry.searchData.CACHE_FAVICON_SOURCE_KEY); } // 触发缓存被清理事件 for(let fun of registry.searchData.dataCacheRemoveEventListener) fun(); } GM_registerMenuCommand("订阅管理",function() { showConfigView(); }); GM_registerMenuCommand("清理缓存",function() { clearCache(); }); function giveTagsStatus(tagsOfData,userUnfollowList) { // 赋予tags一个是否选中状态 // 将 userUnfollowList 转为以key为userUnfollowList的item.name值是Item的方便检索 let userUnfollowMap = userUnfollowList.reduce(function(result, item) { result[item] = ''; return result; }, {}); tagsOfData.forEach(item=>{ if(userUnfollowMap[item.name] != null ) { // 默认都是选中状态,如果item在userUnfollowList上将此tag状态改为未选中状态 item.status = 0; } }) return tagsOfData; } function showConfigView() { // 剃除已转关注的,添加新关注的 function reshapeUnfollowList(userUnfollowList,userFollowList,newUserUnfollowList) { // 剃除已转关注的 userUnfollowList = userUnfollowList.filter(item => !userFollowList.includes(item)); // 添加新关注的 userUnfollowList = userUnfollowList.concat(newUserUnfollowList.filter(item => !userUnfollowList.includes(item))); return userUnfollowList; } if($("#subscribe_save")[0] != null) return; // 显示视图 // 用户维护的取消关注标签列表 let userUnfollowList = cache.get(registry.searchData.USER_UNFOLLOW_LIST_CACHE_KEY)?? registry.searchData.USER_DEFAULT_UNFOLLOW; // 当前数据所有的标签 let tagsOfData = cache.get(registry.searchData.DATA_ITEM_TAGS_CACHE_KEY); // 使用 userUnfollowList 给 tagsOfData中的标签一个是否选中状态,在userUnfollowList中不选中,不在选中,添加一个属性到tagsOfData用boolean表达 tagsOfData = giveTagsStatus(tagsOfData,userUnfollowList); // 生成多选框html let tagsCheckboxHtml = ""; tagsOfData.forEach(item=>{ tagsCheckboxHtml += `
    ` }) DivPage(` #my-search-view { width: 500px; max-height: 100%; max-width: 100%; background: pink; position: fixed; right: 0px; top: 0px; z-index: 2147383656; padding: 20px; box-sizing: border-box; border-radius: 14px; text-align: left; button { cursor: pointer; } ._topController { width: 100%; position: absolute; top: 0px; right: 0px; text-align: right; padding: 15px 15px 0px; box-sizing: border-box; * { cursor: pointer; } #topController_close { font-sise: 15px; color: #e8221e; } } .page { .control_title { margin: 10px 0px 5px; font-size: 17px; color: black; } } .home { .submitable { color: #3CB371; } .tagsCheckBoxDiv > div { width: 32%; display: inline-block; margin: 0px; padding: 0px; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; } #all_subscribe { width: 100%; height: 150px; box-sizing: border-box; border: 4px solid #f5f5f5; } #subscribe_save { margin-top: 20px; border: none; border-radius: 3px; padding: 4px 17px; cursor: pointer; box-sizing: border-box; background: #6161bb; color: #fff; } .view-base-button { background: #fff; border: none; font-size: 15px; padding: 1px 10px; cursor: pointer; margin: 2px; color: black; } ._topController span { color: #3CB371; } .home label { font-size: 13px; } } .tis-hub { .logo-search { display: flex; flex-direction: column; align-items: center; img { display: block; width: 40px; height: 40px; } .keyword { display: flex; font-size: 12px; width: 70%; margin-top: 5px; input { border: none; padding: 0 6px; min-width: 100px; line-height: 25px; height: 25px; flex-grow: 1; } button { padding: 0 12px; border: none; background: #f0f0f0; line-height: 25px; height: 25px; } } } .search-type { display: flex; padding: 10px 0; label { display: flex; align-items: center; margin-right: 20px; font-size: 14px; input { padding: 0; margin: 0 3px 0 0; } } } .result-list { min-height: 300px; padding-top: 15px; .hub-tis { display: flex; justify-content: space-between; margin-bottom: 12px; align-items: center; button { font-size: 10px; line-height: 22px; height: 22px; padding: 0 15px; border-radius: 3px; border: none; } .tis-info { display: flex; flex-direction: column; .title { font-size: 14px; font-weight: bold; color: rgb(103, 0, 0); } .describe { font-size: 12px; font-weight: 400; display: block; font-size: smaller; margin: 0.5em 0px; color: #333333; } } } } } } `,`
    X

    订阅总览:

    公共仓库:

    关注标签:

    ${tagsCheckboxHtml}

    订阅市场


    `,function (selector,remove) { let subscribe_text = selector("#all_subscribe"); let subscribe_save = selector("#subscribe_save"); let topController_close = selector("#topController_close"); let openTisHub = selector("#openTisHub"); let tisHubLink = "https://github.com/My-Search/TisHub/issues"; let pushTis = selector("#pushTis"); let commitableTisList = null; let clearToken = selector("#clearToken"); let mySearchView = selector("#my-search-view"); let currentPage = setPage(); // 默认显示的是home页 // 刷新页 function setPage(page = "home") { $(mySearchView).find('.page').hide().filter(`.${page}`).show(); } setPage("home"); // 刷新视图状态 async function refreshViewState() { // 更新token状态 $(clearToken).css({"display":GithubAPI.getToken() == null?"none":"inline-block"}) // 更新可提交数 let tisList = await TisHub.getTisHubAllTis(); if(tisList != null && tisList.length != 0) { commitableTisList = TisHub.tisFilter(subscribe_text.value,tisList)??[] $(pushTis).find("span").text(commitableTisList.length); } } // 初始化subscribe_text的值 subscribe_text.value = getSubscribe(); // 初始化其它状态,通过调用refreshViewState() refreshViewState(); // 当SubscribeText多行输入框内容发生改变时,刷新更新可提交数,通过调用refreshViewState() let refreshSubscribeText = debounce(()=>{refreshViewState() }, 300) subscribe_text.oninput = ()=>{refreshSubscribeText();} // 保存 function configViewClose() { remove(); } // 点击保存时 subscribe_save.onclick=function() { // 保存用户选择的关注标签(维护数据) // 获取所有多选框元素 var checkboxes = selector(".tagsCheckBoxDiv input",true); // 初始化已选中和未选中的数组 var userFollowList = []; var newUserUnfollowList = []; // 遍历多选框元素,将选中的元素的value值添加到checkedValues数组中, // 未选中的元素的value值添加到uncheckedValues数组中 for (var i = 0; i < checkboxes.length; i++) { if (checkboxes[i].checked) { userFollowList.push(checkboxes[i].value); } else { newUserUnfollowList.push(checkboxes[i].value); } } // 剃除已转关注的,添加新关注的 newUserUnfollowList = reshapeUnfollowList( userUnfollowList,userFollowList,newUserUnfollowList); cache.set(registry.searchData.USER_UNFOLLOW_LIST_CACHE_KEY,newUserUnfollowList); // 保存到对象 let allSubscribe = subscribe_text.value; let validCount = editSubscribe(allSubscribe); // 清除视图 configViewClose(); // 清理缓存,让数据重新加载 clearCache(); alert("保存配置成功!有效订阅数:"+validCount); } // 打开TitHub openTisHub.onclick = function() { // window.open(tisHubLink, "_blank"); setPage("tis-hub"); } // push到TisHub公共仓库中 pushTis.onclick =async function () { if(! confirm("是否确认要提交到TisHub公共仓库?")) return; if(commitableTisList == null || commitableTisList.length == 0) { alert("经过与TisHub中订阅的比较,本地没有可提交的订阅!") return; } if(GithubAPI.getToken(true) == null) { alert("获取token失败,无法继续!"); return; } // 组装提交的body let body = (()=>{ let _body = ""; for(let tis of commitableTisList) _body+=tis; return _body; })(); if ( body == "") return; let userInfo = await GithubAPI.setToken().getUserInfo(); if(userInfo == null) { alert("提交异常,请检查网络或提交的Token信息!") return; } GithubAPI.commitIssues({ "title": userInfo.name+"的订阅", "body": body }).then(response=>{ refreshViewState(); alert("提交成功(issues)!感谢您的参与,脚本因你而更加精彩。") }).catch(error=>alert("提交失败~")) } // 清理token clearToken.onclick = function(){ GithubAPI.clearToken(); // 清理token refreshViewState(); // 刷新视图变量 }; // 关闭 $(topController_close).click(configViewClose) // 点击搜索tis-hub let installedList = cache.get(registry.searchData.USE_INSTALL_TISHUB_CACHE_KEY) || []; // [ {name: "官方订阅",describe: "这是官方订阅...", body: "",status: ""} ] status: disable enable installable let tisSearchInput = $(".tis-hub .keyword input"); let tisSearchBtn = $("#search-tishub"); let searchFun = async function() { const keyword = tisSearchInput.val()?.trim() || ''; // 搜索类型(installed | market) const searchType = $('.search-type input[name="search-type"]:checked').val(); let resultTisList = installedList.filter(item => keyword === "" || item.name.includes(keyword)); if(searchType === "market") { let marketResult = await TisHub.getClosedIssuesTis({keyword}) marketResult = marketResult.map(hubTisInfo => { return { name: hubTisInfo.title, describe: hubTisInfo.describe, body: hubTisInfo.tisList.join('\n') || '', state: "installable" } }) const installedMap = resultTisList.reduce((map, item) => { map[item.name] = item; return map; }, {}); // 看本地是否已安装,如果已安装state就取已安装的项state (resultTisList = marketResult).forEach(hubTis =>{ if(installedMap[hubTis.name]) hubTis.state = installedMap[hubTis.name].state; }); } // 列表渲染 const resultElement = $(".tis-hub .result-list > .list-rol"); resultElement.html('') // 转状态名 function stateAsName(state) { return (state === "disable" && "移除(未启用)") || (state === "enable" && "移除") || "安装"; } for(let tis of resultTisList) { // tis 有该订阅的名 tis.name // tisMetaInfo 是tis.body 包含描述信息 tisMetaInfo.describe const tisMetaInfo = new PageTextHandleChains(tis.body).parseAllDesignatedSingTags("tis")[0]; // 自己的搜索逻辑 if(! `${tis.name}`.includes(keyword) && (tisMetaInfo != null && ! `${tisMetaInfo.describe}`.includes(keyword))) return; // 渲染到页面 resultElement.append(`
    ${tis.name} ${tisMetaInfo.describe || '订阅没有描述信息,请确认订阅安全或信任后再安装!'}
    `) } // 当点击tis-button按钮时 $(".hub-tis .tis-button").click(function() { // 使用 $(this) 获取当前被点击的元素 const button = $(this); const tisName = button.attr("tis-name"); let tis = installedList.find(item=>item.name === tisName); if(tis != null) { // 移除 installedList = installedList.filter(item => item.name !== tisName); tis.state = "installable"; }else { // 安装 const hubTis = resultTisList.find(item=>item.name === tisName); hubTis.state = "enable"; installedList.unshift(tis = hubTis); } // 更新状态 button.html(stateAsName(tis.state)); // 保存 console.log("保存:",installedList) cache.set(registry.searchData.USE_INSTALL_TISHUB_CACHE_KEY,installedList); // 清理缓存 clearCache() }); } // 点击搜索 const searchButton = tisSearchBtn.click(searchFun).click(); // 回车键触发搜索 tisSearchInput.on("keydown", function(event) { if (event.key === "Enter") { searchFun(); // 调用搜索功能 } }); // 单选框值改变时,搜索 const radioButtons = document.querySelectorAll('input[name="search-type"]'); radioButtons.forEach(radio => { radio.addEventListener('change', function() { if (this.checked) { searchButton.click(); } }); }); }) } })(unsafeWindow); // unsafeWindow是真实的window,作为参数传入