- // ==UserScript==
- // @name 我的搜索
- // @namespace http://tampermonkey.net/
- // @version 7.5.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
-
- // ==/UserScript==
-
- // 参数tis: actualWindow 是真实的window,而如果在下面脚本内访问window则不是,是沙盒的XPCNativeWrapper对象,详情https://blog.rxliuli.com/p/e55a67646bf546b3900ce270a6fbc6ca/
- (function(actualWindow) {
- 'use strict';
-
- // 模块一:快捷键触发某一事件 (属于触发策略组)
- // 模块二:搜索视图(显示与隐藏)(属于搜索视图组)
- // 模块三:触发策略组触发策略触发搜索视图组视图
- // 模块四:根据用户提供的策略(策略属于数据生成策略组)生成搜索项的数据库
- // 模块五:视图接入数据库
-
- // 判断当前是否在iframe里面,
- function currentIsIframe() {
- if (self.frameElement && self.frameElement.tagName == "IFRAME") return true;
- if (window.frames.length != parent.frames.length) return true;
- if (self != top) return true;
- return false;
- }
-
-
- // 如果当前是ifrae,结束脚本执行
- let MY_SEARCH_SCRIPT_VIEW_SHOW_EVENT = "MY_SEARCH_SCRIPT_VIEW_SHOW_EVENT";
- if(currentIsIframe()) {
- // 虽然iframe不能初始化脚本,但可以作为父窗口的事件触发源
- triggerAndEvent("ctrl+alt+s", function () { // 通知主容器显示搜索框
- window.parent.postMessage(MY_SEARCH_SCRIPT_VIEW_SHOW_EVENT, '*');
- })
- // 结束脚本执行
- return;
- }
-
- // css resource加载器
- function cssLoad(prefix,css = "",{isResourceName = false,replacePrefix}) {
- if(isResourceName) css = GM_getResourceText(css);
-
- if(replacePrefix && prefix) {
- css = css.toReplaceAll(replacePrefix,prefix);
- }else if(prefix) {
- // 对css原始内容引入前缀
- css = `.${prefix} {
- ${css}
- }`;
- }
- GM_addStyle(css);
- return prefix;
- }
- // 正则捕获
- function captureRegEx(regex, text) {
- let m;
- let result = []; // 一组一组 [[],[],...]
- regex.lastIndex = 0; // 重置lastIndex
- while ((m = regex.exec(text)) !== null) {
- let group = [];
- group.push(...m);
- if(group.length != 0) result.push(group);
- }
- return result;
- }
- // 滚动到目标文本 可以指定容器
- function scrollToText(text, index = 0, container = document.body) {
- // 创建高亮样式(只需执行一次)
- if (!document.getElementById('highlight-style')) {
- const style = document.createElement('style');
- style.id = 'highlight-style';
- style.textContent = `.highlight-text { border-bottom: 2px solid red !important;color:red; }`;
- document.head.appendChild(style);
- }
-
- // 处理容器参数
- const containerElement = typeof container === 'string'
- ? document.querySelector(container)
- : container;
-
- if (!containerElement) {
- console.error('Container not found:', container);
- return;
- }
-
- // 清理旧高亮
- Array.from(containerElement.getElementsByClassName('highlight-text'))
- .forEach(span => {
- const textNode = document.createTextNode(span.textContent);
- span.parentNode.replaceChild(textNode, span);
- });
-
- if (!text) return;
-
- // 获取所有文本节点
- const textNodes = [];
- const walker = document.createTreeWalker(
- containerElement,
- NodeFilter.SHOW_TEXT,
- { acceptNode: () => NodeFilter.FILTER_ACCEPT }
- );
- while (walker.nextNode()) textNodes.push(walker.currentNode);
-
- // 查找所有匹配项
- const matches = [];
- const searchLen = text.length;
- for (const node of textNodes) {
- const nodeText = node.nodeValue;
- let pos = 0;
- while (pos <= nodeText.length - searchLen) {
- const idx = nodeText.indexOf(text, pos);
- if (idx === -1) break;
- matches.push({ node, start: idx, end: idx + searchLen });
- pos = idx + 1; // 允许重叠匹配
- }
- }
-
- // 验证索引
- if (index < 0 || index >= matches.length) {
- console.error(`Index ${index} out of range (0-${matches.length - 1})`);
- return;
- }
-
- // 处理目标匹配项
- const { node, start, end } = matches[index];
- const middle = node.splitText(start);
- const after = middle.splitText(end - start);
- const span = document.createElement('span');
- span.className = 'highlight-text';
- span.textContent = middle.nodeValue;
- middle.parentNode.replaceChild(span, middle);
-
- // 滚动到目标位置
- span.scrollIntoView({
- behavior: 'smooth',
- block: 'center',
- inline: 'nearest'
- });
- }
- // 重写console.log方法
- let originalLog = console.log;
- console.logout = function() {
- const prefix = "[我的搜索log]>>> ";
- const args = [prefix].concat(Array.from(arguments));
- originalLog.apply(console, args);
- }
- // markdown转html 转换器 【1】
- // 更多配置项:https://github.com/showdownjs/showdown
- const converter = new showdown.Converter({
- // 将换行符解析为<br>
- simpleLineBreaks:true,
- // 新窗口打开链接
- openLinksInNewWindow: true,
- metadata:true,
- // 不允许下划线变斜体
- literalMidWordUnderscores: true,
- // 识别md表格
- tables: true,
- // www.baidu.com 会识别为链接
- simplifiedAutoLink: true
- });
-
- function md2html(rawText) {
- // 方案二: return marked.parse(rawText); // @require https://cdnjs.cloudflare.com/ajax/libs/marked/9.0.2/marked.min.js
- return converter.makeHtml( rawText )
- }
-
- // 提取URL根域名
- function getUrlRoot(url,isRemovePrefix = true,isRemoveSuffix = true) {
- if(! (typeof url == "string" || url.length >= 3)) return url;
- // 可处理
- // 判断是否有前缀
- let prefix = "";
- let root = "";
- let suffix = "";
- // 提取前缀
- if(url.indexOf("://") != -1) {
- // 存在前缀
- let prefixSplitArr = url.split("://")
- prefix = prefixSplitArr[0];
- url = prefixSplitArr[1];
- }
- // 提取root 和suffix
- if(url.indexOf("/") != -1) {
- let twoLevelIndex = url.indexOf("/")
- root = url.substr(0,twoLevelIndex);
- suffix = url.substr(twoLevelIndex,url.length-1);
- }else {
- root = url;
- suffix = "";
- }
- return ((!isRemovePrefix && prefix != "")?(prefix+"://"):"") + root + (isRemoveSuffix?"":suffix);
- }
- // 解析出http url 结构
- function parseUrl(url) {
- const regex = /(https?:|)\/\/([^\/]*|[^\/]*)(\/[^\s\?]*|)(\??[^\s]*|)/;
- const matches = regex.exec(url);
- if (matches) {
- const protocol = matches[1];
- const domain = matches[2];
- const path = matches[3];
- const params = matches[4];
- const rootUrl = protocol+"//"+domain
- const rawUrl = url;
- return {protocol,domain,path,params,rootUrl,rawUrl}
- }
- return null;
- }
- function isHttpUrl(url = "") {
- url = url.trim();
- return /^https?:\/\/.+/i.test(url)
- }
-
- // 检查网站是否可用
- function checkUsability(templateUrl,isStopCheck = false) {
- return new Promise(function (resolve, reject) {
- // 判断是否要检查
- if(isStopCheck) {
- reject(null);
- return;
- }
- var img=document.createElement("img");
- img.src = templateUrl.fillByObj(parseUrl("https://www.baidu.com"));
- img.style= "display:none;";
- img.onerror = function(e) {
- setTimeout(function() {img.remove();},20)
- reject(null);
- }
- img.onload = function(e) {
- setTimeout(function() {img.remove();},20)
- resolve(templateUrl);
- }
- document.body.appendChild(img);
- });
- }
-
-
-
- // 数据缓存器
- let cache = {
- prefix: "",
- get(key) {
- return GM_getValue(this.prefix+key);
- },
- set(key,value) {
- this.remove(this.prefix+key);
- GM_setValue(this.prefix+key,value);
- },
- jGet(key) {
- let value = GM_getValue(this.prefix+key);
- if( value == null) return value;
- return JSON.parse(value);
- },
- jSet(key,value) {
- value = JSON.stringify(value)
- GM_setValue(this.prefix+key,value);
- },
- remove(key) {
- GM_deleteValue(this.prefix+key);
- },
- cookieSet(cname,cvalue,exdays) {
- var d = new Date();
- d.setTime(d.getTime()+exdays);
- var expires = "expires="+d.toGMTString();
- document.cookie = cname + "=" + cvalue + "; " + expires;
- },
- cookieGet(cname) {
- var name = cname + "=";
- var ca = document.cookie.split(';');
- for(var i=0; i<ca.length; i++)
- {
- var c = ca[i].trim();
- if (c.indexOf(name)==0) return c.substring(name.length,c.length);
- }
- return "";
- }
- }
-
- // 责任链对象工厂
- function getResponsibilityChain() {
- return {
- chains: [],
- add(chain = {weight:0,fun: (data,ref)=>data}) {
- if(chain == null ) throw new Error("[ERROR]责任链对象: 你添加了一个null Chain!")
- if(chain.weight == undefined || chain.fun == undefined) throw new Error("[ERROR]责任链对象: 你传入的Chain是无效的!")
- this.chains.push(chain)
- },
- trigger(baton) {
- // 排序,通过weight从高到低
- this.chains = this.chains.sort((a, b)=>b.weight - a.weight);
- // 开始执行
- let _baton = baton;
- let ref = {
- stop: false
- }
- for(let chain of this.chains) {
- if( ref.stop) {
- break;
- }
- _baton = chain.fun(_baton,ref)
-
- }
- return _baton;
- }
- }
- }
- // 请求包装
- function request(type, url, { query, body,header = {},config ={}} = {}) {
- return new Promise(function(resolve, reject) {
- var formData = new FormData();
- var isFormData = false;
-
- if (body) {
- for (var key in body) {
- if (body[key] instanceof File) {
- formData.append(key, body[key]);
- isFormData = true;
- } else {
- formData.append(key, JSON.stringify(body[key]));
- }
- }
- }
-
- var ajaxOptions = {
- ...config,
- url: url + (query ? ("?" + $.param(query)) : ""),
- method: type,
- headers: header,
- success: function(response) {
- resolve(response);
- },
- error: function(jqXHR, textStatus, errorThrown) {
- reject(errorThrown);
- }
- };
-
- if (isFormData) {
- ajaxOptions.data = formData;
- ajaxOptions.processData = false;
- ajaxOptions.contentType = false;
- } else {
- ajaxOptions.data = JSON.stringify(body);
- }
- config?.crossDomain
- // ajax兼容
- ? GM_xmlhttpRequest({
- ...ajaxOptions,
- onload: (xhr)=>{
- ajaxOptions.success(xhr.responseText,xhr)
- },
- onerror: ajaxOptions.error,
- ontimeout: ajaxOptions.error
- })
- : $.ajax(ajaxOptions);
- });
- }
- // 正则字符串匹配目录字符串-匹配工具
- function isMatch(regexStr, text) {
- // 创建正则表达式对象
- let regex = new RegExp(regexStr);
- // 使用 test 方法测试字符串是否匹配
- return regex.test(text);
- }
- // `视图渲染完成`后调用
- function waitViewRenderingComplete(callback) {
- // 这里模拟的是当下次渲染完成后执行
- setTimeout(callback,30)
- }
- // 结构化的css 转 平铺的css
- function flattenCSS(cssObject, parentSelector = '') {
- let result = '';
-
- for (const [selector, rules] of Object.entries(cssObject)) {
- if (typeof rules === 'object') {
- // 如果 rules 是对象,说明有嵌套,需要递归处理
- const fullSelector = parentSelector ? `${parentSelector} ${selector}` : selector;
- result += flattenCSS(rules, fullSelector);
- } else {
- // 如果 rules 不是对象,说明是样式规则
- const fullSelector = parentSelector ? `${parentSelector} { ${selector}: ${rules}; }\n` : '';
- result += fullSelector;
- }
- }
-
- return result;
- }
- // ref引用变量
- function ref(initValue = null) {
- return {value: initValue}
- }
- // ==偏业务工具函数==
- // 使用责任链模式——对pageText进行操作的工具
- class PageTextHandleChains {
- pageText = "";
- constructor(pageText = "") {
- this.pageText = pageText;
- }
- setPageText(newPageText) {
- this.pageText = newPageText;
- }
- getPageText() {
- return this.pageText;
- }
- // 解析双标签-获取指定标签下指定属性下的值
- parseDoubleTab(tabName,attrName) {
- // 返回指定标签下指定属性下的值
- const regex = RegExp(`<\\s*${tabName}[^<>]*\\s*${attrName}="([^<>]*)"\\s*>([\\s\\S]*?)<\/\\s*${tabName}\\s*>`,"gm");
- let m;
- let tabNameArr = [];
- let copyPageText = this.pageText;
- // 注意下面的 copyPageText 不能改变
- while ((m = regex.exec(copyPageText)) !== null) {
- // 这对于避免零宽度匹配的无限循环是必要的
- if (m.index === regex.lastIndex) {
- regex.lastIndex++;
- }
- tabNameArr.push({
- attrValue: m[1],
- tabValue: m[2]
- })
- const newPageText =this.pageText.replace(m[0], "");
- this.pageText = newPageText;
- }
- return tabNameArr;
- }
- // 解析双标签-只获取值
- parseDoubleTabValue(tabName) {
- // 返回指定标签下指定属性下的值
- const regex = RegExp(`<\\s*${tabName}[^<>]*\\s*>([\\s\\S]*?)<\/\\s*${tabName}\\s*>`,"gm");
- let m;
- let tabNameArr = [];
- let copyPageText = this.pageText;
- while ((m = regex.exec(copyPageText)) !== null) {
- // 这对于避免零宽度匹配的无限循环是必要的
- if (m.index === regex.lastIndex) {
- regex.lastIndex++;
- }
- tabNameArr.push({
- tabValue: m[1]
- })
- const newPageText =this.pageText.replace(m[0], "");
- this.pageText = newPageText;
- }
-
- return tabNameArr;
- }
-
- // 解析所有指定的单标签
- parseAllDesignatedSingTags(parseTabName) {
- // 匹配标签的正则表达式
- const regex = /<(\w+)::([\S]+)(.*?)\/>/g;
- // 匹配属性键值对的正则表达式,支持连字符
- const attributesRegex = /([\w-]+)="(.*?)"/g;
- let matches;
- const result = [];
- let modifiedString = this.pageText;
-
- while ((matches = regex.exec(this.pageText)) !== null) {
- const tabName = matches[1];
- const tabValue = matches[2];
- const attributesString = matches[3];
-
- console.log(tabName, parseTabName);
- if (tabName !== parseTabName) continue;
-
- const attributes = {};
- let attrMatch;
- while ((attrMatch = attributesRegex.exec(attributesString)) !== null) {
- attributes[attrMatch[1]] = attrMatch[2];
- }
-
- result.push({
- tabName,
- tabValue,
- ...attributes
- });
-
- // 将匹配到的内容替换为空字符串
- modifiedString = modifiedString.replace(matches[0], '');
- }
-
- // 更新 pageText
- this.pageText = modifiedString;
- return result;
- }
- // 根据单标签的元信息进行stringify
- rebuildTags(tagMetaArr = []) {
- return tagMetaArr.map(tag => {
- const { tabName, tabValue, ...attributes } = tag;
- const attributesString = Object.entries(attributes)
- .map(([key, value]) => `${key}="${value}"`)
- .join(' ');
- return `<${tabName}::${tabValue} ${attributesString} />`;
- }).join('\n');
- }
- }
-
- // 根据反馈的错误项调整templates位置,使得错误的靠后
- function feedbackError(saveKey,currentErrorItem) {
- let items = cache.get(saveKey)??[];
- let foundIndex = -1; // -1-查找模式 , n-已找到 n是所在位置模式
- let foundValue = null;
- for(let i = 0; i < items.length; i++) {
- let item = items[i];
- if(foundIndex == -1 ) {
- if(item == currentErrorItem) {
- foundIndex = i;
- foundValue = items[i];
- }
- }else {
- items[i-1] = items[i];
- // 查看是否是最后一个
- if( i == items.length - 1 ) items[i] = foundValue;
- }
- }
- cache.set(saveKey,items);
- return items;
- }
- // 数据项选择记录器
- class SelectHistoryRecorder {
- static HISTORY_CACHE_KEY = "HISTORY_CACHE_KEY";
- static defaultIdFun = (item)=> JSON.stringify(item);
- static select(item,idFun = SelectHistoryRecorder.defaultIdFun) {
- // 记录到历史
- let key = idFun(item);
- let history = cache.get( SelectHistoryRecorder.HISTORY_CACHE_KEY )??[];
- history = history.filter(_item=>idFun(_item) != key) // 将原来的去除
- history.unshift({...item}) // 必须拷贝
- // 清理掉索引,索引只是本次数据加载有效的,而我们存储的历史数据是不随数据加载而改变的,也就是如果缓存索引会失效,没有索引它会自己找,当然我们会提供我们这里的数据给它找,如果在全局数据中匹配不到的话
- history.forEach(_item=>{
- delete _item.index;
- return _item;
- })
- // 缓存历史数据
- cache.set(SelectHistoryRecorder.HISTORY_CACHE_KEY,history);
- }
- static history(count) {
- let history = cache.get( SelectHistoryRecorder.HISTORY_CACHE_KEY )??[];
- if(count == null) return history; // 如果没有传入count,那就是全部
- let result = [];
- for(let i = 0; i < count && i+1 <= history.length; i++) result.push( history[i] ); // 将history前count个放在result数组中
- return result;
- }
-
- }
- // 加分、“加分(取分)”
- class DataWeightScorer {
- static ITEM_WEIGHT_CACHE_KEY = "ITEM_WEIGHT_CACHE_KEY";
- static defaultIdFun = (item)=> JSON.stringify(item);
- static SCORE_RECORD_ATTR_KEY = "weight";
- static select(item,idFun = DataWeightScorer.defaultIdFun) {
- let ItemWeightData = cache.get( DataWeightScorer.ITEM_WEIGHT_CACHE_KEY )??{};
- let key = idFun(item);
- ItemWeightData[key] = (ItemWeightData[key]??0) + 1
- cache.set(DataWeightScorer.ITEM_WEIGHT_CACHE_KEY,ItemWeightData)
- }
- static assign(items=[],idFun = DataWeightScorer.defaultIdFun) {
- let ItemWeightData = cache.get( DataWeightScorer.ITEM_WEIGHT_CACHE_KEY)??{};
- items.forEach(item=>{
- let key = idFun(item);
- item[DataWeightScorer.SCORE_RECORD_ATTR_KEY] = ItemWeightData[key]??0;
- })
- return items;
- }
- static sort(items=[],idFun = DataWeightScorer.defaultIdFun) {
- // 将权重赋于
- DataWeightScorer.assign(items,idFun);
- // 根据权重排序(高->低)
- return items.sort((a, b) => b[DataWeightScorer.SCORE_RECORD_ATTR_KEY] - a[DataWeightScorer.SCORE_RECORD_ATTR_KEY]);
- }
- // 获取高频前count项
- static highFrequency(count) {
- let ItemWeightData = cache.get( DataWeightScorer.ITEM_WEIGHT_CACHE_KEY )??{};
- let orderKeys = Object.keys(ItemWeightData).sort((a, b) => ItemWeightData[b] - ItemWeightData[a]);
- if(count != null) orderKeys = orderKeys.slice(0, count);
- // keys转items
- return registry.searchData.matchItemsByKeys(orderKeys);
- }
- }
- // 将多个
- function parseTis(bodyText) {
- // 提取整个tis标签的正则
- const regex = /(<\s*tis::http[^<>]+\/\s*>)/gm;
- let raw = captureRegEx(regex,bodyText);
- if(raw != null) {
- return raw.map(item=>item[1])
- }
- return null;
- }
- let USER_GITHUB_TOKEN_CACHE_KEY = "USER_GITHUB_TOKEN_CACHE_KEY";
- let GithubAPI = {
- token: cache.get(USER_GITHUB_TOKEN_CACHE_KEY),
- defaultToken: '',
- setToken(token) {
- if(token != null) this.token = token;
- if(this.token == null) {
- token = prompt("请输入您的github Token (只缓存在你的本地):")
- // 获取的内容无效
- if(token == null || token == "") return this;
- // 内容有效-设置
- cache.set(USER_GITHUB_TOKEN_CACHE_KEY,this.token = token);
- }
- return this;
- },
- clearToken() {
- cache.remove(USER_GITHUB_TOKEN_CACHE_KEY)
- this.token = null;
- },
- getToken(isRequest = false) {
- if(this.token == null && isRequest) this.setToken();
- return this.token;
- },
- baseRequest(type,url,{query,body}={},header = {}) {
- query = {...query}
- return request(type, url, { query,body,header });
- },
- getUserInfo() {
- return this.baseRequest("GET","https://api.github.com/user")
- },
- commitIssues(body) {
- const header = {Authorization:`Bearer ${this.token}`};
- return this.baseRequest("POST","https://api.github.com/repos/My-Search/TisHub/issues",{ body, header })
- },
- // get issues不要加 Authorization 头,可能会出现401
- getTisForIssues({keyword,state} = {}) {
- let query = null;
- if(state != null) query = {state};
- let token = this.token;
- if(token == null) token = this.defaultToken;
- return keyword
- ? new Promise((resolve,reject)=>{
- // API兼容处理
- this.baseRequest("GET",`https://api.github.com/search/issues?q=repo:My-Search/TisHub+state:${state}+in:title+${keyword}`)
- .then(response=>resolve(response.items)).catch(error=>resolve([]));
- })
- : this.baseRequest("GET","https://api.github.com/repos/My-Search/TisHub/issues",{query})
- }
- }
-
- // 从订阅标签中提取订阅链接
- let TisHub = {
- // 将第一个tis集与第二个tis集合并
- tisFilter(source,filterList) {
- if(typeof source == "string") source = parseTis(source);
- if(typeof filterList == "string") filterList = parseTis(filterList);
- for(let filterItem of filterList) {
- let pageTextHandler = new PageTextHandleChains(filterItem);
- let tabMetaInfos = pageTextHandler.parseAllDesignatedSingTags("tis");
- let subscribedLink = null;
- // 一个filterItem解析出元信息返回tabMetaInfos只能有一个元素,如果是多个,只取一个
- if(tabMetaInfos != null && tabMetaInfos.length > 0 ) subscribedLink = tabMetaInfos[0].tabValue;
- // 如果取不出来,说明tis无效,断言不需要解析filterItem就是subscribedLink
- if(subscribedLink == null) subscribedLink = filterItem;
- // subscribedLink 这里是filterItem用于filter source的元素实体,下面开始过滤
- source = source.filter(resultSubscribed=>! resultSubscribed.includes(subscribedLink));
- }
- return source;
- },
- getTisHubAllTis(filterList = []) {
- return new Promise((resolve,reject)=>{
- let openIssuesTisPromise = this.getOpenIssuesTis();
- let result = [];
- return Promise.all([ this.getOpenIssuesTis(), this.getClosedIssuesTis() ]).then(values=>{
- for(let value of values) {
- if(value == null ) continue;
- for(let tisListObj of value) {
- if(tisListObj != null ) result.push(...tisListObj.tisList)
- }
- }
- // 过滤并提交结果
- resolve(this.tisFilter(result,filterList));
- })
- })
- },
- // {keyword,state} .其中state {open, closed, all}
- getTisForIssues(params = {}) {
- return new Promise((resolve,reject)=>{
- GithubAPI.getTisForIssues(params).then(response=>{
- if(response != null && Array.isArray(response)) {
- resolve(response.map(obj=>{return {
- owner: obj.user.login,
- ownerProfile: obj.user.html_url,
- title: obj.title,
- tisList: parseTis(obj.body),
- status: obj.state
- }}))
- }
- }).catch(error=>resolve([]));
- })
- },
- getOpenIssuesTis(params = {}) {
- return this.getTisForIssues({state: "open",...params});
- },
- getClosedIssuesTis(params = {}) {
- return this.getTisForIssues({state: "closed",...params});
- },
- tisListToTisText(tisList) {
- let text = "";
- for(let tis of tisList) text += tis.tisList;
- return text;
- }
- }
-
- // 全局注册表
- let ERROR = {
- tell(info) {
- console.error("ERROR " + info)
- }
- }
- let registry = {
- view: {
- viewVisibilityController: () => { ERROR.tell("视图未初始化,但你使用了它的未初始化的注册表信息!") },
- viewDocument: null, // 视图挂载后有值
- element: null, // 存放着视图的关键元素对象 视图挂载后有值
- tis: {
- beginTis(msg) {
- if(msg == null || msg.length === 0) return;
- const tisDocument = document.querySelector("#my_search_box > #tis");
- tisDocument.innerHTML = msg;
- tisDocument.display = "block";
- console.log("设置结束")
- return ()=>{
- tisDocument.innerHTML = ""; // 置空消息内容
- tisDocument.display = "none"; // 让tis不可见
- }
- }
- },
- setButtonVisibility: () => { ERROR.tell("按钮未初始化!") },
- titleTagHandler: {
- handlers: [],
- // 标题tag处理器
- execute: function (title) {
- // 去掉标题内容只剩下tags
- let arr = captureRegEx(/(\[.*\])/gm,title)
- if(arr == null || arr[0] == null || arr[0][0] == null) return "";
- let tagStr = arr[0][0];
- for(let titleTagHandler of this.handlers) {
- let result = titleTagHandler(tagStr.trim());
- if(result != -1) return result;
- }
- return tagStr;
- }
- },
- viewFirstShowEventListener: [],
- viewHideEventAfterListener: [],
- // 在查看详情内容时,返回后事件
- itemDetailBackAfterEventListener: [
- // 简讯内容隐藏-确定存在触发的事件 - 脚本环境变量置空
- () => registry.script.clearMSSE()
- ],
- // 视图延时隐藏时间,避免点击右边logo,还没显示就隐藏了
- delayedHideTime: 100,
- initialized: false,
- textView: {
- cssFillPrefix(css = "", prefix = "") {
- const cssBlocks = css.split('}');
- let outputCSS = '';
- for (let block of cssBlocks) {
- let blockLines = block.split('\n');
- let blockOutput = '';
-
- for (let line of blockLines) {
- // 判断行末是否以 `{` 或 `,` 结尾且行首不能有空格
- if ((line.trim().endsWith('{') || line.trim().endsWith(','))
- && !line.startsWith(' ') && !line.trim().startsWith('@')) {
- blockOutput += `${prefix} ${line.trim()}`; // 在当前行前加上前缀
- } else {
- blockOutput += line; // 其他行保持原样
- }
- blockOutput += '\n'; // 添加换行符,用于分隔CSS内容的各行
- }
-
- if (blockOutput.trim() !== '') {
- outputCSS += blockOutput;
- outputCSS += '}\n'; // 只有在当前块不为空时添加闭合大括号
- }
- }
- return outputCSS;
- },
- show(html,css = "",js = "") {
- const MS_BODY_ID = "ms-page-body";
- html = `<style type='text/css'>${this.cssFillPrefix(css,`#${registry.view.viewDocument.id} #${registry.view.element.textView.attr('id')} #${MS_BODY_ID}` )}</style>`
- + `<div id="${MS_BODY_ID}">${html}</div>`
- // 这里在函数内执行js是为了在同一页面未刷新下可多次执行该js不执行,也就是对变量/函数等进行隔离
- +`<script>(()=>{ ${js} })()</script>`
- let my_search_box = $(registry.view.viewDocument);
- // 视图还没有初始化
- if(my_search_box == null) return;
- let matchResult = registry.view.element.matchResult;
- let textView = registry.view.element.textView
- textView.html(html);
- /*使用code代码块样式*/
- document.querySelectorAll('#text_show pre code').forEach((el) => {
- // 这里没有错,发警告不用理
- hljs.highlightElement(el);
- });
- matchResult.css({
- "display": "none"
- })
- textView.css({
- "display":"block"
- })
- waitViewRenderingComplete(() => {
- const currentKey = registry.searchData.searchHistory.currentKeyword();
- if(currentKey.trim().length > 3) scrollToText(currentKey,0,"#ms-page-body");
- });
- }
- },
- // 搜索框logo控制
- logo: {
- // logo src值
- originalLogoImgSrc: null,
- // logo按钮是否按下状态
- isLogoButtonPressedRef: ref(false),
- getLogoImg: function () {
- let viewDocument = registry.view.viewDocument;
- if(viewDocument == null ) return null;
- let currentLogoImg = registry.view.element.logoButton.find("img");
- if(this.originalLogoImgSrc == null) this.originalLogoImgSrc = currentLogoImg.attr("src");
- return currentLogoImg;
- },
- change: function (imgResource) {
- let logoImg = this.getLogoImg();
- if(imgResource == null || logoImg == null ) return;
- logoImg.attr("src",imgResource)
- },
- reset: function() {
- let logoImg = this.getLogoImg();
- if (logoImg == null ) return;
- logoImg.attr("src",this.originalLogoImgSrc)
- }
- },
- modeEnum: {
- UN_INIT: -2, // 未初始化
- HIDE: -1, // 隐藏
- WAIT_SEARCH: 0, // 待搜索模式
- SHOW_RESULT: 1, // 结果显示模式
- SHOW_ITEM_DETAIL: 2 // 查看项详情 (简述内容查看/脚本页)
- },
- seeNowMode() {
- if(this.viewDocument == null) return this.modeEnum.UN_INIT;
- if(this.element.textView.css('display')!== "none") return this.modeEnum.SHOW_ITEM_DETAIL;
- if(this.element.matchResult.css('display') !== "none") return this.modeEnum.SHOW_RESULT;
- return this.viewDocument.style.display !== "none" ? this.modeEnum.WAIT_SEARCH : this.modeEnum.HIDE;
- }
- },
- other: {
- UPDATE_CDNS_CACHE_KEY: "UPDATE_CDNS_CACHE_KEY"
- },
- searchData: {
- // 处理的历史
- processHistory: [],
- // 用于数据显示后,数据又更新了
- version: 0,
- // 全局搜索数据
- data: [],
- // 数据更新后有效时长
- effectiveDuration: 1000*60*60*12,
- // 数据设置到全局中的时间
- dataMountTime: null,
- clearData() {
- this.data = [];
- this.dataMountTime = null;
- },
- setData(data) {
- if(data == null || data.length == 0) return;
- this.data = data;
- this.dataMountTime = Date.now();
- },
- getData() {
- let dataPackage = cache.get(registry.searchData.SEARCH_DATA_KEY);
- if(dataPackage == null || dataPackage.data == null) return this.data;
- // 缓存信息不为空,深入判断是否使用缓存的数据
- let updateDataTime = dataPackage.expire - this.effectiveDuration;
- // 如果数据在挂载后面已经更新了,重新加载数据到全局中
- // 全局data (即this.data)与dataMountTime是同时设置的,两者理论上是必须同时有值的
- if(this.data == null || updateDataTime > this.dataMountTime) {
- console.logout("== 数据未加载或已检查到在其它页面已重新更新数据 ==")
- this.setData(dataPackage.data);
- }
- return this.data;
- },
- // 根据keys(由idFun决定)从data中匹配items
- matchItemsByKeys: function (keys = []) {
- let that = this;
- if(keys.length == 0) return [];
- // 有keys转items
- let items = keys.map(key=>{
- for(let item of that.data) {
- if(that.idFun(item) == key) return item;
- }
- return null;
- })
- // 对数组keys去空,注意此时keys已经是items了
- return items.filter(item => item != null);
- },
- specialKeyword: { // 特殊的keyword
- new: "<new>",
- history: "<history>",
- highFrequency: "<highFrequency>"
- },
- // 决定着数据是否要再次初始化
- isDataInitialized: false,
- // 从可自定义搜索数据中根据title与desc进行数据匹配
- findSearchDataItem: function (title = "",desc = "",matchData) {
- if(matchData == null ) matchData = this.data;
- for(let item of matchData) {
- if( (item.title.includes(title) || title.includes(item.title) )
- && ( item.desc.includes(desc) || desc.includes(item.desc) )) return item;
- }
- return null;
- },
- // 数组差异-获取不同的元素比较的基值
- idFun(item) { // 自定义比较
- if(item == null || !( item instanceof Object && item.title != null)) return null;
- return item.title.replace(/\[.*\]/,"").trim()+(""+item.desc).trim();
- },
- // 旧的新数据缓存KEY
- OLD_SEARCH_DATA_KEY: "OLD_SEARCH_DATAS_KEY",
- // 标签数据缓存KEY
- DATA_ITEM_TAGS_CACHE_KEY: "DATA_ITEM_TAGS_CACHE_KEY",
- // 用户维护的不关注标签列表,缓存KEY
- USER_UNFOLLOW_LIST_CACHE_KEY: "USER_UNFOLLOW_LIST_CACHE_KEY",
- // 用户安装tishub订阅的缓存
- USE_INSTALL_TISHUB_CACHE_KEY: "USE_INSTALL_TISHUB_CACHE_KEY",
- // 默认用户不关注标签
- USER_DEFAULT_UNFOLLOW: ["成人内容","Adults only"],
- // 已经清理了用户不关注的与隐藏的标签,这是用户应真正搜索的数据
- CLEANED_SEARCH_DATA_CACHE_KEY: "CLEANED_SEARCH_DATA_CACHE_KEY",
- subscribeKey: "subscribeKey",
- showSize: 15,
- isSearchAll: false,
- searchEven: {
- event:{},
- // 搜索状态,失去焦点隐藏的一要素
- isSearching:false,
- async send(search,rawKeyword) {
- try {
- // 标记为搜索进行中
- this.isSearching = true;
- for(let subscriptionRegular of Object.keys(this.event)) {
- const regex = new RegExp(subscriptionRegular,"i"); // 将正则字符串转换为正则表达式对象
- if(regex.test(rawKeyword) && typeof this.event[subscriptionRegular] == "function" ) {
- return this.event[subscriptionRegular](search,rawKeyword);
- }
- }
- return await search(rawKeyword);
- }finally {
- // 标记为搜索结束
- this.isSearching = false;
- }
- }
- },
- // 新数据设置的过期天数
- NEW_DATA_EXPIRE_DAY_NUM:7,
- // 搜索逻辑,可用来手动触发搜索
- triggerSearchHandle: function (keyword){
- // 获取input元素
- const inputEl = registry.view.element?.input;
- // 如果视图还没有初始化,触发搜索无效
- if(! inputEl) return;
- if(keyword == null) {
- keyword = inputEl.val() ?? ""
- }else {
- inputEl.val(keyword);
- }
- // 手动触发input事件
- inputEl[0].dispatchEvent(new Event("input", { bubbles: true }));
- // 维护全局搜索keyword
- this.keyword = keyword;
- },
- // 数据改变事件
- dataChangeEventListener: [],
- // 缓存被删除事件
- dataCacheRemoveEventListener:[],
- onSearch: [],
- // 新数据块处理完成事件
- // 更新搜索数据的责任链
- USDRC: getResponsibilityChain(),
- onNewDataBlockHandleAfter: [],
- // 新数据的tag
- NEW_ITEMS_TAG: "[新]",
- // 搜索的keyword
- keyword: "",
- // 搜索的附加文件
- searchForFile: {
- files: [],
- refreshFileListView() {
- registry.view.element.files.html('')
- for(let [index,file] of this.files.entries()) {
- const reader = new FileReader();
- reader.onload = function (e) {
- // 创建 img 元素并设置 src 为文件的 Base64 数据
- registry.view.element.files.append(`
- <div class="ms-input-file" key="${index}">
- <img src="${e.target.result}" />
- </div>
- `);
- };
- reader.readAsDataURL(file);
- }
- },
- push(file) {
- this.files.push(file);
- this.refreshFileListView();
- },
- delete() {
- this.files.pop();
- this.refreshFileListView();
- },
- clear() {
- const trashCan = this.files;
- this.files = [];
- this.refreshFileListView();
- return trashCan;
- },
- start() {
- registry.view.element.input.on('paste', event => {
- // 希望input外层元素无法感知到paste的动作
- event.stopPropagation();
- // 获取粘贴事件对象中的剪贴板数据
- const clipboardData = event.originalEvent.clipboardData || window.clipboardData;
- if (clipboardData) {
- const items = clipboardData.items;
- for (let i = 0; i < items.length; i++) {
- const file = items[i].getAsFile();
- if (! file.type.startsWith('text/')) this.push(file);
- }
- }
- });
- registry.view.viewHideEventAfterListener.push(() => this.clear());
- }
- },
- // 持久化Key
- SEARCH_DATA_KEY: "SEARCH_DATA_KEY",
- SEARCH_NEW_ITEMS_KEY:"SEARCH_NEW_ITEMS_KEY",
- // 搜索搜索出来的数据
- searchData: [],
- pos: 0,
- clearUrlSearchTemplate(url) {
- return url.replace(/\[\[[^\[\]]*\]\]/gm,"");
- },
- faviconSources: [ // favicon来源:https://api.cxr.cool/
- // "https://favicon.yandex.net/favicon/${domain}", 淘汰原因:当获取不到favicon时不报错,而是显示空白图标
- // "https://api.cxr.cool/ico/?url=${domain}", 淘汰原因:慢
- // "https://api.vvhan.com/api/ico?url=${domain}", 淘汰原因:快,但存在很多网站的图标无法获取
- // "https://statistical-apricot-seahorse.faviconkit.com/${domain}/32", 淘汰原因:有些图标无法获取,但会根据网站生成其网站单字符图标
- // "https://favicons.fuzqing.workers.dev/api/getFavicon?url=${rootUrl}", 淘汰原因:快,但存在很多网站的图标无法获取
- // "https://tools.ly522.com/ico/favicon.php?url=${rootUrl}", 淘汰原因:废的,10个8个获取不到
- "https://api.iowen.cn/favicon/${domain}.png", // 神
- "https://api.xinac.net/icon/?url=${rootUrl}", // 可选:但有点慢
- // "https://favicon.qqsuu.cn/${rootUrl}", 淘汰原因:外国站点不行
- // "https://api.uomg.com/api/get.favicon?url=${rootUrl}", 淘汰原因:外国站点不行
- // "https://api.vvhan.com/api/ico?url=${domain}", // 淘汰原因:存在很多网站的图标无法获取
- // "https://api.15777.cn/get.php?url=${rootUrl}",// 淘汰原因:存在很多网站的图标无法获取
- "https://ico.txmulu.com/${domain}", // 可选,但有些站点不行
- // "https://api.cxr.cool/ico/?url=${domain}", // 很多网站获取不到,国外站点不行
- "${rootUrl}/favicon.ico", // 永久有效的兜底
- ],
- CACHE_FAVICON_SOURCE_KEY: "CACHE_FAVICON_SOURCE_KEY", // 可见远程源
- CACHE_FAVICON_SOURCE_TIMEOUT: 1000*60*60*4, // 4个小时重新检测一下favicon源/过期时间,只会在呼出搜索框后检查
- getFaviconAPI: (function(){
- let defaultFaviconUrlTemplate = "${rootUrl}/favicon.ico";
- let faviconUrlTemplate = defaultFaviconUrlTemplate;
- let isRemoteTemplate = false;
- // 查看是否已经检查模板
- function checkTemplateAndUpdateTemplate() {
- let faviconSourceCache = cache.get(registry.searchData.CACHE_FAVICON_SOURCE_KEY);
- if( !isRemoteTemplate && faviconSourceCache != null && faviconSourceCache.sourceTemplate != null ) {
- faviconUrlTemplate = faviconSourceCache.sourceTemplate;
- // 设置已经是远程Favicon模板
- isRemoteTemplate = true;
- }
- }
- return function(url,isStandby = false) {
- checkTemplateAndUpdateTemplate();
- let useFaviconUrlTemplate = faviconUrlTemplate;
- // 如果是要获取备用favicon,那直接使用上面定义的faviconUrlTemplate
- if(isStandby) useFaviconUrlTemplate = defaultFaviconUrlTemplate;
- // 去掉资源的“可搜索”模板,才是真正的URL
- url = registry.searchData.clearUrlSearchTemplate(url);
- // 将资源url放到获取favicon的源模板中
- let urlObj = parseUrl(url)
- return useFaviconUrlTemplate.fillByObj(urlObj);
- }
- })(),
- tmpVar: null, // 用于防抖
- searchPlaceholder(target = "SELECT",placeholder,duration = 1200) {
- // 全部的输入提示
- let inputDescs = ["我的搜索"];
- // 当前应用“输入提示”
- let inputDesc = inputDescs[Math.floor(Math.random()*inputDescs.length)];
- if(target == "UPDATE") {
- const inputEl = registry.view.element?.input;
- // 如果视图还没挂载,无需显示
- if(! inputEl) return;
- if(this.tmpVar != null) {
- clearTimeout(this.tmpVar);
- }
- this.tmpVar = setTimeout(()=>{
- inputEl.attr("placeholder",this.searchPlaceholder());
- },duration)
- let updateResult = placeholder==null?`🔁 数据库更新到 ${this.data==null?0:this.data.length}条`:placeholder;
- inputEl.attr("placeholder",updateResult);
- return updateResult;
- }
- return inputDesc;
-
- },
- // 存储着text转pinyin的历史 registry.searchData.subSearch.isSubSearchMode
- TEXT_PINYIN_KEY: "TEXT_PINYIN_MAP",
- // 默认数据不应初始化,不然太占内存了,只用调用了toPinyin才会初始化 getGlobalTextPinyinMap()
- getGlobalTextPinyinMap: (function() {
- let textPinyinMap = null;
- return function (){
- if(textPinyinMap != null) return textPinyinMap;
- return (textPinyinMap = cache.jGet("TEXT_PINYIN_MAP")??{});
- }
- })(),
- subSearch: {
- searchBoundary: " : ",
- isEnteredSubSearchMode: false, // 是否已经进入子搜索模式
- // 不传参数是看当前是否为子搜索模式 , [0] 是最近一个
- isSubSearchMode(by = undefined) {
- let byKeyword = typeof(by) === 'string'
- ? by // by就是keyword
- : (by === undefined ? registry.searchData.searchHistory.currentKeyword() : registry.searchData.searchHistory.history[by]); // by是index
- return byKeyword && byKeyword.includes(this.searchBoundary);
- },
- // 获取父级(根keyword)
- getParentKeyword(keyword) {
- // 如果没有传入使用搜索框的value
- if(! keyword) keyword = registry.searchData.searchHistory.currentKeyword();
- return (keyword || "").split(this.searchBoundary)[0].trim()
- },
- // 获取子搜索keyword
- getSubSearchKeyword(keyword) {
- // 如果没有传入使用搜索框的value
- if(! keyword) keyword = registry.searchData.searchHistory.currentKeyword();
- let _arr = (keyword || "").split(this.searchBoundary);
- if( _arr.length < 2 ) return undefined;
- return _arr[1].trim();
- }
-
- },
- searchHistory: {
- history: [], // 新,旧...
- add(keyword) {
- if(! keyword) return;
- // 维护isEnteredSubSearchMode变量状态(进入与退出)
- const searchBoundary = registry.searchData.subSearch.searchBoundary
- if(keyword !== searchBoundary && keyword.endsWith(searchBoundary)) {
- registry.searchData.subSearch.isEnteredSubSearchMode = true;
- console.logout("进入了子搜索")
- }else if(registry.searchData.subSearch.isEnteredSubSearchMode && !keyword.includes(searchBoundary)){
- registry.searchData.subSearch.isEnteredSubSearchMode = false;
- console.logout("退出了子搜索")
- }
- // 加入到历史
- const _history = this.history;
- _history.unshift(keyword);
- _history.slice(10); // 不能超过10个元素
- },
- currentKeyword() {
- return registry.view.element.input.val()
- },
- // 当前keyword "123 : 哈哈" 与 最近"123 : 嘻嘻" 则返回true,即看左边
- seeCurrentEqualsLastByRealKeyword() {
- // 上一次真实搜索keyword === 当前真实搜索keyword
- return registry.searchData.subSearch.getParentKeyword(this.history[0]) === registry.searchData.subSearch.getParentKeyword(this.currentKeyword());
- }
- },
- searchProTag: "[可搜索]",
- links: {
- stringifyForSearch(links) {
- if(links == null) return ''
- links = links.filter(link => link != null);
- return links.reduce((acc, cur) => acc + `${cur.text}${cur.url}${cur.title}`, '');
- }
- }
- },
- script: {
- // MSSE默认值/模板
- MS_SCRIPT_ENV_TEMPLATE: {
- event: {
- // 发送sub keyword事件
- sendListener: [] // 脚本页监听IPush事件
- },
- // 让脚本页可使用cache,为避免冲突,加前缀于区别
- cache: {...cache, prefix: "MS_SCRIPT_CACHE:"},
- // 让脚本页面获取脚本项数据
- getSearchDB() {
- return [...registry.searchData.getData()]
- },
- // 让脚本页面获取选择的文本
- getSelectedText,
- // 挂载markdown
- md2html,
- // 挂载http request对象
- request
- },
- // 当值为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 = "<Space>"
- let spaceChar = " ";
- text = text.toReplaceAll(spaceChar,space)
- let pinyinArr = pinyin(text,options);
- // 保存到全局字典对象 ( 会话级别 )
- textPinyinMap[this] = pinyinArr.join("").toReplaceAll(space,spaceChar).toUpperCase();
- return textPinyinMap[this];
- }
- // 加载全局样式
- loadStyleString(`
- /*搜索视图样式*/
- #searchBox {
- height: 45px;
- background: #ffffff;
- padding: 0 10px;
- box-sizing: border-box;
- z-index: 10001;
- position: relative;
- display: flex;
- align-items: center;
- flex-wrap: nowrap;
- }
- #searchBox #ms-input-files {
- display: flex;
- flex-wrap: nowrap;
- align-items: center;
- height: 100%;
- }
- #ms-input-files .ms-input-file {
- height: 70%;
- display: flex;
- align-items: center;
- margin-right: 2px;
- }
- #ms-input-files .ms-input-file img {
- height: 100%;
- box-sizing: border-box;
- padding: 3px;
- background: #d3d3d3;
- }
-
- #my_search_input {
- text-align: left;
- width: 100%;
- height: 100%;
- border: none;
- outline: none;
- font-size: 15px;
- background: #fff;
- padding: 0px;
- box-sizing: border-box;
- color: rgba(0, 0, 0, .87);
- font-weight: 400;
- margin: 0px;
- }
-
- #matchResult {
- display: none;
- }
-
- #matchResult > ol {
- margin: 0px;
- padding: 0px 15px 5px;
- }
-
- #text_show {
- display: none;
- width: 100%;
- box-sizing: border-box;
- padding: 5px 10px 7px;
- font-size: 15px;
- line-height: 25px;
- max-height: 450px;
- overflow: auto;
- text-align: left;
- color: #000000;
- user-select: text !important; /* 允许用户选中复制 */
- }
- #text_show img {
- width: 100%;
- }
- #text_show .copy-btn {
- position: absolute;
- top: 8px;
- right: 8px;
- background: #7f7f7fa1;
- color: white;
- border: none;
- padding: 3px 10px;
- font-size: 12px;
- cursor: pointer;
- border-radius: 3px;
- opacity: 0.8;
- transition: opacity 0.3s;
- }
-
- #text_show .copy-btn:hover {
- opacity: 1;
- }
-
- /*定义字体*/
- @font-face {
- font-family: 'HarmonyOS';
- src: url('https://s1.hdslb.com/bfs/static/jinkela/long/font/HarmonyOS_Medium.a1.woff2');
- }
- #my_search_view {
- font-family: 'HarmonyOS', sans-serif !important;
- }
- .searchItem {
- background-image: url();
- background-size: 100% 100%;
- background-clip: content-box;
- background-origin: content-box;
- }
-
- #my_search_input {
- animation-duration: 1s;
- animation-name: my_search_view;
- outline: none;
- border: none;
- box-shadow: none;
- }
- #my_search_input:focus{
- outline: none;
- border: none;
- box-shadow: none;
- }
-
- .resultItem {
- animation-duration: 0.5s;
- animation-name: resultItem;
- }
- .resultItem .enter_main_link{
- display: flex !important ;
- justify-content: start;
- align-items: center;
- flex-grow:3;
- }
- /*关联图标样式*/
- .resultItem .vassal {
- /*对下面的svg位置进行调整*/
- display: flex !important;
- align-items: center;
- flex-shrink:0;
- margin-right:2px;
- }
- .related-links {
- margin: 0 3px;
- display: flex;
- gap: 4px; /* 增加链接之间的间距 */
- }
-
- .related-links > a {
- white-space: nowrap;
- line-height: 16px;
- font-size: 12px;
- padding: 3px 10px;
- background: #f0f4ff;
- color: #3578FE;
- text-decoration: none;
- transition: all 0.3s ease;
- }
-
- .related-links > a:hover {
- background: #e6f0ff; /* 悬停时背景颜色变浅 */
- border-color: #cce1ff; /* 改变边框颜色 */
- color: #255ec8; /* 深蓝色字体加深 */
- box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); /* 增加阴影效果 */
- }
-
- .resultItem svg{
- width: 16px;
- height:16px;
- }
- @-webkit-keyframes my_search_view {
-
- 0% {
- width: 0px;
- }
-
- 50% {
- width: 50%;
- }
-
- 100% {
- width: 100%;
- }
- }
-
- @-webkit-keyframes resultItem {
-
- 0% {
- opacity: 0;
- }
-
- 40% {
- opacity: 0.6;
- }
-
- 50% {
- opacity: 0.7;
- }
-
- 60% {
- opacity: 0.8;
- }
-
- 100% {
- opacity: 1;
- }
- }
-
- /*简述超链接样式*/
- #text_show a {
- color: #1a0dab !important;
- text-decoration:none;
- }
- /*自定义markdown的html样式*/
- #text_show>p>code {
- padding: 2px 0.4em;
- font-size: 95%;
- background-color: rgba(188, 188, 188, 0.2);
- border-radius: 5px;
- line-height: normal;
- font-family: SFMono-Regular,Consolas,'Liberation Mono',Menlo,monospace;
- color: #558eda;
- }
- #my_search_input::placeholder {
- color: #757575;
- }
- /*简述文本颜色为统一*/
- #text_show p {
- color: #202122;
- }
- /*让简述内容的li标签不受页面样式影响*/
- #text_show > ul > li {
- list-style-type: disc !important;
- }
- #text_show > ul > li > ul > li {
- list-style-type: circle !important;
- }
- #text_show > ol > li {
- list-style-type: decimal !important;
- }
- /*当视图大于等于1400.1px时*/
- @media (min-width: 1400.1px) {
- #my_search_box {
- left: 24%;
- right:24%;
- }
- }
- /*当视图小于等于1400px时*/
- @media (max-width: 1400px) {
- #my_search_box {
- left: 20%;
- right:20%;
- }
- }
- /*当视图小于等于1200px时*/
- @media (max-width: 1200px) {
- #my_search_box {
- left: 15%;
- right:15%;
- }
- }
- /*当视图小于等于800px时*/
- @media (max-width: 800px) {
- #my_search_box {
- left: 10%;
- right:10%;
- }
- }
- /*输入框右边按钮*/
- #logoButton {
- position: absolute;
- font-size: 12px;
- right: 5px;
- padding: 0px;
- border: none;
- display: block;
- background: rgba(255, 255, 255, 0);
- margin: 0px 7px 0px 0px;
- cursor: pointer !important;
- outline: none;
- }
- #logoButton:active {
- opacity: 0.4;
- }
- #logoButton img {
- display: block;
- width: 25px;
- }
-
- /*代码颜色*/
- #text_show code,#text_show pre{
- color:#5f6368;
- }
-
-
- /* 滚动条整体宽度 */
- #text_show::-webkit-scrollbar,
- #text_show pre::-webkit-scrollbar {
- -webkit-appearance: none;
- width: 5px;
- height: 5px;
- }
-
- /* 滚动条滑槽样式 */
- #text_show::-webkit-scrollbar-track,
- #text_show pre::-webkit-scrollbar-track {
- background-color: #f1f1f1;
- }
-
- /* 滚动条样式 */
- #text_show::-webkit-scrollbar-thumb,
- #text_show pre::-webkit-scrollbar-thumb {
- background-color: #c1c1c1;
- }
-
- #text_show::-webkit-scrollbar-thumb:hover,
- #text_show pre::-webkit-scrollbar-thumb:hover {
- background-color: #a8a8a8;
- }
-
- #text_show::-webkit-scrollbar-thumb:active,
- #text_show pre::-webkit-scrollbar-thumb:active {
- background-color: #a8a8a8;
- }
- /*结果项样式*/
- #matchResult li {
- line-height: 30.2px;
- height: 30.2px;
- color: #0088cc;
- list-style: none;
- width: 100%;
- padding: 0.5px;
- display: flex;
- justify-content: space-between;
- align-items: center;
- margin: 0.5px 0 !important;
- }
-
- #matchResult li > a {
- display: inline-block;
- font-size: 15.5px;
- text-decoration: none;
- text-align: left;
- cursor: pointer;
- font-weight: 400;
- background: rgb(255 255 255 / 0%);
- overflow: hidden;
- text-overflow: ellipsis;
- white-space: nowrap;
- margin: 0 3px;
- }
- #matchResult .flag {
- color: #fff;
- height: 21px;
- line-height: 21px;
- font-size: 10px;
- padding: 0px 6px;
- border-radius: 5px;
- font-weight: 600;
- box-sizing: border-box;
- margin-right: 3.5px;
- box-shadow: rgba(0, 0, 0, 0.3) 0px 0px 0.5px;
- }
- #matchResult .item_title {
- color: #1a0dab;
- }
- #matchResult .obsolete {
- text-decoration:line-through;
- color:#a8a8a8;
- }
- #matchResult .item_desc {
- color: #474747;
- overflow: hidden;
- text-overflow: ellipsis;
- white-space: nowrap;
- }
-
- #matchResult img {
- display: inline-block;
- width: 24px;
- height: 24px;
- margin: 0 6px 0 3px;
- box-shadow: 0 0 2px rgba(0, 0, 0, 0.5);
- border-radius: 30%;
- box-sizing: border-box;
- padding: 3px;
- flex-shrink: 0; /* 当容量不够时,不压缩图片的大小 */
- }
- #my_search_box {
- position: fixed;top:50px;
- border:2px solid #cecece;z-index:2147383656;
- background: #ffffff;
- }
- #my_search_box > #tis {
- position: absolute;
- left: 5px;
- top: -20px;
- font-size: 12px;
- color: #d5a436;
- font-weight: bold;
- }
-
- `)
-
-
- //防抖函数模板
- function debounce(fun, wait) {
- let timer = null;
- return function (...args) {
- // 清除原来的定时器
- if (timer) clearTimeout(timer)
- // 开启一个新的定时器
- timer = setTimeout(() => {
- fun.apply(this, args)
- }, wait)
- }
- }
- // 判断是否为指定指令
- function isInstructions(cmd) {
- let searchInputDocument = registry.view.element.input;
- if(searchInputDocument == null) return false;
- let regexString = "^\\s*:" + cmd + "\\s*$";
- let regex = new RegExp(regexString,"i");
- return regex.test(searchInputDocument.val());
- }
-
-
-
- // 获取一个同步执行器实例
- let pinyinActuator = syncActuator();
- // 向数据项中加入拼音项 如:title加了titlePinyin, desc加了descPinyin
- function genDataItemPinyin(threadHandleItems){
- let textPinyinMap = registry.searchData.getGlobalTextPinyinMap();
- // console.logout("分配的预热item:",threadHandleItems)
- pinyinActuator(()=>{
- if(threadHandleItems.length < 1) return;
- for(let item of threadHandleItems) {
- // 查看字典是否存在,只有没有预热过再预热
- if( textPinyinMap[threadHandleItems.title] != null ) continue;
- item.title.toPinyin();
- item.desc.toPinyin();
- }
- // 持久化-textPinyinMap字典 (这里需要判断是否值已经被初始化)
- if(textPinyinMap != null ) {
- cache.jSet(registry.searchData.TEXT_PINYIN_KEY,textPinyinMap);
- }
- });
- }
- // 当页面加载完成时触发-转拼音库操作
- const refresh = debounce(()=>{
- console.logout("==pinyin word==")
- let threadHandleItemSize = 100;
- let threadHandleItems = [];
- let currentSize = 0;
- let data = registry.searchData.getData();
- for(let item of data) {
- // 加入处理容器中
- threadHandleItems.push(item);
- currentSize++;
- // 判断是否已满
- if(currentSize >= threadHandleItemSize || data[data.length-1] == item ) {
- // 已满-去操作
- genDataItemPinyin(threadHandleItems);
- // 重置数据
- currentSize = 0;
- threadHandleItems = [];
- }
- }
- }, 2000)
- registry.searchData.dataChangeEventListener.push(refresh);
-
- // 实现模块一:使用快捷键触发指定事件
- function triggerAndEvent(goKeys = "ctrl+alt+s", fun, isKeyCode = false) {
- // 监听键盘按下事件
-
- let handle = function (event) {
- let isCtrl = goKeys.indexOf("ctrl") >= 0;
- let isAlt = goKeys.indexOf("alt") >= 0;
- let lastKey = goKeys.replace("alt", "").replace("ctrl", "").replace(/\++/gm,"").trim();
- // 判断 Ctrl+S
- if (event.ctrlKey != isCtrl || event.altKey != isAlt) return;
- if (!isKeyCode) {
- // 查看 lastKey == 按下的key
- if (lastKey.toUpperCase() == event.key.toUpperCase()) fun();
- } else {
- // 查看 lastKey == event.keyCode
- if (lastKey == event.keyCode) fun();
- }
-
- }
- // 如果使用 document.onkeydown 这种,只能有一个监听者
- $(document).keyup(handle);
- }
- function codeCopyMount(elementSelector) {
- document.querySelectorAll(`${elementSelector} pre code`).forEach((codeBlock) => {
- // 创建复制按钮
- const copyButton = document.createElement("button");
- copyButton.innerText = "复制";
- copyButton.classList.add("copy-btn");
-
- // 复制代码逻辑
- copyButton.addEventListener("click", () => {
- const text = codeBlock.innerText || codeBlock.textContent;
- navigator.clipboard.writeText(text).then(() => {
- copyButton.innerText = "已复制";
- setTimeout(() => (copyButton.innerText = "复制"), 2000);
- }).catch(err => {
- console.error("复制失败:", err);
- });
- });
-
- // 让 <pre> 相对定位,以便按钮放在右上角
- const pre = codeBlock.parentElement;
- pre.style.position = "relative";
-
- // 添加按钮到 <pre> 容器
- pre.appendChild(copyButton);
- });
- }
-
- // 【数据初始化】
- // 获取存在的订阅信息
- function getSubscribe() {
- // 查看是否有订阅信息
- let subscribeKey = registry.searchData.subscribeKey;
- let subscribeInfo = cache.get(subscribeKey);
- if(subscribeInfo == null ) {
- // 初始化订阅信息(初次)
- subscribeInfo = `
- <tis::https://raw.githubusercontent.com/18476305640/xiaozhuang/dev/%E6%88%91%E7%9A%84%E6%90%9C%E7%B4%A2%E8%AE%A2%E9%98%85%E6%96%87%E4%BB%B6.txt />
- `;
- cache.set(subscribeKey,subscribeInfo);
- }
- return subscribeInfo;
- }
- function editSubscribe(subscribe) {
- // 判断导入的订阅是否有效
- // 获取订阅信息(得到的值肯定不会为空)
- let pageTextHandleChainsY = new PageTextHandleChains(subscribe);
- let tisArr = pageTextHandleChainsY.parseAllDesignatedSingTags("tis");
- // 生成订阅信息存储
- let subscribeText = "\n" + pageTextHandleChainsY.rebuildTags(tisArr) + "\n";
- // 持久化
- let newSubscribeInfo = subscribeText.replace(/\n+/gm,"\n\n");
- cache.set(registry.searchData.subscribeKey,newSubscribeInfo);
- return tisArr.length;
- }
- // 存储订阅信息,当指定 sLineFetchFun 时,表示将解析“直接页”的配置,如果没有指定 sLineFetchFun 时,只解析内容
- // 在提取函数中 \n 要改写为 \\n
- function getDataSources() {
- let localDataSources = `
- <fetchFun name="mLineFetchFun">
- function(pageText) {
- let type = "sketch"; // url sketch
- let lines = pageText.split("\\n");
- let search_data_lines = []; // 扫描的搜索数据 {},{}
- let current_build_search_item = {};
- let appendTarget = "resource"; // resource 或 vassal
- let current_build_search_item_resource = ""; // 主要内容
- let current_build_search_item_vassal = ""; // 附加内容
- let current_build_search_item_links = []; // 快捷链接列表
- let point = 0; // 指的是上面的 current_build_search_item
- let default_desc = "--无描述--"
- function extractLinkInfo(str) {
- const regex = /\\[(.*?)\\]\\((https?:\\/\\/[^\\s]+)\\s*(?:\\s+"([^"]+)?")?\\s*\\)/;
- const match = str.match(regex);
- if (match) {
- return {
- text: match[1], // 链接文本
- url: match[2], // URL
- title: match[3] || '' // 标题,如果没有则为空字符串
- };
- } else {
- return null; // 如果没有匹配到内容,返回 null
- }
- }
- function isOnlyLinkLine(str) {
- // 按行拆分,并检查每一行
- return !str.split('\\n').some(line => line.trim() !== '' && !line.trim().startsWith('> '));
- }
- function getTitleLineData(titleLine) {
- try {
- const regex = /^#\\s*([^((]+)(?:[((](.*)[))])?\\s*$/;
- let matchData = regex.exec(titleLine)
- return {
- title: matchData[1],
- desc: ((matchData[2]==null || matchData[2] == "")?default_desc:matchData[2])
- }
- }catch(e) {
- 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;
- }
- </fetchFun>
- <fetchFun name="sLineFetchFun">
- function(pageText) {
- let type = "url"; // url sketch
- let lines = pageText.split("\\n");
- let search_data_lines = []
- for (let line of lines) {
-
- let search_data_line = (function(line) {
- const baseReg = /([^::\\n(())]+)[((]([^()()]*)[))]\\s*[::]\\s*(.+)/gm;
- const ifNotDescMatchReg = /([^::]+)\\s*[::]\\s*(.*)/gm;
- let title = "";
- let desc = "";
- let resource = "";
-
- let captureResult = null;
- if( !(/[()()]/.test(line))) {
- // 兼容没有描述
- captureResult = ifNotDescMatchReg.exec(line);
- if(captureResult == null ) return;
- title = captureResult[1];
- desc = "-暂无描述信息-";
- resource = captureResult[2];
- }else {
- // 正常语法
- captureResult = baseReg.exec(line);
- if(captureResult == null ) return;
- title = captureResult[1];
- desc = captureResult[2];
- resource = captureResult[3];
- }
- return {
- title: title,
- desc: desc,
- resource: resource
- };
- })(line);
- if (search_data_line == null || search_data_line.title == null) continue;
- search_data_lines.push(search_data_line)
- }
-
- for(let line of search_data_lines) {
- line.type = type;
- }
- return search_data_lines;
- }
- </fetchFun>
- ` + getSubscribe();
- return new Promise(async (resolve,reject)=>{
- // 这里请求tishub datasources
- // [ {name: "官方订阅",body: "<tis::http... />",status: ""} ] // status: disable enable
- const installHubTisList = cache.get(registry.searchData.USE_INSTALL_TISHUB_CACHE_KEY) || [];
- const installDataSources = installHubTisList.map(installTis => `${installTis.body}`).join("\n");
- resolve(installDataSources+localDataSources);
- })
- }
-
-
- // 判断是否是github文件链接
- let githubUrlTag = "raw.githubusercontent.com";
- // cdn模板+数据=完整资源加速链接 -> 返回
- function cdnTemplateWrapForUrl(cdnTemplate,initUrl) {
- let result = parseUrl(initUrl)??{};
- if(Object.keys(result) == 0 ) return null;
- return cdnTemplate.fillByObj(result);
- }
- // github CDN加速包装器
- // 根据传入的状态,返回适合的新状态(状态中包含资源加速下载链接|原始链接|null-表示不再试)
- let cdnPack = (function () { // index = 1 用原始的(不加速链接), -2 表示原始链接打不开此时要退出
-
- let cdnrs = cache.get(registry.other.UPDATE_CDNS_CACHE_KEY);
- // 提供的加速模板(顺序会在后面的请求中进行重排序-请求错误反馈的使重排序)
- // protocol、domain、path、params
- let initCdnrs = ["https://ghproxy.net/${rootUrl}${path}","https://ghps.cc/${rootUrl}${path}","https://github.moeyy.xyz/${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: {time: new Date().getTime()} ,config : {timeout: 5000} }).then(resolve).catch(()=>{
- console.log("CDN失败,不加速请求!");
- // 反馈错误,调整请求顺序,避免错误还是访问
- // 获取请求错误的根域名
- let { domain } = parseUrl(url);
- // 根据根域名从模板中找出完整域名
- let templates = allCdns.filter(item=>item.includes(domain));
- // 反馈
- if(templates.length > 0 ) {
- if(index > 0 || index <= cache.get(registry.other.UPDATE_CDNS_CACHE_KEY).length ) feedbackError(registry.other.UPDATE_CDNS_CACHE_KEY,templates[0]);
- }
- console.logout("反馈重调整后:",cache.get(registry.other.UPDATE_CDNS_CACHE_KEY)); // 反馈的结果只会在下次起作用
- // 处理失败后的回调函数代码
- rq(cdnPack({index,url,initUrl}));
- })
-
- }
- rq(cdnPack({index:0,url:dataSourceUrl,initUrl:dataSourceUrl}));
- });
- }
- // 下面的 dataSourceHandle 函数
- let globalFetchFun = [];
- // tis处理队列
- let waitQueue = [];
- // 缓存数据
- function cacheSearchData(newSearchData) {
- if(newSearchData == null) return;
- console.logout("触发了缓存,当前数据",registry.searchData.data)
- // 数据加载后缓存
- cache.set( registry.searchData.SEARCH_DATA_KEY,{
- data: newSearchData,
- expire: new Date().getTime() + registry.searchData.effectiveDuration
- })
- }
- // 更新历史数据
- function compareAndPushDiffToHistory(items = [],isCompared = false) {
- // 更新“旧全局数据”:searchData 追加-> oldSearchData
- let oldSearchData = cache.get(registry.searchData.OLD_SEARCH_DATA_KEY)??[];
- let newItemList = items;
- if(! isCompared && oldSearchData.length != 0) {
- // 比较后,差异项加入(取并集)
- newItemList = compareArrayDiff(items,oldSearchData,registry.searchData.idFun,1) ;
- }
- oldSearchData.push(... newItemList)
- console.log("旧数据缓存",oldSearchData)
- cache.set(registry.searchData.OLD_SEARCH_DATA_KEY,oldSearchData);
- if(! Array.isArray(newItemList)) newItemList = [];
- return newItemList;
- }
- // 防抖函数->处理新数据
- let blocks = [];
- let processingBlock = [];
- let triggerDataChageActuator = syncActuator();
- let refreshNewData = debounce(()=>{
- if(blocks.length == 0) return;
- // 倒动作
- processingBlock = blocks;
- blocks = [];
- // 将经过处理链得到的数据放到全局注册表中
- let globalSearchData = registry.searchData.getData();
- triggerDataChageActuator(()=>{
- globalSearchData.push(... registry.searchData.USDRC.trigger(processingBlock))
- // 数据版本改变
- registry.searchData.version++;
- // 更新视图显示条数
- registry.searchData.searchPlaceholder("UPDATE")
- // 触发搜索数据改变事件(做缓存等操作,观察者模式)
- for(let fun of registry.searchData.dataChangeEventListener) fun(globalSearchData);
- // 重新搜索
- registry.searchData.triggerSearchHandle();
- })
- }, 200) // 积累时间
- const triggerRefreshNewData = (block)=>{
- // 块积累
- blocks.push(...block);
-
- // 开始去处理
- refreshNewData();
- }
- // 转义与恢复,数据进行解析前进行转义,解析后恢复——比如文本中出现“/”,就会出现:SyntaxError: Octal escape sequences are not allowed in template strings.
- function CallBeforeParse() {
- this.obj = {
- "`":"<反引号>",
- "\\":"<转义>",
- "$": "<美元符>"
- }
- this.escape = function(text) {
- let obj = this.obj;
- for (var key in obj) {
- text = text.toReplaceAll(key,obj[key]);
- }
- return text;
- }
- this.recovery = function(text) {
- let obj = this.obj;
- for (var key in obj) {
- text = text.toReplaceAll(obj[key],key);
- }
- return text;
- }
- }
- let callBeforeParse = new CallBeforeParse();
-
- // recovery作用:将之前修改为 <wrapLine> 改为真正的换行符 \n
- function contentRecovery(item) {
- item.title = callBeforeParse.recovery(item.title);
- item.desc = callBeforeParse.recovery(item.desc);
- item.resource = callBeforeParse.recovery(item.resource);
- if(item.vassal != null ) item.vassal = callBeforeParse.recovery(item.vassal);
- }
- // 如果tisMetaInfo中有"default-tag"属性表示标签有这个属性,属性处理器在此
- function defaultTagHandle(item,tisMetaInfo = {}) {
- const defaultTag = tisMetaInfo['default-tag'];
- if(!defaultTag) return;
- // 假设defaultTag是 h'游戏' 那下面processedDefaultTag是 [h'游戏']
- const processedDefaultTag = `[${defaultTag}]`
- // defaultTagContent就是 游戏
- const defaultTagContent = parseTag(processedDefaultTag)[0][3];
- // 这里看item.title是否已经有 '游戏'] 或 [游戏] 如果都没有才加,也就是子数据项如果手动加就,default-tag就不会生效
- if( !parseTag(item.title).some(captureMeta => captureMeta[3] === defaultTagContent) ) {
- item.title = processedDefaultTag + item.title;
- }
- }
- // baseUrl + relativePath(文件 ./文件 ../文件)= relativePath的绝对路径
- function resolveUrl(baseUrl, relativePath) {
- // 创建一个链接对象,方便解析路径
- const base = new URL(baseUrl);
- // 处理相对路径
- const resolvedUrl = new URL(relativePath, base);
- return resolvedUrl.href;
- }
- function dataSourceHandle(resourcePageUrl,tisMetaInfo = {}, parentResourcePageUrl) { //resourcePageUrl 可以是url也可以是已经url解析出来的资源
- const tisTabFetchFunName = tisMetaInfo && tisMetaInfo.fetchFun;
- if(! registry.searchData.isDataInitialized) {
- registry.searchData.isDataInitialized = true;
- registry.searchData.processHistory = []; // 清空处理历史
- registry.searchData.clearData(); // 清理旧数据
- }
- let processHistory = registry.searchData.processHistory; // 处理过哪些链接需要记住,避免重复
- if(processHistory.includes(resourcePageUrl)) return; // 判断
- processHistory.push(resourcePageUrl); // 记录
- // 如果不根,且不是resourcePageUrl不是httpUrl,需要将resourcePageUrl(相对路径)根据parentResourcePageUrl(绝对路径)转为http url
- if( ! tisMetaInfo.root && !isHttpUrl(resourcePageUrl) ) {
- // if(parentResourcePageUrl == null) throw new Error(`订阅异常,相对路径: ${resourcePageUrl},没有父绝对路径!`);
- resourcePageUrl = resolveUrl(parentResourcePageUrl,resourcePageUrl);
- }
- urlToText(resourcePageUrl).then(text => {
- if(tisTabFetchFunName == null) {
- // --> 是配置 <--
- let data = []
- // 解析配置
- let config = getConfigFromDataSource(text);
- console.logout("解析的配置:",config)
- // 解析FetchFun:将FetchFun放到全局解析器中
- globalFetchFun.push(...config.fetchFuns);
- // 解析订阅:将tis放到处理队列中
- waitQueue.push(...config.tis);
- let tis = null;
- while((tis = waitQueue.pop()) != undefined) {
- // tis第一个是url,第二是fetchFun
- dataSourceHandle(tis.tabValue,tis, resourcePageUrl);
- }
- }else {
- // --> 是内容 <--
- // 解析内容
- if(tisTabFetchFunName === "") return;
- let fetchFunStr = getFetchFunGetByName(tisTabFetchFunName);
-
- let searchDataItems = [];
- try {
- searchDataItems =(new Function('text', "return ( " + fetchFunStr + " )(`"+callBeforeParse.escape(text)+"`)"))();
- }catch(e) {
- throw new Error("我的搜索 run log: 由于页面站点限制,导致数据解析失败!",e)
- }
- // 处理并push到全局数据容器中
- for(let item of searchDataItems) {
- // 转义-恢复
- contentRecovery(item);
- // "default-tag"标签属性处理器
- defaultTagHandle(item,tisMetaInfo)
- }
- // 加入到push到全局的搜索数据队列中,等待加入到全局数据容器中
- triggerRefreshNewData(searchDataItems)
- }
- })
-
-
- }
- // 根据fetchFun名返回字符串函数
- function getFetchFunGetByName(fetchFunName) {
- for(let fetchFunData of globalFetchFun) {
- if(fetchFunData.name == fetchFunName) {
- return fetchFunData.fetchFun;
- }
- }
- }
- // 检查是否已经执行初始化
- function checkIsInitializedAndSetInitialized(secondTime) {
- let key = "DATA_INIT";
- let value = cache.cookieGet(key);
- if(value != null && value != "") return true;
- cache.cookieSet(key,key,1000*secondTime);
- return false;
- }
- // 【数据初始化主函数】
- // 调用下面函数自动初始化数据,刚进来直接检查更新(如果数据已过期就更新数据)
- function dataInitFun() {
- // 从缓存中获取数据,判断是否还有效
- // cache.remove(SEARCH_DATA_KEY)
- let dataPackage = cache.get(registry.searchData.SEARCH_DATA_KEY);
- if(dataPackage != null && dataPackage.data != null) {
- // 缓存信息不为空,深入判断是否使用缓存的数据
- let dataExpireTime = dataPackage.expire;
- let currentTime = new Date().getTime();
- // 判断是否有效,有效的话放到全局容器中
- let isNotExpire = (dataExpireTime != null && dataExpireTime > currentTime && dataPackage.data != null && dataPackage.data.length > 0);
- // 如果网站比较特殊,忽略数据过期时间
- if( window.location.host.includes("github.com") ) isNotExpire = true;
-
- if(isNotExpire) {
- // 当视图已经初始化时-从缓存中将挂载数据挂载 (条件是视图已经初始化)
- console.logout(`视图${registry.view.initialized?'已加载':'未加载'}:数据有效期还有${parseInt((dataExpireTime - currentTime)/1000/60)} 分钟!`,dataPackage.data);
- if( registry.view.initialized ) registry.searchData.setData(dataPackage.data);
- // 如果数据状态未过期(有效)不会去请求数据
- return;
- }
-
- }
- // 在去网络请求获取数据前-检查是否已经执行初始化-防止多页面同时加载导致的数据重复加载
- if(! askIsExpiredByTopic("SEARCH_DATA_INIT",6*1000)) return;
- // 清理掉当前缓存数据
- cache.remove(registry.searchData.SEARCH_DATA_KEY);
- registry.searchData.clearData();
- // 重置数据初始化状态
- registry.searchData.isDataInitialized = false;
- // 持续执行
- registry.searchData.searchPlaceholder("UPDATE","🔁 数据准备更新中...",5000)
- // 内部将使用递归,解析出信息
- getDataSources().then(dataSources=>{dataSourceHandle(dataSources,{ root: true})})
- }
- // 检查数据有效性,且只有数据无效时挂载到数据
- dataInitFun();
- // 当视图第一次显示时,再执行
- registry.view.viewFirstShowEventListener.push(dataInitFun);
-
- // 解析标签函数-core函数
- function parseTag(title) {
- return captureRegEx(/\[\s*(([^'\]\s]*)\s*')?\s*([^'\]]*)\s*'?\s*]/gm,title);
- }
- // 解析出传入的所有项标签数据
- function parseTags(data = [],selecterFun = (_item)=>_item,tagsMap = {}) {
- let isArray = Array.isArray(data);
- let items = isArray?data:[data];
- // 解析 item.name中包含的标签
- items.forEach(function(item) {
- let captureGroups = parseTag(selecterFun(item));
- captureGroups.forEach(function(group) {
- let params = group[2]??"";
- let label = group[3];
- // 判断是否已经存在
- if(label != null && tagsMap[label] == null ) {
- let currentHandleTagObj = tagsMap[label] = {
- name: label,
- status: 1, // 正常
- //visible: params.includes("h"), // 参数中包含h字符表示可见
- count: 1
- //params: params
- //items: [item]
- }
- // 如果传入的不是一个数组,那设置下面参数才有意义
- if(! isArray) {
- currentHandleTagObj.params = params;
- }
- }else {
- if(tagsMap[label] != null) {
- tagsMap[label].count++;
- //tagsMap[label].items.push(item);
- }
-
- }
- })
- });
- // 这里不能是不是数组(上面的isArray)都返回tag数组,因为一项也可能有多个标签
- return Object.values(tagsMap);
- }
-
- let tagsMap = {}
- const parseSearchItem = function (searchData){
- console.log("==1:解析出数据标签==")
- // 将现有的所有标签提取出来
- // 解析
- let dataItemTags = parseTags(searchData,(_item=>_item.title),tagsMap);
- // 缓存
- if(dataItemTags.length > 0) {
- cache.set(registry.searchData.DATA_ITEM_TAGS_CACHE_KEY,dataItemTags)
- }
- return searchData;
- }
- // ################# 执行顺序从大到小 1000 -> 500
- registry.searchData.USDRC.add({weight:600 ,fun:parseSearchItem});
- // 解析script项的text
- function scriptTextParser(text) {
- if (text == null) return null;
- let scriptLines = text.split("\n");
- if (scriptLines != null && scriptLines.length != 0) {
- // 可以解析
- let result = {};
- let key = null;
- let value = null;
- for (let i = 0; i < scriptLines.length; i++) {
- let line = scriptLines[i];
- // 判断是否为新的变量开始
- let captureArr = captureRegEx(/^--\s*([^-\s]*)\s*--\s*$/gm, line);
- let isStartNewVar = captureArr != null && captureArr[0] != null && captureArr[0].length >= 2;
- let isLastLine = (i + 1 == scriptLines.length);
- if(isStartNewVar) {
- // 保存前面的
- if (key != null) result[key] = value.trim();
- // 开始新的
- key = captureArr[0][1];
- value = ""; // 重置value
- }else {
- value += ("\n" + line);
- }
-
- if ( isLastLine) {
- // 保存前面的
- if (key != null) result[key] = value.trim();
- return result;
- }
- }
- return result;
- }
-
- return null;
- }
- // 将形如“aa bb” 转为 {aa:"bb"} ,并且如果是布尔类型或数值字符串转为对应的类型
- function extractVariables(varsString) {
- const lines = varsString.split('\n');
- const result = {};
-
- for (const line of lines) {
- const parts = line.trim().split(/\s+/);
- if (parts.length === 2) {
- const key = parts[0].trim();
- const value = parts[1].trim();
-
- // 检查是否为 true/false 字符串
- if (value === 'true' || value === 'false') {
- result[key] = value === 'true'; // 转换为布尔值
- } else if (!isNaN(value)) { // 检查是否为数值字符串
- result[key] = parseFloat(value); // 转换为数值类型
- } else {
- result[key] = value; // 保持原始字符串
- }
- }
- }
-
- return result;
- }
- const parseScriptItem = function (searchData){
- console.log("==1:简述项解析出脚本项==")
- for(let item of searchData) {
- if((item == null || item.title == null) || item.type != "sketch" ) continue;
- if( /\[\s*(.*')?\s*(脚本|script)\s*'?\s*\]/.test( item.title ) ) {
- // 是脚本项
- item.type = "script";
- // 将resource解析为对象
- item.resourceObj = scriptTextParser(item.resource);
- item.resource = "--脚本项resource已解析到resourceObj--"
- // 解析脚本中的env(环境变量)
- if(item.resourceObj.env != null) {
- item.resourceObj.env = extractVariables(item.resourceObj.env);
- // 将提取的icon变量放到数据项根上,这样显示时,可读取作为icon
- let customIcon = item.resourceObj.env._icon;
- if( customIcon != null) item.icon = customIcon;
- let vassal = item.resourceObj.vassal;
- if(vassal != null) item.vassal = vassal;
- }
- }
- }
- return searchData;
- }
- // ################# 执行顺序从大到小 1000 -> 500
- registry.searchData.USDRC.add({weight:599 ,fun:parseScriptItem});
- // 监听缓存被清理,当被清理时,置空之前收集的标签数据
- registry.searchData.dataCacheRemoveEventListener.push(()=>{tagsMap = {}})
-
- const refreshTags = function (searchData){
- // 在添加前,进行额外处理添加,如给有”{keyword}“的url搜索项添加”可搜索“标签
- for(let searchItem of searchData) {
- let resource = searchItem.resource;
- let isHttpUrl =/^[^\n]*\.[^\n]*$/.test(`${resource}`.trim());
- let isSearchable = /\[\[[^\[\]]+keyword[^\[\]]+\]\]/.test(resource);
- // 判断是否为可搜索
- if( resource == null || !isHttpUrl || !isSearchable ) continue;
- if(! searchItem.title.includes(registry.searchData.searchProTag)) searchItem.title = registry.searchData.searchProTag+searchItem.title;
- }
- return searchData;
- }
-
- // ################# 执行顺序从大到小 1000 -> 500
- registry.searchData.USDRC.add({weight:500 ,fun:refreshTags});
- // 清理标签(参数中有h的)
- function clearHideTag(data,get = (item)=>item.title,set = (item,cleaned)=>{item.title=cleaned}) {
- let isArray = Array.isArray(data);
- let items = isArray?data:[data];
- for(let item of items) {
- let target = get(item);
- const regex = /\[\s*[^:\]]*h[^:\]]*\s*'\s*[^'\]]*\s*'\s*]/gm;
- let cleanedTarget = target.replace(regex, '');
- set(item,cleanedTarget);
- }
- return isArray?items:items[0];
- }
- // 给title清理掉“h”标签
- function clearHideTagForTitle(rawTitle) {
- const regex = /\[\s*[^:\]]*h[^:\]]*\s*'\s*[^'\]]*\s*'\s*]/gm;
- return rawTitle.replace(regex, '');
- }
- // 解析出标题中的所有标签-返回string数组
- function extractTagsAndCleanContent(inputString = "") {
- // 使用正则表达式匹配所有方括号包围的内容
- const regex = /\[.*?\]/g;
- const tags = inputString.match(regex) || [];
- // 清理掉标签的内容
- const cleanedContent = inputString.replace(regex, '').trim();
- return {
- tags: tags,
- cleaned: cleanedContent
- };
- }
- const filterSearchData = function (searchData) {
- const filterDataByUserUnfollowList = (itemsData,userUnfollowList = []) => {
- var userUnfollowMap = userUnfollowList.reduce(function(result, item) {
- result[item] = '';
- return result;
- }, {});
- // 开始过滤
- return itemsData.filter(item=>{
- let tags = parseTags(item.title);
- for(let tag of tags){
- if(userUnfollowMap[tag.name] != null){
- // 被过滤
- return false;
- }
- }
- return true;
- })
- }
- console.log("==去除用户不关注的数据项==")
- // 用户维护的取消关注标签列表
- let userUnfollowList = cache.get(registry.searchData.USER_UNFOLLOW_LIST_CACHE_KEY)?? registry.searchData.USER_DEFAULT_UNFOLLOW;
- // 利用用户维护的取消关注标签列表 过滤 搜索数据
- let filteredSearchData = filterDataByUserUnfollowList(searchData,userUnfollowList);
- // 去标签(参数h),清理每个item中title属性的tag , 下面注释掉是因为清理后置了仅在显示时不显示
- // let clearedSearchData = clearHideTag(filteredSearchData);
- return filteredSearchData;
- }
- // ############### 执行顺序从大到小 1000 -> 500
- registry.searchData.USDRC.add({weight:400 ,fun:filterSearchData});
- let isHasLaftData = true;
- const compareBlocks = function (searchData = []) {
- let oldSearchData = cache.get(registry.searchData.OLD_SEARCH_DATA_KEY)??[];
- if(isHasLaftData) isHasLaftData = oldSearchData != null && oldSearchData.length > 0;
- console.log("块数据与旧数据对比中>>")
- // 新数据加载完成-进行数据对比
- // 旧数据,也就是上一次数据,用于与本次比较,得出新添加数据
- // 当前时间戳
- let currentTime = new Date().getTime();
- // 准备一个存储新数据项的容器
- let newDataItems = compareAndPushDiffToHistory(searchData);
- // 给新添加的过期时间(新数据有效期)
- newDataItems.forEach(item=> {
- // 添加过期时间
- item.expires = (currentTime++) + ( 1000*60*60*24*registry.searchData.NEW_DATA_EXPIRE_DAY_NUM )
- });
- console.log("数据对比-新差异项:",[...newDataItems]);
- // 过滤掉新数据中带有“带注释”的项
- newDataItems = newDataItems.filter(item=> !item.title.startsWith("#"));
- // 以前的新增数据
- let oldNewItems = cache.get(registry.searchData.SEARCH_NEW_ITEMS_KEY);
- // 如果第一次加载数据,那不要这次的最新数据
- if(oldNewItems == null) {
- cache.set(registry.searchData.SEARCH_NEW_ITEMS_KEY,[]);
- return searchData;
- }
- // 如果还没有过期的,保留下来放在最新数据中
- for(let item of oldNewItems) {
- if(item != null && item.expires > currentTime) newDataItems.push(item);
- }
- console.log("数据对比-总新数据:",[...newDataItems])
- // 总新增去重 (标记 - 过滤标记的 )
- newDataItems = removeDuplicates(newDataItems,(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 = (`
- <div id="tis"></div>
- <div id="my_search_view">
- <div id="${searchBoxId}" >
- <div id="${msInputFilesId}">
- <!-- <div class="ms-input-file">
- <img src="https://greasyfork.org/vite/assets/blacklogo96-CxYTSM_T.png" />
- </div> -->
- </div>
- <input placeholder="${registry.searchData.searchPlaceholder()}" id="${searchInputDocumentId}" />
- <button id="${logoButtonId}" >
- <img src="${menu_icon}" draggable="false" />
- </button>
- </div>
- <div id="${matchResultDocumentId}">
- <ol id="${matchItemsId}">
- </ol>
- </div>
- <!--加“markdown-body”是使用了github-markdown.css 样式!加在markdown文档父容器中-->
- <div id="${textViewDocumentId}" class="${cssLoad("ms-markdown-body","markdown-css",{isResourceName:true, replacePrefix: 'markdown-body'})} ${cssLoad("ms-code-body","code-css",{isResourceName: true})}" style="min-height:auto !important;">
- </div>
- </div>
- `)
- // 挂载到文档中
- document.body.appendChild(view)
- // 整个视图对象放在组件全局中/注册表中
- viewDocument = registry.view.viewDocument = view;
- // 想要追加请看下面registry.view.element是否已经包含,没有在那下面追加即可~
-
- // 搜索框对象
- let searchInputDocument = $(document.getElementById(searchInputDocumentId)),
- matchItems = $(document.getElementById(matchItemsId)),
- searchBox = $(document.getElementById(searchBoxId)),
- logoButton = $(document.getElementById(logoButtonId)),
- textView = $(document.getElementById(textViewDocumentId)),
- matchResult = $(document.getElementById(matchResultDocumentId)),
- msInputFiles = $(document.getElementById(msInputFilesId));
- // 将视图对象放到注册表中
- registry.view.element = {
- input: searchInputDocument,
- logoButton,
- matchItems,
- searchBox,
- matchResult,
- textView,
- files: msInputFiles
- }
- // 开启files 粘贴事件监听
- registry.searchData.searchForFile.start();
- // 菜单函数(点击输入框右边按钮时会调用)
- function onClickLogo() {
- // alert("小彩蛋:可以搜索一下“系统项”了解脚本基本使用哦~");
- // 调用手动触发搜索函数,如果已经搜索过,搜索空串(清理)
- let keyword = "[系统项]";
- registry.searchData.triggerSearchHandle(searchInputDocument.val()==keyword?'':keyword);
- // 重新聚焦搜索框
- registry.view.element.input.focus()
- }
- const isLogoButtonPressedRef = registry.view.logo.isLogoButtonPressedRef;
- // 按下按钮时设置变量为 true
- logoButton.on('mousedown', function() {
- isLogoButtonPressedRef.value = true;
- });
-
- // 按钮弹起时设置变量为 false,并让输入框聚焦
- logoButton.on('mouseup', function() {
- isLogoButtonPressedRef.value = false;
- onClickLogo() // 触发logo点击事件
- searchInputDocument.focus(); // 输入框聚焦
- });
-
- // 防止鼠标拖出按钮后弹起无法触发 mouseup
- logoButton.on('mouseleave', function() {
- if (isLogoButtonPressedRef.value) {
- isLogoButtonPressedRef.value = false;
- searchInputDocument.focus(); // 输入框聚焦
- }
- });
-
- // 设置视图已经初始化
- registry.view.initialized = true;
-
- // 在搜索的结果集中上下选择移动然后回车(相当点击)
- searchInputDocument.keyup(function(event){
- let keyword = $(event.target).val().trim();
- // 当不为空时,放到全局keyword中
- if(keyword) {
- registry.searchData.keyword = event.target.value;
- }
- // 处理keyword中的":"字符
- if(keyword.endsWith("::") || keyword.endsWith("::")) {
- keyword = keyword.replace(/::|::/,registry.searchData.subSearch.searchBoundary).replace(/\s+/," ");
- // 每次要形成一个" : "的时候去掉重复的" : : " -> " : "
- keyword = keyword.replace(/((\s{1,2}:)+ )/,registry.searchData.subSearch.searchBoundary);
- $(event.target).val(keyword.toUpperCase());
- }
- });
- // searchInputDocument.keydown:这个监听用来处理其它键(非上下选择)的。
- searchInputDocument.keydown(function (event){
- // 阻止键盘事件冒泡 | 阻止输入框外监听到按钮,应只作用于该输入框
- event.stopPropagation();
- // 判断一个输入框的东西,如果如果按下的是删除,判断一下是不是"搜索模式"
- let keyword = $(event.target).val();
- let input = event.target;
- if(event.key == "Backspace" ) { // 按的是删除键-块删除
- if(keyword.endsWith(registry.searchData.subSearch.searchBoundary)) {
- // 取消默认事件-删除
- event.preventDefault();
- return;
- }else if(/^\s*[\[<][^\[\]<>]*[\]>]\s*$/.test( keyword )) {
- // 如果输入框只有[xxx]或<xxx>那就清空掉输入框
- searchInputDocument.val('')
- // keyword重置为空字符后触发搜索
- registry.searchData.triggerSearchHandle();
- event.preventDefault();
- return;
- }else if(keyword === ""){
- registry.searchData.searchForFile.delete();
- }
- }else if ( ! event.shiftKey && event.keyCode === 9 ) { // Tab键
- if(! registry.searchData.subSearch.isSubSearchMode()) {
- // 转大写
- event.target.value = event.target.value.toUpperCase()
- // 添加搜索pro模式分隔符
- event.target.value += registry.searchData.subSearch.searchBoundary
- // 阻止默认行为,避免跳转到下一个元素
- registry.searchData.triggerSearchHandle();
- }
- event.preventDefault();
- }else if (event.shiftKey && event.keyCode === 9 ) { // 按下shift + tab键时取消搜索模式
- if(registry.searchData.subSearch.isSubSearchMode()) {
- // 在这里编写按下shift+tab键时要执行的代码
- let input = event.target;
- input.value = input.value.split(registry.searchData.subSearch.searchBoundary)[0]
- event.target.value = event.target.value.toLowerCase();
- // 手动触发输入事件
- input.dispatchEvent(new Event("input", { bubbles: true }));
- }
- event.preventDefault();
- }
-
- })
-
- // searchInputDocument.keydown:这个监听用来处理上下选择范围的操作
- searchInputDocument.keydown(function (event){
- let e = event || window.event;
-
- if(e && e.keyCode!=38 && e.keyCode!=40 && e.keyCode!=13) return;
- if(e && e.keyCode==38){ // 上
- registry.searchData.pos --;
-
- }
- if(e && e.keyCode==40){ //下
- registry.searchData.pos ++;
- }
- // 如果是回车 && registry.searchData.pos == 0 时,设置 registry.searchData.pos = 1 (这样是为了搜索后回车相当于点击第一个)
- if(e && e.keyCode==13 && registry.searchData.pos == 0){ // 回车选择的元素
- // 如果当前是在搜索中就忽略回车这个操作
- if(registry.searchData.searchEven.isSearching) return;
- registry.searchData.pos = 1;
- }
-
- // 当指针位置越出时,位置重定向
- if(registry.searchData.pos < 1 || registry.searchData.pos > registry.searchData.searchData.length ) {
- if(registry.searchData.pos < 1) {
- // 回到最后一个
- registry.searchData.pos = registry.searchData.searchData.length;
- }else {
- // 回到第一个
- registry.searchData.pos = 1;
- }
- }
- // 设置显示样式
- let activeItem = $(registry.view.element.matchItems.find('li')[registry.searchData.pos-1]);
- // 设置活跃背景颜色
- let activeBackgroundColor = "#dee2e6";
- activeItem.css({
- "background":activeBackgroundColor
- })
-
- // 设置其它子元素背景为默认统一背景
- activeItem.siblings().css({
- "background":"#fff"
- })
-
- // 看是不是item detail内容显示中,如果是回车发送send事件,否则才是结果集显示的回车选择
- if(e && e.keyCode==13 && activeItem.find("a").length > 0 && !registry.script.tryRunTextViewHandler()){ // 回车
- // 点击当前活跃的项,点击
- 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,`<span style="${titleTagColorMatchHandler(tagCore)}" class="flag">${tagCore}</span>`);
- }
- 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 `<img src="${customIcon}" />`
- }else {
- return `<img src="${registry.searchData.getFaviconAPI(resource)}" standbyFavicon="${registry.searchData.getFaviconAPI(resource,true)}" class="searchItem" />`
- }
- }
-
- // 标题内容处理器
- function titleContentHandler(title) {
- // 对标题去掉所有tag
- const { cleaned } = extractTagsAndCleanContent(title)
- title = cleaned
- // 如果带#将加上删除线,通过加obsolete类名方式
- return `<span class="item_title ${title.startsWith("#")?'obsolete':''}">${title.replace(/^#/,"")}</span>`;
- }
-
-
- let matchItemsHtml = "";
- // 真正渲染到列表的数据项
- let searchData = []
- for(let searchResultItem of searchResultData ) {
- // 限制条数
- if(show_item_number-- <= 0 && !registry.searchData.isSearchAll) {
- break;
- }
- // 显示时清理标签-虽然在加载数据时已经清理了,但这是后备方案
- // clearHideTag(searchResultItem);
- // 将数据放入局部容器中
- searchData.push(searchResultItem)
-
- let isSketch = !isUrl(searchResultItem.resource);// searchResultItem.resource.trim().toUpperCase().indexOf("HTTP") != 0;
- let vassalSvg = `<svg t="1685187993813" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="3692" width="200" height="200"><path d="M971.904 372.736L450.901333 887.338667a222.976 222.976 0 0 1-312.576 0 216.362667 216.362667 0 0 1 0-308.736l468.906667-463.232a148.736 148.736 0 0 1 208.469333 0 144.298667 144.298667 0 0 1 0 205.824L346.752 784.469333a74.325333 74.325333 0 0 1-104.192 0 72.106667 72.106667 0 0 1 0-102.912l416.853333-411.733333-52.181333-51.456-416.853333 411.733333a144.298667 144.298667 0 0 0 0 205.781334 148.650667 148.650667 0 0 0 208.426666 0l468.906667-463.146667a216.490667 216.490667 0 0 0 0-308.736 223.061333 223.061333 0 0 0-312.661333 0L60.16 552.832l1.792 1.792a288.384 288.384 0 0 0 24.277333 384.170667c106.24 104.917333 273.322667 112.768 388.906667 23.936l1.792 1.834666L1024 424.192l-52.096-51.456z" fill="#666666" p-id="3693"></path></svg>`;
- // 构建快捷link html
- function buildRelatedLinksHtml(links) {
- if (links == null || links.length === 0) return '';
- let html = `<div class="related-links">`;
- // 遍历 links 数组,为每个链接生成对应的 <a> 标签
- links.forEach(link => {
- html += `<a href="${link.url}" target="_blank" title="${link.title}">${link.text}</a>`;
- });
- html += '</div>';
- return html;
- }
- // 将符合的数据装载到视图
- let item = `
- <li class="resultItem">
- <!--图标-->
- ${getFaviconImgHtml(searchResultItem)}
- <a href="${isSketch?'':searchResultItem.resource}" target="_blank" title="${searchResultItem.desc}" index="${searchResultItem.index}" version="${version}" class="enter_main_link">
- <!--tag与标题-->
- ${registry.view.titleTagHandler.execute(clearHideTagForTitle(searchResultItem.title))}${titleContentHandler(searchResultItem.title)}
- <!--描述信息-->
- <span class="item_desc">(${searchResultItem.desc})</span>
- </a>
- ${buildRelatedLinksHtml(searchResultItem.links)}
- ${searchResultItem.vassal !=null?'<a index="'+searchResultItem.index+'" version="'+version+'" vassal="true" class="vassal" title="查看相关联/同类项内容" target="_blank">'+vassalSvg+'</a>':''}
- </li>`
- matchItemsHtml += item;
- }
- matchItems.html(matchItemsHtml);
-
- let loadErrorTagIcon = "";
-
- // 给刚才添加的img添加事件
- for(let imgObj of registry.view.element.matchItems.find('img')) {
- // 加载完成事件,去除加载背景
- imgObj.onload = function(e) {
- $(e.target).css({
- "background": "#fff"
- })
- }
- // 加载失败,设置自定义失败的本地图片
- imgObj.onerror = function(e,a,b,c) {
- let currentErrorImg = $(e.target);
- let standbyFaviconAttr = "standbyFavicon";
- let standbyFavicon = currentErrorImg.attr(standbyFaviconAttr);
- if(standbyFavicon != null) {
- // 如果备用favicon使用
- currentErrorImg.attr("src",standbyFavicon)
- currentErrorImg.removeAttr(standbyFaviconAttr)
- }else {
- // 如果备用favicon直接使用加载失败图标base64
- currentErrorImg.attr("src",loadErrorTagIcon)
- }
- }
- }
-
- // 隐藏文本显示视图
- textView.css({
- "display":"none"
- })
- // 让搜索结果显示
- let matchResultDisplay = "block";
- if(searchResultData.length < 1) matchResultDisplay="none";
- matchResult.css({
- "display":matchResultDisplay,
- "overflow":"hidden"
- })
- // 将显示搜索的数据放入全局容器中
- registry.searchData.searchData = searchData;
- // 指令归位(置零)
- registry.searchData.pos = 0;
- }
-
- registry.view.element.matchItems.on("click","li > a",function(e) {
- let targetObj = e.target;
- // 如果当前标签是svg标签,那委托给父节点
- while ( targetObj != null && !/^(a|A)$/.test(targetObj.tagName)) {
- targetObj = targetObj.parentNode
- }
- // 取消默认事件,全部都是手动操作
- e.preventDefault();
- // 取消冒泡
- window.event? window.event.cancelBubble = true : e.stopPropagation();
- // 设置为阅读模式
- // $("#my_search_input").val(":read");
- // 获取当前结果在搜索数组中的索引
- let dataIndex = parseInt($(targetObj).attr("index"));
- let dataVersion = parseInt($(targetObj).attr("version"));
- let currentSearchDataVersion = registry.searchData.version;
- let itemData = registry.searchData.getData()[dataIndex];
- if(itemData == null || dataVersion != currentSearchDataVersion ) {
- console.log("后备方案(没有找到了?"+(itemData == null)+",数据版本改变了?"+(dataVersion != currentSearchDataVersion)+")")
- // 索引出现问题-启动后备方案-全局搜索
- let title = $(targetObj).parent().find(".item_title").text();
- let desc = $(targetObj).parent().find(".item_desc").text();
- // 从全局数据中根据title与desc进行匹配
- itemData = registry.searchData.findSearchDataItem(title,desc)
- // 从历史数据中找,根据title与desc进行匹配
- if(itemData == null) itemData = registry.searchData.findSearchDataItem(title,desc,SelectHistoryRecorder.history)
- }
- // 给选择的item加分,便于后面调整排序 (这里的idFun使用注册表中已经有的,也是我们确认item唯一的函数)
- if(itemData != null) DataWeightScorer.select(itemData,registry.searchData.idFun);
- // 记录选择的item项
- SelectHistoryRecorder.select(itemData,registry.searchData.idFun);
- // === 如果是简述搜索信息,那就取消a标签的默认跳转事件===
- let hasVassal = $(targetObj).attr("vassal") != null;
- // 初始化textView注册表中的对象
- function showTextPage(title,desc,body) {
- registry.view.textView.show(`<span style='color:red'>标题</span>:${title}<br /><span style='color:red'>描述:</span>${desc}<br /><span style='color:red'>简述内容:</span><br />${md2html(body)} `)
- }
- if(hasVassal) {
- showTextPage(itemData.title,"主项的相关/附加内容",itemData.vassal);
- // 挂载一键code复制
- codeCopyMount("#text_show");
- return;
- }else if(itemData.type == "script"){
- // 是脚本,执行脚本
- let callBeforeParse = new CallBeforeParse();
- let jscript = ( itemData.resourceObj == null || itemData.resourceObj.script == null ) ?"function (obj) {alert('- _ - 脚本异常!')}":itemData.resourceObj.script;
- // 调用里面的函数,传入注册表对象
- // 打开网址函数
-
- function open(url) {
- let openUrl = url;
- return {
- simulator(operate = (click, roll, dimension) => {}) { // 模拟器
- if(openUrl == null || operate == null || typeof operate != 'function') return;
- let pageSimulatorScript = operate.toString();
- addPageSimulatorScript(openUrl,pageSimulatorScript); // 保存模拟操作,模拟脚本将在指定时间内打开指定网址有效
- window.open(openUrl); // 打开网址
- return this;
- }
- }
- }
- let view = {
- beforeCallback: null,
- afterCallback: null,
- mountBefore(handle) {
- this.beforeCallback = handle;
- return this;
- },
- mountAfter(handle) {
- this.afterCallback = handle;
- return this;
- },
- // mount是脚本项-脚本js调用
- mount() {
- // 看脚本js是否给beforeCallback ,如果有在此执行
- if(this.beforeCallback != null) this.beforeCallback();
- // 挂载MS_SCRIPT_ENV 实现系统脚本API到视图
- registry.script.openSessionForMSSE();
- // 挂载视图
- let viewHtml = itemData.resourceObj['view:html'];
- let viewCss = itemData.resourceObj['view:css'];
- let viewJs = itemData.resourceObj['view:js'];
- registry.view.textView.show(viewHtml,viewCss,viewJs);
- // wait view complate alfter ...
- waitViewRenderingComplete(()=>{
- registry.script.tryRunTextViewHandler();
- // 看脚本js是否给afterCallback ,如果有在此执行
- if(this.afterCallback != null) this.afterCallback();
- })
- }
- }
- // 设置logo为运行图标
- registry.view.logo.change("")
- try {
- Function('obj',`(${jscript})(obj)`)({registry,cache,$,open,view})
- } catch (error) {
- setTimeout(()=>{alert("Ծ‸Ծ 你选择的是脚本项,而当前页面安全策略不允许此操作所依赖的函数!这种情况是极少数的,请换个页面试试!")},20)
- console.logout("脚本执行失败!",error);
- }
- // logo图标还原
- setTimeout(()=>{registry.view.logo.reset();},200)
- return;
- }else if(! isUrl(itemData.resource)) {
- showTextPage(itemData.title,itemData.desc,itemData.resource)
- return;
- }
- // 隐藏视图
- registry.view.viewVisibilityController(false)
-
- const initUrl = itemData.resource;//$(targetObj).attr("href"); // 不作改变的URL
- let url = initUrl; // 进行修改,形成要跳转的真正url
- let temNum = url.matchFetch(/\[\[[^\[\]]*\]\]/gm, function (matchStr,index) { // temNum是url中有几个 "[[...]]", 得到后,就已经得到解析了
- let templateStr = matchStr;
- // 使用全局的keyword, 构造出真正的keyword
- let keyword = registry.searchData.keyword.split(":").reverse();
- keyword.pop();
- keyword = keyword.reverse().join(":").trim();
-
- let parseAfterStr = matchStr.replace(/{keyword}/g,keyword).replace(/\[\[+|\]\]+/g,"");
- url = url.replace(templateStr,parseAfterStr);
- });
- // 如果搜索的真正keyword为空字符串,则去掉模板跳转
- if( registry.searchData.keyword.split(registry.searchData.subSearch.searchBoundary).length < 2
- || registry.searchData.keyword.split(registry.searchData.subSearch.searchBoundary)[1].trim() == "" ) {
- url = registry.searchData.clearUrlSearchTemplate(initUrl);
- }
- // 跳转(url如果有模板,可能已经去掉模板,取决于是“搜索模式”)
- window.open(url);
-
- })
- //registry.searchData.searchHandle = handler;
- const refresh = debounce(handler, 300)
- // 第一次触发 scroll 执行一次 fn,后续只有在停止滑动 1 秒后才执行函数 fn
- searchInputDocument.on('input', refresh)
- }
- function ensureViewHide() {
- // 隐藏视图
- // 如果视图还没有初始化,直接退出
- if (! registry.view.initialized) return;
- // 如果正在查看查看“简讯”,先退出简讯
- const nowMode = registry.view.seeNowMode();
- if(nowMode === registry.view.modeEnum.SHOW_ITEM_DETAIL) {
- // 让简讯隐藏
- registry.view.element.textView.css({"display":"none"})
- // 让搜索结果显示
- registry.view.element.matchResult.css({ display:"block",overflow: "hidden" })
- // 通知简讯back事件
- registry.view.itemDetailBackAfterEventListener.forEach(listener=>listener())
- return;
- }
- // 让视图隐藏
- viewDocument.style.display = "none";
- // 将输入框内容置空,在置空前将值备份,好让未好得及的操作它
- registry.view.element.input.val("")
- // 将之前搜索结果置空
- registry.view.element.matchItems.html("")
- // 隐藏文本显示视图
- registry.view.element.textView.css({
- "display":"none"
- })
- // 让搜索结果隐藏
- registry.view.element.matchResult.css({
- "display":"none"
- })
- // 视图隐藏-清理旧数据
- registry.searchData.clearData();
- // 触发视图隐藏事件
- registry.view.viewHideEventAfterListener.forEach(fun=>fun());
- }
- function showView() {
- // 让视图可见
- viewDocument.style.display = "block";
- //聚焦
- registry.view.element.input.focus()
- // 当输入框失去焦点时,隐藏视图
- registry.view.element.input.blur(function() {
- const isLogoButtonPressedRef = registry.view.logo.isLogoButtonPressedRef
- if(isLogoButtonPressedRef.value) {
- console.logout("隐藏跳过,因为isLogoButtonPressedRef")
- return
- };
- setTimeout(function(){
- const isDebuging = isInstructions("debug");
- const isSearching = registry.searchData.searchEven.isSearching;
- // 当前视图是否在展示数据,如搜索结果,简述内容?如果在展示不隐藏
- let isWaitSearch = registry.view.seeNowMode() === registry.view.modeEnum.WAIT_SEARCH;
- if(isDebuging || isSearching || !isWaitSearch || isLogoButtonPressedRef.value) {
- console.logout("隐藏跳过,条件列表不满足!")
- return
- };
- registry.view.viewVisibilityController(false);
- },registry.view.delayedHideTime)
- });
- }
-
- // 返回给外界控制视图显示与隐藏
- return function (isSetViewVisibility) {
- if (isSetViewVisibility) {
- // 让视图可见 >>>
- // 如果还没初始化先初始化 // 初始化数据 initData();
- if (! registry.view.initialized) {
- // 初始化视图
- initView();
- // 初始化数据
- // initData();
- }
- // 让视图可见
- showView();
- } else {
- // 隐藏视图 >>>
- ensureViewHide();
- }
- }
- })();
- // 触发策略——快捷键
- let useKeyTrigger = function (viewVisibilityController) {
- let isFirstShow = true;
- // 将视图与触发策略绑定
- function showFun() {
- // 让视图可见
- viewVisibilityController(true);
- // 触发视图首次显示事件
- if(isFirstShow) {
- for(let e of registry.view.viewFirstShowEventListener) e();
- isFirstShow = false;
- }
- }
- window.addEventListener('message', event => {
- // console.log("父容器接收到了信息~~")
- if(event.data == MY_SEARCH_SCRIPT_VIEW_SHOW_EVENT) {
- showFun() // 接收显示呼出搜索框
- }
- });
- triggerAndEvent("ctrl+alt+s", showFun)
- triggerAndEvent("Escape", function () {
- // 如果视图还没有初始化,就跳过
- if(registry.view.viewDocument == null ) return;
- // 让视图不可见
- viewVisibilityController(false);
- })
- }
-
- // 触发策略组
- let trigger_group = [useKeyTrigger];
- // 初始化入选的触发策略
- (function () {
- for (let trigger of trigger_group) {
- trigger(registry.view.viewVisibilityController);
- }
- })();
-
- // 打开视图进行配置
- // 显示配置视图
- // 是否显示进度 - 进度控制
- function clearCache() {
- cache.remove(registry.searchData.SEARCH_DATA_KEY);
- // 如果处于debug模式,也清理其它的
- if(isInstructions("debug")) {
- cache.remove(registry.searchData.CACHE_FAVICON_SOURCE_KEY);
- }
- // 触发缓存被清理事件
- for(let fun of registry.searchData.dataCacheRemoveEventListener) fun();
- }
- GM_registerMenuCommand("订阅管理",function() {
- showConfigView();
- });
- GM_registerMenuCommand("清理缓存",function() {
- clearCache();
- });
-
- function giveTagsStatus(tagsOfData,userUnfollowList) {
- // 赋予tags一个是否选中状态
- // 将 userUnfollowList 转为以key为userUnfollowList的item.name值是Item的方便检索
- let userUnfollowMap = userUnfollowList.reduce(function(result, item) {
- result[item] = '';
- return result;
- }, {});
- tagsOfData.forEach(item=>{
- if(userUnfollowMap[item.name] != null ) {
- // 默认都是选中状态,如果item在userUnfollowList上将此tag状态改为未选中状态
- item.status = 0;
- }
- })
- return tagsOfData;
- }
- function showConfigView() {
- // 剃除已转关注的,添加新关注的
- function reshapeUnfollowList(userUnfollowList,userFollowList,newUserUnfollowList) {
- // 剃除已转关注的
- userUnfollowList = userUnfollowList.filter(item => !userFollowList.includes(item));
- // 添加新关注的
- userUnfollowList = userUnfollowList.concat(newUserUnfollowList.filter(item => !userUnfollowList.includes(item)));
- return userUnfollowList;
- }
-
- if($("#subscribe_save")[0] != null) return;
- // 显示视图
- // 用户维护的取消关注标签列表
- let userUnfollowList = cache.get(registry.searchData.USER_UNFOLLOW_LIST_CACHE_KEY)?? registry.searchData.USER_DEFAULT_UNFOLLOW;
- // 当前数据所有的标签
- let tagsOfData = cache.get(registry.searchData.DATA_ITEM_TAGS_CACHE_KEY);
- // 使用 userUnfollowList 给 tagsOfData中的标签一个是否选中状态,在userUnfollowList中不选中,不在选中,添加一个属性到tagsOfData用boolean表达
- tagsOfData = giveTagsStatus(tagsOfData,userUnfollowList);
- // 生成多选框html
- let tagsCheckboxHtml = "";
- tagsOfData.forEach(item=>{
- tagsCheckboxHtml += `
- <div>
- <input type="checkbox" id="${item.name}" name="_tagsCheckBox" value="${item.name}" ${item.status==1?'checked':''} >
- <label for="${item.name}">${item.name} (${item.count})</label>
- </div>
- `
- })
-
- DivPage(`
- #my-search-view {
- width: 500px;
- max-height: 100%;
- max-width: 100%;
- background: pink;
- position: fixed;
- right: 0px;
- top: 0px;
- z-index: 2147383656;
- padding: 20px;
- box-sizing: border-box;
- border-radius: 14px;
- text-align: left;
- button {
- cursor: pointer;
- }
-
- ._topController {
- width: 100%;
- position: absolute;
- top: 0px;
- right: 0px;
- text-align: right;
- padding: 15px 15px 0px;
- box-sizing: border-box;
- * {
- cursor: pointer;
- }
- #topController_close {
- font-sise: 15px;
- color: #e8221e;
- }
- }
- .page {
- .control_title {
- margin: 10px 0px 5px;
- font-size: 17px;
- color: black;
- }
- }
- .home {
- .submitable {
- color: #3CB371;
- }
- .tagsCheckBoxDiv > div {
- width: 32%;
- display: inline-block;
- margin: 0px;
- padding: 0px;
- overflow: hidden;
- text-overflow: ellipsis;
- white-space: nowrap;
- }
- #all_subscribe {
- width: 100%;
- height: 150px;
- box-sizing: border-box;
- border: 4px solid #f5f5f5;
- }
- #subscribe_save {
- margin-top: 20px;
- border: none;
- border-radius: 3px;
- padding: 4px 17px;
- cursor: pointer;
- box-sizing: border-box;
- background: #6161bb;
- color: #fff;
- }
- .view-base-button {
- background: #fff;
- border: none;
- font-size: 15px;
- padding: 1px 10px;
- cursor: pointer;
- margin: 2px;
- color: black;
- }
- ._topController span {
- color: #3CB371;
- }
- .home label {
- font-size: 13px;
- }
- }
- .tis-hub {
- .logo-search {
- display: flex;
- flex-direction: column;
- align-items: center;
- img {
- display: block;
- width: 40px;
- height: 40px;
- }
- .keyword {
- display: flex;
- font-size: 12px;
- width: 70%;
- margin-top: 5px;
- input {
- border: none;
- padding: 0 6px;
- min-width: 100px;
- line-height: 25px;
- height: 25px;
- flex-grow: 1;
- }
- button {
- padding: 0 12px;
- border: none;
- background: #f0f0f0;
- line-height: 25px;
- height: 25px;
- }
- }
- }
- .search-type {
- display: flex;
- padding: 10px 0;
- label {
- display: flex;
- align-items: center;
- margin-right: 20px;
- font-size: 14px;
- input {
- padding: 0;
- margin: 0 3px 0 0;
- }
- }
- }
- .result-list {
- min-height: 300px;
- padding-top: 15px;
- .hub-tis {
- display: flex;
- justify-content: space-between;
- margin-bottom: 12px;
- align-items: center;
- button {
- font-size: 10px;
- line-height: 22px;
- height: 22px;
- padding: 0 15px;
- border-radius: 3px;
- border: none;
- }
- .tis-info {
- display: flex;
- flex-direction: column;
- .title {
- font-size: 14px;
- font-weight: bold;
- color: rgb(103, 0, 0);
- }
- .describe {
- font-size: 12px;
- font-weight: 400;
- display: block;
- font-size: smaller;
- margin: 0.5em 0px;
- color: #333333;
- }
-
- }
- }
- }
- }
- }
- `,`
- <div id="my-search-view">
- <div class="_topController">
- <span id="topController_close">X</span>
- </div>
- <div class="page home">
- <div>
- <p class="control_title">订阅总览:</p>
- <textarea id="all_subscribe"></textarea>
- </div>
- <div>
- <p class="control_title">公共仓库:</p>
- <button id="pushTis" class="view-base-button">提交我的订阅到TisHub(<span class="submitable"> - </span>)</button>
- <button id="openTisHub" class="view-base-button">Tis订阅市场</button>
- <button id="clearToken" class="view-base-button" style="display:none;">清理Token (存在)</button>
- </div>
- <div>
- <p class="control_title">关注标签:</p>
- <div class="tagsCheckBoxDiv">
- ${tagsCheckboxHtml}
- </div>
- </div>
- <button id="subscribe_save">保存并应用</button>
- </div>
- <div class="page tis-hub">
- <p class="control_title">订阅市场</p>
- <div class="logo-search">
- <a href="https://github.com/My-Search/TisHub" target="_blank">
- <img src="https://cdn.jsdelivr.net/gh/My-Search/TisHub/favicon.ico" title="TisHub是一个GitHub仓库,订阅以Issues的方式存在!" />
- </a>
- <div class="keyword">
- <input name="keyword" placeholder="请输入搜索关键字..." />
- <button id="search-tishub">搜索</button>
- </div>
- </div>
- <div class="search-type">
- <label>
- <input type="radio" name="search-type" value="installed" checked>
- 已安装
- </label><br>
- <label>
- <input type="radio" name="search-type" value="market">
- 市场订阅
- </label>
- </div>
- <div class="result-list">
- <div class="list-rol">
- </div>
- </div>
- </div>
- </div>
-
- `,function (selector,remove) {
- let subscribe_text = selector("#all_subscribe");
- let subscribe_save = selector("#subscribe_save");
- let topController_close = selector("#topController_close");
- let openTisHub = selector("#openTisHub");
- let tisHubLink = "https://github.com/My-Search/TisHub/issues";
- let pushTis = selector("#pushTis");
- let commitableTisList = null;
- let clearToken = selector("#clearToken");
- let mySearchView = selector("#my-search-view");
- let currentPage = setPage(); // 默认显示的是home页
-
- // 刷新页
- function setPage(page = "home") {
- $(mySearchView).find('.page').hide().filter(`.${page}`).show();
- }
- setPage("home");
- // 刷新视图状态
- async function refreshViewState() {
- // 更新token状态
- $(clearToken).css({"display":GithubAPI.getToken() == null?"none":"inline-block"})
- // 更新可提交数
- let tisList = await TisHub.getTisHubAllTis();
- if(tisList != null && tisList.length != 0) {
- commitableTisList = TisHub.tisFilter(subscribe_text.value,tisList)??[]
- $(pushTis).find("span").text(commitableTisList.length);
- }
- }
- // 初始化subscribe_text的值
- subscribe_text.value = getSubscribe();
- // 初始化其它状态,通过调用refreshViewState()
- refreshViewState();
- // 当SubscribeText多行输入框内容发生改变时,刷新更新可提交数,通过调用refreshViewState()
- let refreshSubscribeText = debounce(()=>{refreshViewState() }, 300)
- subscribe_text.oninput = ()=>{refreshSubscribeText();}
- // 保存
- function configViewClose() {
- remove();
- }
- // 点击保存时
- subscribe_save.onclick=function() {
- // 保存用户选择的关注标签(维护数据)
- // 获取所有多选框元素
- var checkboxes = selector(".tagsCheckBoxDiv input",true);
- // 初始化已选中和未选中的数组
- var userFollowList = [];
- var newUserUnfollowList = [];
- // 遍历多选框元素,将选中的元素的value值添加到checkedValues数组中,
- // 未选中的元素的value值添加到uncheckedValues数组中
- for (var i = 0; i < checkboxes.length; i++) {
- if (checkboxes[i].checked) {
- userFollowList.push(checkboxes[i].value);
- } else {
- newUserUnfollowList.push(checkboxes[i].value);
- }
- }
- // 剃除已转关注的,添加新关注的
- newUserUnfollowList = reshapeUnfollowList( userUnfollowList,userFollowList,newUserUnfollowList);
- cache.set(registry.searchData.USER_UNFOLLOW_LIST_CACHE_KEY,newUserUnfollowList);
-
- // 保存到对象
- let allSubscribe = subscribe_text.value;
- let validCount = editSubscribe(allSubscribe);
- // 清除视图
- configViewClose();
- // 清理缓存,让数据重新加载
- clearCache();
- alert("保存配置成功!有效订阅数:"+validCount);
-
- }
- // 打开TitHub
- openTisHub.onclick = function() {
- // window.open(tisHubLink, "_blank");
- setPage("tis-hub");
- }
- // push到TisHub公共仓库中
- pushTis.onclick =async function () {
- if(! confirm("是否确认要提交到TisHub公共仓库?")) return;
- if(commitableTisList == null || commitableTisList.length == 0) {
- alert("经过与TisHub中订阅的比较,本地没有可提交的订阅!")
- return;
- }
- if(GithubAPI.getToken(true) == null) {
- alert("获取token失败,无法继续!");
- return;
- }
- // 组装提交的body
- let body = (()=>{
- let _body = "";
- for(let tis of commitableTisList) _body+=tis;
- return _body;
- })();
- if ( body == "") return;
- let userInfo = await GithubAPI.setToken().getUserInfo();
- if(userInfo == null) {
- alert("提交异常,请检查网络或提交的Token信息!")
- return;
- }
- GithubAPI.commitIssues({
- "title": userInfo.name+"的订阅",
- "body": body
- }).then(response=>{
- refreshViewState();
- alert("提交成功(issues)!感谢您的参与,脚本因你而更加精彩。")
- }).catch(error=>alert("提交失败~"))
- }
- // 清理token
- clearToken.onclick = function(){
- GithubAPI.clearToken(); // 清理token
- refreshViewState(); // 刷新视图变量
- };
- // 关闭
- $(topController_close).click(configViewClose)
-
- // 点击搜索tis-hub
- let installedList = cache.get(registry.searchData.USE_INSTALL_TISHUB_CACHE_KEY) || []; // [ {name: "官方订阅",describe: "这是官方订阅...", body: "<tis::http... />",status: ""} ] status: disable enable installable
- let tisSearchInput = $(".tis-hub .keyword input");
- let tisSearchBtn = $("#search-tishub");
- let searchFun = async function() {
- const keyword = tisSearchInput.val()?.trim() || '';
- // 搜索类型(installed | market)
- const searchType = $('.search-type input[name="search-type"]:checked').val();
- let resultTisList = installedList.filter(item => keyword === "" || item.name.includes(keyword));
- if(searchType === "market") {
- let marketResult = await TisHub.getClosedIssuesTis({keyword})
- marketResult = marketResult.map(hubTisInfo => {
- return {
- name: hubTisInfo.title,
- describe: hubTisInfo.describe,
- body: hubTisInfo.tisList.join('\n') || '',
- state: "installable"
- }
- })
- const installedMap = resultTisList.reduce((map, item) => {
- map[item.name] = item;
- return map;
- }, {});
-
- // 看本地是否已安装,如果已安装state就取已安装的项state
- (resultTisList = marketResult).forEach(hubTis =>{
- if(installedMap[hubTis.name]) hubTis.state = installedMap[hubTis.name].state;
- });
- }
- // 列表渲染
- const resultElement = $(".tis-hub .result-list > .list-rol");
- resultElement.html('')
- // 转状态名
- function stateAsName(state) {
- return (state === "disable" && "移除(未启用)") || (state === "enable" && "移除") || "安装";
- }
-
- for(let tis of resultTisList) {
- // tis 有该订阅的名 tis.name
- // tisMetaInfo 是tis.body 包含描述信息 tisMetaInfo.describe
- const tisMetaInfo = new PageTextHandleChains(tis.body).parseAllDesignatedSingTags("tis")[0];
- // 自己的搜索逻辑
- if(! `${tis.name}`.includes(keyword) && (tisMetaInfo != null && ! `${tisMetaInfo.describe}`.includes(keyword))) return;
- // 渲染到页面
- resultElement.append(`
- <div class="hub-tis">
- <div class="tis-info">
- <a class="title" href="${tisMetaInfo.tabValue}" target="_blank">${tis.name}</a>
- <span class="describe">${tisMetaInfo.describe || '订阅没有描述信息,请确认订阅安全或信任后再安装!'}</span>
- </div>
- <button class="tis-button" tis-name="${tis.name}">${ stateAsName(tis.state) }</button>
- </div>
- `)
- }
- // 当点击tis-button按钮时
- $(".hub-tis .tis-button").click(function() {
- // 使用 $(this) 获取当前被点击的元素
- const button = $(this);
- const tisName = button.attr("tis-name");
- let tis = installedList.find(item=>item.name === tisName);
- if(tis != null) {
- // 移除
- installedList = installedList.filter(item => item.name !== tisName);
- tis.state = "installable";
- }else {
- // 安装
- const hubTis = resultTisList.find(item=>item.name === tisName);
- hubTis.state = "enable";
- installedList.unshift(tis = hubTis);
- }
- // 更新状态
- button.html(stateAsName(tis.state));
- // 保存
- console.log("保存:",installedList)
- cache.set(registry.searchData.USE_INSTALL_TISHUB_CACHE_KEY,installedList);
- // 清理缓存
- clearCache()
- });
- }
- // 点击搜索
- const searchButton = tisSearchBtn.click(searchFun).click();
- // 回车键触发搜索
- tisSearchInput.on("keydown", function(event) {
- if (event.key === "Enter") {
- searchFun(); // 调用搜索功能
- }
- });
-
- // 单选框值改变时,搜索
- const radioButtons = document.querySelectorAll('input[name="search-type"]');
- radioButtons.forEach(radio => {
- radio.addEventListener('change', function() {
- if (this.checked) {
- searchButton.click();
- }
- });
- });
-
- })
- }
-
- })(unsafeWindow);
- // unsafeWindow是真实的window,作为参数传入