// ==UserScript== // @name 我的搜索 // @namespace http://tampermonkey.net/ // @version 3.4.0 // @description 打造订阅式搜索,让我的搜索,只搜精品! // @license MIT // @author zhuangjie // @match *://*/* // @exclude http://127.0.0.1* // @exclude http://localhost* // @exclude http://192.168.* // @icon  // @require https://cdn.bootcdn.net/ajax/libs/jquery/3.6.2/jquery.min.js // @grant window.onurlchange // @grant GM_setValue // @grant GM_getValue // @grant GM_deleteValue // @grant GM_registerMenuCommand // @downloadURL none // ==/UserScript== (function() { 'use strict'; // 模块一:快捷键触发某一事件 (属于触发策略组) // 模块二:搜索视图(显示与隐藏)(属于搜索视图组) // 模块三:触发策略组触发策略触发搜索视图组视图 // 模块四:根据用户提供的策略(策略属于数据生成策略组)生成搜索项的数据库 // 模块五:视图接入数据库 // 引入js $(document.head).html($(document.head).html()+` `) // 数据缓存器 let cache = { get(key) { return GM_getValue(key); }, set(key,value) { GM_setValue(key,value); }, jGet(key) { let value = GM_getValue(key); if( value == null) return value; return JSON.parse(value); }, jSet(key,value) { value = JSON.stringify(value) GM_setValue(key,value); }, remove(key) { GM_deleteValue(key); }, cookieSet(cname,cvalue,exdays) { var d = new Date(); d.setTime(d.getTime()+exdays); var expires = "expires="+d.toGMTString(); document.cookie = cname + "=" + cvalue + "; " + expires; }, cookieGet(cname) { var name = cname + "="; var ca = document.cookie.split(';'); for(var i=0; i { ERROR.tell("视图未初始化,但你使用了它的未初始化的注册表信息!") }, viewDocument: null, setButtonVisibility: () => { ERROR.tell("按钮未初始化!") }, titleHandlerFuns: [], showControlButton(buttonStr,callback = ()=>{}) { // 会显示一个按钮 // 初始化按钮 let viewDocument = $(this.viewDocument); if(viewDocument == null) return; // 视图已初始化,可以显示按钮 let controlButton = viewDocument.find("#controlButton") controlButton.text(buttonStr); controlButton.click(callback); // 设置合理宽度 controlButton.css({ "width":"20%" }) $("#search_input").css({ "width":"77%" }) controlButton.show(); }, hideControlButton() { // 隐藏掉输入框右边按钮 let controlButton = $(this.viewDocument).find("#controlButton") $("#search_input").css({ "width":"100%" }) controlButton.hide(); } }, searchData: { //registry.searchData.isSearchPro data: [], subscribeKey: "subscribeKey", showSize: 15, // 事件函数 dataChange: [], // 搜索的keyword keyword: "", // 持久化Key SEARCH_DATA_KEY: "SEARCH_DATA_KEY", // 搜索搜索出来的数据 searchData: [], pos: 0, getFaviconAPI: "https://ico.di8du.com/get.php?url=", tmpVar: null, // 用于防抖 getDataLength(target = "SELECT") { // 持久化的数据项数 let cacheData = cache.get(this.SEARCH_DATA_KEY) == null?[]:cache.get(this.SEARCH_DATA_KEY).data; let tmpData = this.data.length === 0?cacheData:this.data; let dataLength = (tmpData == null)?0:tmpData.length; // 全部的输入提示 let inputDescs = ["我的搜索,只搜精品","想要提交精品内容,快搜索“申请录入”吧~","想了解脚本的新功能?搜索“使用说明”吧~"] // 当前应用“输入提示” let inputDesc = inputDescs[Math.floor(Math.random()*inputDescs.length)]; if(target == "UPDATE") { if(this.tmpVar != null) { clearTimeout(this.tmpVar); } this.tmpVar = setTimeout(()=>{ $("#search_input").attr("placeholder",this.getDataLength()); },1200) return `可以搜索( 🔁 数据库更新到 ${dataLength}条)`; } return inputDesc; }, searchBoundary: " : ", dataAddBeforeHandler: { handlers: [], handler(items) { for(let handlerFun of this.handlers) { items = handlerFun(items); } return items; } }, // 存储着text转pinyin的历史 registry.searchData.TEXT_PINYIN_KEY TEXT_PINYIN_KEY: "TEXT_PINYIN_MAP", // 默认数据不应初始化,不然太占内存了,只用调用了toPinyin才会初始化 getGlobalTextPinyinMap() getGlobalTextPinyinMap: (function() { let textPinyinMap = null; return function (){ if(textPinyinMap != null) return textPinyinMap; return (textPinyinMap = cache.jGet("TEXT_PINYIN_MAP")??{}); } })(), isSearchPro: false } } let dao = {} // 【函数库】 // 加载样式 function loadStyleString(css) { var style = document.createElement("style"); style.type = "text/css"; try { style.appendChild(document.createTextNode(css)); } catch(ex) { style.styleSheet.cssText = css; } var head = document.getElementsByTagName('head')[0]; head.appendChild(style); } // 异步函数 function asyncFun(fun,time = 20) { setTimeout(()=>{ fun(); },time) } // 提取URL根域名 function getUrlRoot(url) { if(! (typeof url == "string" || url.length >= 3)) return url; // 可处理 let arr = url.split("://"); url = (arr.length == 1)?arr[0]:arr[1]; return url.substr(0,url.indexOf("/")); } // 往字符原型中添加新的方法 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; }; // 向原型中添加方法:文字转拼音 String.prototype.toPinyin = function (isOnlyFomCacheFind= false,options = { toneType: 'none', type: 'array' }) { let textPinyinMap = registry.searchData.getGlobalTextPinyinMap(); // 如果字典中没有,再转拼音 if(textPinyinMap[this] != null) { // console.log("命中了") return textPinyinMap[this]; } // 如果 isOnlyFomCacheFind = true,那返回原数据 if(isOnlyFomCacheFind) return null; // console.log("字典没有,将进行转拼音",Object.keys(textPinyinMap).length) let {pinyin} = pinyinPro; let text = this; let space = "" let spaceChar = " "; text = text.replaceAll(spaceChar,space) let pinyinArr = pinyin(text,options); // 保存到全局字典对象 ( 会话级别 ) textPinyinMap[this] = pinyinArr.join("").replaceAll(space,spaceChar).toUpperCase(); return textPinyinMap[this]; } // 加载全局样式 loadStyleString(` .searchItem { background-image: url(); background-size: 100% 100%; background-clip: content-box; background-origin: content-box; } #my_search_view { animation-duration: 1s; animation-name: my_search_view; } .resultItem { animation-duration: 0.5s; animation-name: resultItem; } @-webkit-keyframes my_search_view{ 0%{width: 0px;} 50%{width: 60%;} 100%{width: 80%;} } @-webkit-keyframes resultItem{ 0%{opacity: 0;} 40%{opacity: 0.6;} 50%{opacity: 0.7;} 60%{opacity: 0.8;} 100%{opacity: 1;} } `) //防抖函数模板 function debounce(fun, wait) { let timer = null; return function (...args) { // 清除原来的定时器 if (timer) clearTimeout(timer) // 开启一个新的定时器 timer = setTimeout(() => { fun.apply(this, args) }, wait) } } // 判断是否为指定指令 function isInstructions(val,cmd) { return val == ":"+cmd; } // 向数据项中加入拼音项 如:title加了titlePinyin, desc加了descPinyin function genDataItemPinyin(threadHandleItems,data){ let textPinyinMap = registry.searchData.getGlobalTextPinyinMap(); // console.log("分配的预热item:",threadHandleItems) asyncFun(()=>{ if(threadHandleItems.length < 1) return; for(let item of threadHandleItems) { // 查看字典是否存在,只有没有预热过再预热 if( textPinyinMap[threadHandleItems.title] != null ) continue; item.title.toPinyin(); item.desc.toPinyin(); } // 持久化-textPinyinMap字典 (这里需要判断是否值已经被初始化) if(textPinyinMap != null ) { cache.jSet(registry.searchData.TEXT_PINYIN_KEY,textPinyinMap); } }); } // 当页面加载完成时触发 const refresh = debounce(()=>{ let threadHandleItemSize = 100; let threadHandleItems = []; let currentSize = 0; let data = registry.searchData.data; // 如果最后一个元素或第一个元素已经解析出拼音,就不要进行下面的操作了 if(data[0].titlePinyin != null && data[data.length-1].titlePinyin != null) return; for(let item of data) { // 加入处理容器中 threadHandleItems.push(item); currentSize++; // 判断是否已满 if(currentSize >= threadHandleItemSize || data[data.length-1] == item ) { // 已满-去操作 genDataItemPinyin(threadHandleItems,data); // 重置数据 currentSize = 0; threadHandleItems = []; } } }, 2000) registry.searchData.dataChange.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.replaceAll("alt", "").replaceAll("ctrl", "").replace(/\++/gm,"").trim(); // 判断 Ctrl+S if (event.ctrlKey != isCtrl || event.altKey != isAlt) return; if (!isKeyCode) { // 查看 lastKey == 按下的key if (lastKey.toUpperCase() == event.key.toUpperCase()) fun(); } else { // 查看 lastKey == event.keyCode if (lastKey == event.keyCode) fun(); } } // 如果使用 document.onkeydown 这种,只能有一个监听者 $(document).keyup(handle); } // 解决有些网站不加载图片 // 插入 meta 标签 /* var oMeta = document.createElement('meta'); oMeta.content = "default-src 'self' data: * 'unsafe-eval'; style-src 'self' 'unsafe-inline'; media-src * ;img-src *"; oMeta['http-equiv'] = "Content-Security-Policy"; document.getElementsByTagName('head')[0].appendChild(oMeta); */ // 【数据初始化】 // 获取存在的订阅信息 function getSubscribe() { // 查看是否有订阅信息 let subscribeKey = registry.searchData.subscribeKey; let subscribeInfo = cache.get(subscribeKey); if(subscribeInfo == null ) { // 初始化订阅信息(初次) subscribeInfo = ` `; cache.set(subscribeKey,subscribeInfo); } return subscribeInfo; } function editSubscribe(subscribe) { // 判断导入的订阅是否有效 // 获取订阅信息(得到的值肯定不会为空) let pageTextHandleChainsY = pageTextHandleChains.init(subscribe); let tisHasFetchFun = pageTextHandleChainsY.parseSingleTab("tis","fetchFun"); let tisNotFetchFun = pageTextHandleChainsY.parseSingleTabValue("tis"); let tis = [...tisHasFetchFun, ...tisNotFetchFun]; // 生成订阅信息存储 let subscribeText = "\n"; for(let aTis of tisHasFetchFun) { subscribeText += `\n` } for(let aTis of tisNotFetchFun) { subscribeText += `\n` } // 持久化 let newSubscribeInfo = subscribeText.replace(/\n+/gm,"\n\n"); cache.set(registry.searchData.subscribeKey,newSubscribeInfo); return tis.length; } // 存储订阅信息,当指定 sLineFetchFun 时,表示将解析“直接页”的配置,如果没有指定 sLineFetchFun 时,只解析内容 // 在提取函数中 \n 要改写为 \\n let dataSources = getSubscribe()+ ` function(pageText) { let type = "sketch"; // url sketch let lines = pageText.split("\\n"); let search_data_lines = []; // 扫描的搜索数据 {},{} let current_build_search_item = {}; let current_build_search_item_resource = ""; let point = 0; // 指的是上面的 current_build_search_item let default_desc = "--无描述--" function getTitleLineData(titleLine) { const regex = /# ([^()()]+)[((]?([^()()]*)[^))]?/; let matchData = regex.exec(titleLine) return { title: matchData[1], desc: ((matchData[2]==null || matchData[2] == "")?default_desc:matchData[2]) } } for (let i = 0; i < lines.length; i++) { let line = lines[i]; if(line.indexOf("# ") == 0) { // 当前新的开始工作 point++; // 创建新的搜索项目容器 current_build_search_item = {...getTitleLineData(line)} // 重置resource current_build_search_item_resource = ""; continue; } // 如果是刚开始,没有标题的内容行,跳过 if(point == 0) continue; // 向当前搜索项目容器追加当前行 current_build_search_item_resource += (line+"\\n"); // 如果是最后一行,打包 let nextLine = lines[i+1]; if(i === lines.length-1 || ( nextLine != null && nextLine.indexOf("#") == 0 )) { // 加入resource,最后一项 current_build_search_item.resource = current_build_search_item_resource; // 打包装箱 search_data_lines.push(current_build_search_item); } } // 添加种类 for(let line of search_data_lines) { line.type = type; } return search_data_lines; } 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; } `; // github CDN加速包装器 function cdnPack(githubResourceUrl) { let githubUrlFlag = "raw.githubusercontent.com"; // 如何不满足github url ,不加速 if(githubResourceUrl.indexOf(githubUrlFlag) < 0) return githubResourceUrl; return "https://proxy.zyun.vip/"+githubResourceUrl; } // 模块四:初始化数据源 // 使用责任链模式——对pageText进行操作的工具 const pageTextHandleChains = { pageText: "", setPageText(newPageText) { this.pageText = newPageText; }, getPageText() { return this.pageText; }, init(newPageText = "") { // 深拷贝一份实例 let wo = {...this}; // 初始化 wo.setPageText(newPageText); return wo; }, // 解析双标签-获取指定标签下指定属性下的值 parseDoubleTab(tabName,attrName) { // 返回指定标签下指定属性下的值 const regex = RegExp(`<\\s*${tabName}[^<>]*\\s*${attrName}="([^<>]*)"\\s*>([\\s\\S]*?)<\/\\s*${tabName}\\s*>`,"gm"); let m; let tabNameArr = []; let copyPageText = this.pageText; // 注意下面的 copyPageText 不能改变 while ((m = regex.exec(copyPageText)) !== null) { // 这对于避免零宽度匹配的无限循环是必要的 if (m.index === regex.lastIndex) { regex.lastIndex++; } tabNameArr.push({ attrValue: m[1], tabValue: m[2] }) const newPageText =this.pageText.replace(m[0], ""); this.pageText = newPageText; } return tabNameArr; }, // 解析双标签-只获取值 parseDoubleTabValue(tabName) { // 返回指定标签下指定属性下的值 const regex = RegExp(`<\\s*${tabName}[^<>]*\\s*>([\\s\\S]*?)<\/\\s*${tabName}\\s*>`,"gm"); let m; let tabNameArr = []; let copyPageText = this.pageText; while ((m = regex.exec(copyPageText)) !== null) { // 这对于避免零宽度匹配的无限循环是必要的 if (m.index === regex.lastIndex) { regex.lastIndex++; } tabNameArr.push({ tabValue: m[1] }) const newPageText =this.pageText.replace(m[0], ""); this.pageText = newPageText; } return tabNameArr; }, // 获取指定单标签指定属性与标签值(标签::值) parseSingleTab(tabName,attrName) { // 返回指定标签下指定属性下的值 const regex = RegExp(`<${tabName}::([^\\s<>]*)\\s*${attrName}="([^"<>]*)"\\s*\/>`,"gm"); let m; let tabNameArr = [] let copyPageText = this.pageText; while ((m = regex.exec(copyPageText)) !== null) { // 这对于避免零宽度匹配的无限循环是必要的 if (m.index === regex.lastIndex) { regex.lastIndex++; } tabNameArr.push({ tabValue: m[1], attrValue: m[2] }) const newPageText =this.pageText.replace(m[0], ""); this.pageText = newPageText; } return tabNameArr; }, parseSingleTabValue(tabName) { // 返回指定标签下指定属性下的值 const regex = RegExp(`<${tabName}::([^\\s<>]*)[^<>]*\/>`,"gm"); let m; let tabNameArr = [] let copyPageText = this.pageText; while ((m = regex.exec(copyPageText)) !== null) { // 这对于避免零宽度匹配的无限循环是必要的 if (m.index === regex.lastIndex) { regex.lastIndex++; } tabNameArr.push({ tabValue: m[1] }) const newPageText =this.pageText.replace(m[0], ""); this.pageText = newPageText; } return tabNameArr; }, // 清除指定单双标签 cleanTabByTabName(tabName) { const regex = RegExp(`<\\s*${tabName}[^<>]*>([^<>]*)(<\/[^<>]*>)*`,"gm"); // 替换的内容 const subst = ``; // 被替换的值将包含在结果变量中 const cleanedText = this.pageText.replace(regex, subst); this.pageText = cleanedText; } } // 从 订阅信息(或页) 中解析出配置(json) function getConfigFromDataSource(pageText) { let config = { // {url、fetchFun属性} tis: [], // {name与fetchFun属性} fetchFuns: [] } // 从config中放在返回对象中 let pageTextHandleChainsX = pageTextHandleChains.init(pageText); let fetchFunTabDatas = pageTextHandleChainsX.parseDoubleTab("fetchFun","name"); for(let fetchFunTabData of fetchFunTabDatas) { config.fetchFuns.push( { name:fetchFunTabData.attrValue,fetchFun:fetchFunTabData.tabValue } ) } // 获取tis let tisHasFetchFun = pageTextHandleChainsX.parseSingleTab("tis","fetchFun"); let tisNotFetchFun = pageTextHandleChainsX.parseSingleTabValue("tis"); let tisArr = [...tisHasFetchFun, ...tisNotFetchFun] for(let tis of tisArr) { config.tis.push( { url:tis.tabValue, fetchFun:tis.attrValue } ) } return config; } // 将url转为文本(url请求得到的就是文本),当下面的dataSourceUrl不是http的url时,就会直接返回,不作请求 function urlToText(dataSourceUrl) { // dataSourceUrl 转text return new Promise(function (resolve, reject) { if((dataSourceUrl.trim().indexOf("http") != 0 ) ) return resolve(dataSourceUrl) ; $.ajax({ url: cdnPack(dataSourceUrl+"?time="+new Date().getTime()), success: function (result) { resolve(result) } }); }); } // 下面的 dataSourceHandle 函数 let globalFetchFun = []; // tis处理队列 let waitQueue = []; registry.searchData.dataAddBeforeHandler.handlers.push(function (items) { for(let searchItem of items) { let url = searchItem.resource; let isSearchableItem = /\[\[[^\[\]]+keyword[^\[\]]+\]\]/.test(url); if(isSearchableItem) searchItem.title = "[可搜索]"+searchItem.title; } return items; }) // 将解析出来的部分数据push 到 registry.searchData.data的操作队列 (push到全局不能并发,所以这里必须是单线程操作) let searchDataController = { // 加入到全局的等待队列 queueData: [], // 是否空闲 isIdle: true, // 是否第一次 pushToGlobal isFirsPush: true, // 给外面触发,加入到队列中 pushToGlobal:function(newItems) { if(this.isFirsPush) { this.isFirsPush = false; // 清空全局数据容器的数据 registry.searchData.data = []; } // 在添加前,进行额外处理添加,如给有”{keyword}“的url搜索项添加”可搜索“标签 newItems = registry.searchData.dataAddBeforeHandler.handler(newItems); this.queueData.push(...newItems); // 如果当前不是空闲的,其它线程当会处理刚才push的数据到全局中 if(!this.isIdle) return; // 设置当前为工作模式 this.isIdle = false; // 处理队列中的数据 let newItem = null; while((newItem = this.queueData.pop() ) != null) { // 下一个编号索引号 let nextIndex = registry.searchData.data.length; newItem.index = nextIndex; registry.searchData.data[nextIndex] = newItem; } // 设置当前空闲 this.isIdle = true; // 更新视图显示条数 $("#search_input").attr("placeholder",registry.searchData.getDataLength("UPDATE")); }, } function dataSourceHandle(resourcePageUrl,tisTabFetchFunName) { urlToText(resourcePageUrl).then(text => { if(tisTabFetchFunName == null) { // --> 是配置 <-- let data = [] // 解析配置,是一个json let config = getConfigFromDataSource(text); console.log("解析的配置:",config) // 将FetchFun放到全局解析器中 globalFetchFun.push(...config.fetchFuns); // 将tis放到处理队列中 waitQueue.push(...config.tis); let tis = null; while((tis = waitQueue.pop()) != undefined) { // tis有两个url,第二是fetchFun dataSourceHandle(tis.url,tis.fetchFun); } // 清理内容 pageTextHandleChains.setPageText(""); }else { // --> 是内容 <-- // 解析内容 if(tisTabFetchFunName === "") return; let fetchFunStr = getFetchFunGetByName(tisTabFetchFunName); let search_data_line =(new Function('text', "return ( " + fetchFunStr + " )(`"+text.replace(/[`]/gm,"<反引号>")+"`)"))(); // 将之前修改为 改为真正的换行符 \n let replaceBefore = "<反引号>"; let replaceAfter = "`"; // 处理并push到全局数据容器中 for(let item of search_data_line) { item.title = item.title.replaceAll(replaceAfter,replaceBefore); item.desc = item.desc.replaceAll(replaceAfter,replaceBefore); item.resource = item.resource.replaceAll(replaceAfter,replaceBefore); } // 加入到push到全局的搜索数据队列中,等待加入到全局数据容器中 searchDataController.pushToGlobal(search_data_line); // 触发搜索数据改变事件(做缓存等操作,观察者模式) for(let fun of registry.searchData.dataChange) { fun(search_data_line); } } }) } // 根据fetchFun名返回字符串函数 function getFetchFunGetByName(fetchFunName) { for(let fetchFunData of globalFetchFun) { if(fetchFunData.name == fetchFunName) { return fetchFunData.fetchFun; } } } // 缓存数据 function cacheSearchData(newSearchData) { console.log("触发了缓存,当前数据",registry.searchData.data) // 当有数据加入到全局数据容器时,会触发缓存,当前函数会执行 let SEARCH_DATA_KEY = registry.searchData.SEARCH_DATA_KEY; cache.remove(SEARCH_DATA_KEY) cache.set(SEARCH_DATA_KEY,{ data: registry.searchData.data, expire: new Date().getTime() + (1000*60*60*1) // 一个小时 }) // 加载一下图片,让服务器的redis缓存图标 loadWebFavicon(newSearchData); } // 加载一下网站图标,让远程redis有记录,下次会很快 async function loadWebFavicon(searchItems) { let CACHE_FLAG = "loadHistory"; let loadHistory = cache.get(CACHE_FLAG); if(loadHistory == null) loadHistory = []; // 创建容器 let imgBox=document.createElement("div"); imgBox.style="display:none"; document.body.appendChild(imgBox); for(let item of searchItems) { let resource = item.resource.trim(); if(resource.toUpperCase().indexOf("HTTP") == 0) { // 如果历史中有,跳过此项后续操作 if(loadHistory.includes(resource)) continue; // 记录一下,下次不会再进行此操作,给服务减压 loadHistory.push(resource); // 装箱 var img=document.createElement("img"); img.src = registry.searchData.getFaviconAPI+getUrlRoot(resource); imgBox.appendChild(img); } if(item === searchItems[searchItems.length-1] ) { // 这是最后一个,去掉痕迹 setTimeout(function() { // 将记录持久化 cache.set(CACHE_FLAG,loadHistory); imgBox.remove(); },500) } } } // 检查是否已经执行初始化 function checkIsInitializedAndSetInitialized(secondTime) { let key = "DATA_INIT"; let value = cache.cookieGet(key); if(value != null && value != "") return true; cache.cookieSet(key,key,1000*secondTime); return false; } // 【数据初始化主函数】 let isInitialized = false; function dataInitFun() { // 检查是否已经执行初始化 if(isInitialized) return; // 设置为已初始化,保障只初始化一次,如果不能保障,同时初始化时,出现重复数据 isInitialized = true; // 从缓存中获取数据,判断是否还有效 const SEARCH_DATA_KEY = registry.searchData.SEARCH_DATA_KEY; // cache.remove(SEARCH_DATA_KEY) let dataBox = cache.get(registry.searchData.SEARCH_DATA_KEY); if(dataBox != null) { // 只要数据不为空,不管是否过期,先用着,直接将之前缓存的数据放在全局搜索数据容器中 registry.searchData.data = dataBox.data // 缓存信息不为空,深入判断是否使用缓存的数据 let dataExpireTime = dataBox.expire; let currentTime = new Date().getTime(); // console.log("缓存的数据:",dataBox.data) // 数据多大时,才开启缓存 const TRIGGER_CACHE_DATA_LENGTH = 300; // 判断是否有效,有效的话放到全局容器中 let isValid = (dataExpireTime != null && dataExpireTime > currentTime && dataBox.data != null && dataBox.data.length > 0); // 如果网站比较特殊,忽略数据过期时间 if(!isValid && window.location.host.toUpperCase().indexOf("GITHUB.COM") >= 0) { isValid = true; } // 如果数据过期,或数据量不满足缓存大小,会去请求数据 if(isValid && dataBox.data.length >= TRIGGER_CACHE_DATA_LENGTH ) { console.log("我的搜索:本次从缓冲中获取, 数据有效期还有"+parseInt((dataExpireTime - currentTime)/1000/60)+"分钟!", dataBox.data) return }; } // 内部将使用递归,解析出信息 dataSourceHandle(dataSources,null); // 监听数据改变 registry.searchData.dataChange.push(cacheSearchData) } // 判断是否要直接执行初始化函数-如果没有数据,这里要直接执行 (function() { let SEARCH_DATA_KEY = registry.searchData.SEARCH_DATA_KEY; if(cache.get(SEARCH_DATA_KEY) == null) { // 执行初始化函数 dataInitFun(); } })(); // 模块二 registry.view.viewVisibilityController = (function () { // 整个视图对象 let viewDocument = null; let searchInputDocument = null; let matchItems = null; let searchBox = null; let isInitializedView = false; let viewName = "my_search_view" let controlButton = null; let textShow = null; let matchResult = null; let initView = function () { // 初始化视图 let view = document.createElement("div") view.innerHTML = (`
`) // 设置样式 view.style = ` position: fixed;left: 25%;right: 25%;top:50px; border:2px solid #cecece;z-index:10000; background: #ffffff; overflow: hidden; `; // 挂载到文档中 document.body.appendChild(view) // 整个视图对象放在组件全局中/注册表中 registry.view.viewDocument = viewDocument = view; // 搜索框对象 searchInputDocument = $("#search_input") matchItems = $("#matchItems"); searchBox = $("#searchBox") controlButton = $("#controlButton") textShow = $("#text_show") matchResult = $("#matchResult"); searchBox.css({ "height": "45px", "background": "#ffffff", "padding": "0px", "box-sizing": " border-box", "z-index": "10001", "position":"relative", "display": "flex", "justify-content": "space-between", "align-items": "center", "flex-wrap": "nowrap" }) searchInputDocument.css({ "width": "100%", "height": "100%", "border": "none", "outline": "none", "font-size": "15px", "background": "#fff", "padding": "0px 10px", "box-sizing": " border-box", "color":"rgba(0,0,0,.87)", "font-weight":"400" }) $("#matchResult").css({ "display":"none" }) $("#matchResult > ol").css({ "margin": "0px", "padding": "0px 15px 5px" }) controlButton.css({ "position": "absolute", "font-size":"12px", "right": "5px", "padding":"3px 12px", "border":"none", "display":"none", // 默认隐藏,由函数控制 "color": "#C0C2C8", "border":"1px solid #f5f5f5", "background": "#ffffff", "margin-right": "7px", "font-weight": "700", "cursor": "pointer", "box-shadow": "0px 0px 2px", "border-radius": "11px" }) textShow.css({ "display":"none", "width":"100%", "box-sizing": "border-box", "padding": "5px 10px 7px", "font-size": "15px", "line-height":"25px", "max-height":"450px", "overflow": "auto", "text-align":"left", "color":"#000000" }) // 初始化搜索数据 dataInitFun(); // 在搜索的结果集中上下选择移动然后回车(相当点击) searchInputDocument.keyup(function(event){ let keyword = $(event.target).val().trim(); // 当不为空时,放到全局keyword中 if(keyword != "" && keyword != null) { registry.searchData.keyword = event.target.value; } // 处理keyword中的":"字符 if(keyword.endsWith("::") || keyword.endsWith("::")) { keyword = keyword.replace(/::|::/,registry.searchData.searchBoundary).replace(/\s+/," "); // 每次要形成一个" : "的时候去掉重复的" : : " -> " : " keyword = keyword.replace(/((\s{1,2}:)+ )/,registry.searchData.searchBoundary); $(event.target).val(keyword.toUpperCase()); // 设置当前为“搜索PRO”模式 registry.searchData.isSearchPro = true; // registry.view.showControlButton("PRO模式") } // 判断是否要退出搜索模式 if($(event.target).val().indexOf(registry.searchData.searchBoundary) == -1) { // registry.view.hideControlButton(); registry.searchData.isSearchPro = false; } }); searchInputDocument.keydown(function(event){ let e = event || window.event || arguments.callee.caller.arguments[0]; // 判断一个输入框的东西,如果如果按下的是删除,判断一下是不是"搜索模式" let keyword = $(event.target).val().trim(); if(e.key == "Backspace" ) { // 按的是删除键 if(event.target.value.endsWith(registry.searchData.searchBoundary)) { // 取消默认事件 e.preventDefault(); return; } } if(e && e.keyCode!=38 && e.keyCode!=40 && e.keyCode!=13) return; if(e && e.keyCode==38){ // 上 registry.searchData.pos --; } if(e && e.keyCode==40){ //下 registry.searchData.pos ++; } // 如果是回车 && registry.searchData.pos == 0 时,设置 registry.searchData.pos = 1 (这样是为了搜索后回车相当于点击第一个) if(e && e.keyCode==13 && registry.searchData.pos == 0){ // 回车选择的元素 registry.searchData.pos = 1; } // 当指针位置越出时,位置重定向 if(registry.searchData.pos < 1 || registry.searchData.pos > registry.searchData.searchData.length ) { if(registry.searchData.pos < 1) { // 回到最后一个 registry.searchData.pos = registry.searchData.searchData.length; }else { // 回到第一个 registry.searchData.pos = 1; } } // 设置显示样式 let activeItem = $($("#matchItems > li")[registry.searchData.pos-1]); // if(activeItem == null) return; // 设置活跃背景颜色 let activeBackgroundColor = "#dee2e6"; //let activeFontColor = "rgb(26, 13, 171)"; // 如果是搜索项,可用别的颜色 //if(activeItem.find("a").attr("href").indexOf("keyword") != -1) activeFontColor = "rgb(251,182,54)" activeItem.css({ "background":activeBackgroundColor }) /*activeItem.find("a").css({ "color":activeFontColor })*/ // 设置其它子元素背景为默认统一背景 activeItem.siblings().css({ "background":"#fff" }) if(e && e.keyCode==13 && activeItem.find("a").length > 0){ // 回车 // 点击当前活跃的项,点击 activeItem.find("a")[0].click(); } // 取消冒泡 e.stopPropagation(); // 取消默认事件 e.preventDefault(); }) // 将输入框的控制按钮设置可见性函数公开放注册表中 registry.view.setButtonVisibility = function (buttonVisibility = false) { // registry.view.setButtonVisibility controlButton.css({ "display": buttonVisibility?"block":"none" }) } function searchUnitHandler(beforeData = [],keyword = "") { // 如果没有搜索内容,返回空数据 keyword = keyword.trim().toUpperCase(); if(keyword == "" || registry.searchData.data.length == 0 ) return []; // 切割搜索内容以空格隔开,得到多个 keyword let searchUnits = keyword.split(/\s+/); // 弹出一个 keyword keyword = searchUnits.pop(); // 本次搜索的总数据容器 let searchResultData = []; let searchLevelData = [ [],[],[] // 分别是匹配标题/desc/url 的结果 ] // 数据出来的总数据 //let searchData = [] // 前置处理函数,这里使用观察者模式 // searchPreFun(keyword); // 搜索操作 let pinyinKeyword = keyword.toPinyin(); for (let dataItem of beforeData) { // 如果已达到搜索要显示的条数,则不再搜索 && 已经是本次最后一次过滤了 => 就不要扫描全部数据了,只搜出15条即可 let currentMeetConditionItemSize = searchLevelData[0].length + searchLevelData[1].length + searchLevelData[2].length; if(currentMeetConditionItemSize >= registry.searchData.showSize && searchUnits.length == 0 ) break; // 将数据放在指定搜索层级数据上 if ( (( (dataItem.title.toPinyin(true)??"").indexOf(pinyinKeyword) >= 0 || dataItem.title.indexOf(keyword) >= 0 ) && searchLevelData[0].push(dataItem) ) || (( (dataItem.desc.toPinyin(true)??"").indexOf(pinyinKeyword) >= 0 || dataItem.desc.toUpperCase().indexOf(keyword) >= 0) && searchLevelData[1].push(dataItem) ) || (dataItem.resource.toUpperCase().indexOf(keyword) >= 0 && searchLevelData[2].push(dataItem) ) ) { // 向满足条件的数据对象添加在总数据中的索引 } } // 将搜索项放到上面 function searchItemToTop(items) { // 只有是搜索PRO模式 才干扰排序 if(!registry.searchData.isSearchPro) return; let searchableItem = []; let noSearchableItem = []; let currentTop = 0; for(let i = 0; i < items.length; i++) { let item = items[i]; if(item.title.trim() == "可搜索" ) { // 替换 let tmp = items[currentTop++]; items[currentTop] = item; items[i] = tmp; } } } // 将上面层级数据放在总容器中 searchResultData.push(...searchLevelData[0]); searchResultData.push(...searchLevelData[1]); searchItemToTop(searchResultData); // 搜索PRO模式时会干扰排序 searchResultData.push(...searchLevelData[2]); if(searchUnits.length > 0 && searchUnits[searchUnits.length-1].trim() != ":") { // 递归搜索 searchResultData = searchUnitHandler(searchResultData,searchUnits.join(" ")); } return searchResultData; } // 给输入框加事件 // 执行 debounce 函数返回新函数 let handler = function (e) { let key = e.target.value.trim().split(/\s+/).reverse().join(" "); // 过滤 // 数据出来的总数据 let searchData = [] // 递归搜索,根据空字符切换出来的多个keyword let searchResultData = searchUnitHandler(registry.searchData.data,key) // console.log("搜索总数据:",searchResultData) // 放到视图上 // 置空内容 matchItems.html("") // 最多显示条数 let show_item_number = registry.searchData.showSize ; function getFlag(searchResultItem) { let resource = searchResultItem.resource.trim(); let sketchFavicon = ""; let loadErrorFlagIcon = ""; if(!resource.toUpperCase().indexOf("HTTP") == 0) return searchResultItem.type=="sketch"?``:``; function loaded() { alert("loaded!") } return `` // return `` } /* $("#matchItems").on("load","li img", function (){ alert("加载完成!") })*/ // 标题flag颜色选择器 function titleFlagColorMatchHandler(flagValue) { let vcObj = { "系统项":"rgb(0,210,13)", "非最佳":"#fbbc05", "推荐":"#ea4335", "装机必备":"#9933E5", "好物":"rgb(247,61,3)", "Adults only": "rgb(244,201,13)", "可搜索":"#4c89fb" } let resultFlagColor = "#5eb95e"; Object.getOwnPropertyNames(vcObj).forEach(function(key){ if(key == flagValue) { resultFlagColor = vcObj[key]; } }); return resultFlagColor; } // 标题内容处理程序 function titleFlagHandler(title) { if(!(/[\[]?/.test(title) && /[\]]?/.test(title))) return -1; // 格式是:[flag]title(desc):resource 这种的 const regex = /\[\s*([^\]]+)\s*\]\s*(.+)\s*/; let m; if ((m = regex.exec(title)) !== null) { if(m.length != 3) return title; // 正确提取 let style = ` background: ${titleFlagColorMatchHandler(m[1])}; color: #fff; font-size: 10px; padding: 3px 6px; border-radius: 5px; font-weight: 700; box-sizing: border-box; `; return `${m[1]} ${m[2]}`; } return -1; } // 标题前面带“#”的titleHandler function title井Handler(title) { // 去掉flag title = title.replace(/\[.*\]/,"").trim(); if(title.indexOf("#") == 0) { let style = `text-decoration:line-through;color:#a8a8a8;`; return `${title.replace("#","")}`; } return -1; } function titleHandler(title) { let titleHandlerFuns = registry.view.titleHandlerFuns; for(let titleHandlerFun of titleHandlerFuns) { let result = titleHandlerFun(title.trim()); if(result != -1) return result; } return title; } // 添加标题处理器 title井Handler (优化级较高) registry.view.titleHandlerFuns.push(title井Handler); // 添加标题处理器 titleFlagHandler registry.view.titleHandlerFuns.push(titleFlagHandler); for(let searchResultItem of searchResultData ) { // 限制条数 if(show_item_number-- <= 0) { break; } // 将数据放入局部容器中 searchData.push(searchResultItem) let isSketch = searchResultItem.resource.trim().toUpperCase().indexOf("HTTP") != 0; // 将符合的数据装载到视图 // let item = `
  • ${getFlag(searchResultItem)} ${titleHandler(searchResultItem.title)} (${searchResultItem.desc})
  • ` matchItems.html(matchItems.html() + item) } // 给刚才添加的img添加加载完成事件,去除加载背景 for(let imgObj of $("#matchItems").find('img')) { imgObj.onload = function(e) { $(e.target).css({ "background": "rgba(0,0,0,0)" }) } } // 隐藏文本显示视图 textShow.css({ "display":"none" }) // 让搜索结果显示 let matchResultDisplay = "block"; if(searchResultData.length < 1) matchResultDisplay="none"; matchResult.css({ "display":matchResultDisplay, "overflow":"hidden" }) // 将搜索的数据放入全局容器中 registry.searchData.searchData = searchData; // 指令归位(置零) registry.searchData.pos = 0; // 设置li样式 $("#matchResult li").css({ "line-height": "30px", "height": "30px", "color": "#0088cc", "list-style": "none", //decimal "width":"100%", "margin":"0px", "display":"flex", "justify-content":"left", "align-items":"center", "padding":"0px", "margin":"0px" }) $("#matchResult li>a").css({ "display":"inline-block", "font-size":"15px", "color": "#1a0dab", "text-decoration":"none", "text-align":"left", "overflow":"hidden", //超出的文本隐藏 "text-overflow":"ellipsis", //溢出用省略号显示 "white-space":"nowrap", //溢出不换行 "cursor":"pointer", "font-weight":"400" }) $("#matchResult .desc").css({ "color":"#4d5156" }) $("#matchResult img").css({ "display": "inline-block", "width": "20px", "height":"20px", "margin":"0px 7px 0px 5px", "box-shadow": "0 0 2px rgba(0,0,0,0.5)", "border-radius": "30%", "box-sizing": " border-box", "padding":"2px", "flex-shrink":"0" // 当容量不够时,不压缩图片的大小 }) } function sketchResourceParaphrase(txtStr = "") { // 3、特殊字符转义 txtStr = txtStr.replace(//gm,">").replace(/"/gm,""").replace(/'/gm,"'"); // 4、为简讯内容的url添加可链接 const regex = /(https?:\/\/[^\s()()\[\] `]+)/gm; const subst = `$1`; // 被替换的值将包含在结果变量中 txtStr = txtStr.replace(regex, subst); txtStr = txtStr.replace(/\n/gm,"
    "); // 5、转义后特殊再转义 txtStr = txtStr.replace(/<\s*br\s*\/\s*>/gm,"
    ") return txtStr; } $("#matchItems").on("click","li > a",function(e) { // 设置为阅读模式 // $("#search_input").val(":read"); // 获取当前结果在搜索数组中的索引 let dataIndex = parseInt($(e.target).attr("index")); let itemData = registry.searchData.data[dataIndex]; // 如果是简述搜索信息,那就取消a标签的默认跳转事件 if(itemData.resource.trim().toUpperCase().indexOf("HTTP") !== 0) { // 取消默认事件 e.preventDefault(); matchResult.css({ "display": "none" }) textShow.css({ "display":"block" }) textShow.html("标题:"+itemData.title+"
    "+ "描述:"+itemData.desc+"
    "+"简述内容:
    "+sketchResourceParaphrase(itemData.resource)) return; } // 取消冒泡 window.event? window.event.cancelBubble = true : e.stopPropagation(); // 隐藏视图 registry.view.viewVisibilityController(false) // 解析URL(主要将keyword补充上去) let targetObj = e.target; const initUrl = $(targetObj).attr("href"); // 不作改变的URL let url = initUrl; // 进行修改,形成要跳转的真正url let temNum = url.matchFetch(/\[\[[^\[\]]*\]\]/gm, function (matchStr,index) { // temNum是url中有几个 "[[...]]", 得到后,就已经得到解析了 let templateStr = matchStr; // 使用全局的keyword, 构造出真正的keyword let keyword = registry.searchData.keyword.split(":").reverse(); keyword.pop(); keyword = keyword.reverse().join(":").trim(); let parseAfterStr = matchStr.replace(/{keyword}/g,keyword).replace(/\[\[+|\]\]+/g,""); url = url.replace(templateStr,parseAfterStr); }); // 如果搜索的真正keyword为空字符串,则去掉模板跳转 if( registry.searchData.keyword.split(registry.searchData.searchBoundary).length < 2 || registry.searchData.keyword.split(registry.searchData.searchBoundary)[1].trim() == "" ) { url = initUrl.replace(/\[\[[^\[\]]*\]\]/gm,""); } // 如果是URL存在模板 if(temNum > 0 ) { window.open(url); // 取消默认事件 e.preventDefault(); return; } // 否则是URL跳转 }) const refresh = debounce(handler, 460) // 第一次触发 scroll 执行一次 fn,后续只有在停止滑动 1 秒后才执行函数 fn searchBox.on('input', refresh) // 初始化后将isInitializedView变量设置为true isInitializedView = true; } let hideView = function () { // 隐藏视图 // 如果视图还没有初始化,直接退出 if (!isInitializedView) return; // 如果正在查看查看“简讯”,先退出简讯 if($("#text_show").css("display")=="block") { // 让简讯隐藏 $("#text_show").css({"display":"none"}) // 让搜索结果显示 $("#matchResult").css({ "display":"block", "overflow": "hidden", }) return; } // 让输入框中的“控制按钮”隐藏 registry.view.hideControlButton(); // 让视图隐藏 viewDocument.style.display = "none"; // 将输入框内容置空,在置空前将值备份,好让未好得及的操作它 searchInputDocument.val("") // 将之前搜索结果置空 matchItems.html("") // 隐藏文本显示视图 textShow.css({ "display":"none" }) // 让搜索结果显示 matchResult.css({ "display":"none" }) } let showView = function () { // 让视图可见 viewDocument.style.display = "block"; //聚焦 searchInputDocument.focus() // 当输入框失去焦点时,隐藏视图 searchInputDocument.blur(function() { // 判断输入框的内容是不是":debug"或是否正处于阅读模式,如果是,不隐藏 if(isInstructions(searchInputDocument.val(),"debug") || isInstructions(searchInputDocument.val(),"read")) return; // 当前视图是否在展示数据,如搜索结果,简述内容?如果在展示不隐藏 let isNotExhibition = (($("#matchResult").css("display") == "none" || $("#matchItems > li").length == 0 ) && ($("#text_show").css("display") == "none" || $("#text_show").text().trim() == "") ); if(!isNotExhibition) return; setTimeout(()=>{registry.view.viewVisibilityController(false)},200) }); } // 返回给外界控制视图显示与隐藏 return function (isSetViewVisibility) { if (isSetViewVisibility) { // 让视图可见 >>> // 如果还没初始化先初始化 // 初始化数据 initData(); if (!isInitializedView) { // 初始化视图 initView(); // 初始化数据 // initData(); } // 让视图可见 showView(); } else { // 隐藏视图 >>> if (isInitializedView) hideView(); } } })(); // 触发策略——快捷键 let useKeyTrigger = function (viewVisibilityController) { // 将视图与触发策略绑定 triggerAndEvent("ctrl+alt+s", function () { // 让视图可见 viewVisibilityController(true); }) triggerAndEvent("Escape", function () { // 让视图不可见 viewVisibilityController(false); }) } // 触发策略组 let trigger_group = [useKeyTrigger]; // 初始化入选的触发策略 (function () { for (let trigger of trigger_group) { trigger(registry.view.viewVisibilityController); } })(); // 打开视图进行配置 // 显示配置视图 // 是否显示进度 - 进度控制 GM_registerMenuCommand("订阅管理",function() { showConfigView(); }); GM_registerMenuCommand("清理缓存",function() { cache.remove(registry.searchData.SEARCH_DATA_KEY); }); // 显示配置规则视图 function showConfigView() { if($("#subscribe_save")[0] != null) return; // 显示视图 var configViewContainer = document.createElement("div"); configViewContainer.style=` width:450px; background:pink; position: fixed;right: 0px; top: 0px; z-index:10000; padding: 20px; border-radius: 14px; ` configViewContainer.innerHTML = `

    订阅总览:

    `; // 设置样式 document.body.appendChild(configViewContainer); document.getElementById("title").style="margin-bottom: 10px; font-size: 16px;"; document.getElementById("all_subscribe").style="width:100%;height:150px"; document.getElementById("subscribe_save").style=" margin-top: 20px; border: none; border-radius: 3px; padding: 4px 20px; cursor: pointer;"; // 回显 document.getElementById("all_subscribe").value = getSubscribe(); // 保存 document.getElementById("subscribe_save").onclick=function() { // 保存到对象 let allSubscribe = document.getElementById("all_subscribe").value; let validCount = editSubscribe(allSubscribe); // 清除视图 configViewContainer.remove(); alert("保存配置成功!有效订阅数:"+validCount); } } })();