// ==UserScript==
// @name 我的搜索
// @namespace http://tampermonkey.net/
// @version 6.9.5
// @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://unpkg.com/pinyin-pro
// @require https://cdn.jsdelivr.net/npm/showdown@1.9.0/dist/showdown.min.js
// @resource markdown-css https://sindresorhus.com/github-markdown-css/github-markdown.css
// @require https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.7.0/highlight.min.js
// @resource code-css https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.7.0/styles/default.min.css
// @require https://update.greasyfork.org/scripts/501646/1429885/string-overlap-matching-degree.js
// @grant window.onurlchange
// @grant GM_setValue
// @grant GM_xmlhttpRequest
// @grant GM_getValue
// @grant GM_addStyle
// @grant GM_getResourceText
// @grant GM_getResourceURL
// @grant GM_deleteValue
// @grant GM_registerMenuCommand
// @grant GM_info
// ==/UserScript==
// 参数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文件
GM_addStyle(GM_getResourceText("code-css"));
GM_addStyle(GM_getResourceText("markdown-css"));
// 正则捕获
function captureRegEx(regex, text) {
let m;
let result = []; // 一组一组 [[],[],...]
regex.lastIndex = 0; // 重置lastIndex
while ((m = regex.exec(text)) !== null) {
let group = [];
group.push(...m);
if(group.length != 0) result.push(group);
}
return result;
}
// 重写console.log方法
let originalLog = console.log;
console.logout = function() {
const prefix = "[我的搜索log]>>> ";
const args = [prefix].concat(Array.from(arguments));
originalLog.apply(console, args);
}
// markdown转html 转换器 【1】
// 更多配置项: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
});
// 提取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 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 = {
get(key) {
return GM_getValue(key);
},
set(key,value) {
this.remove(key);
GM_setValue(key,value);
},
jGet(key) {
let value = GM_getValue(key);
if( value == null) return value;
return JSON.parse(value);
},
jSet(key,value) {
value = JSON.stringify(value)
GM_setValue(key,value);
},
remove(key) {
GM_deleteValue(key);
},
cookieSet(cname,cvalue,exdays) {
var d = new Date();
d.setTime(d.getTime()+exdays);
var expires = "expires="+d.toGMTString();
document.cookie = cname + "=" + cvalue + "; " + expires;
},
cookieGet(cname) {
var name = cname + "=";
var ca = document.cookie.split(';');
for(var i=0; i<ca.length; i++)
{
var c = ca[i].trim();
if (c.indexOf(name)==0) return c.substring(name.length,c.length);
}
return "";
}
}
// 责任链对象工厂
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) {
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 = {
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);
ajaxOptions.contentType = "application/json; charset=utf-8";
}
$.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) {
return this.baseRequest("POST","https://api.github.com/repos/My-Search/TisHub/issues",{body},{Authorization:`Bearer ${this.token}`})
},
// 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.script_env_var = undefined;
}
],
menuActive: false,
// 视图延时隐藏时间,避免点击右边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"
})
}
},
// 搜索框logo控制
logo: {
// logo src值
originalLogoImgSrc: null,
// logo按钮是否按下状态
isLogoButtonPressedRef: ref(false),
getLogoImg: function () {
let viewDocument = registry.view.viewDocument;
if(viewDocument == null ) return null;
let currentLogoImg = $(viewDocument).find("#logoButton 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') === "block") return this.modeEnum.SHOW_ITEM_DETAIL;
if(this.element.matchResult.css('display') === "block") return this.modeEnum.SHOW_RESULT;
return this.viewDocument.style.display === "block" ? 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) {
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);
}
}
let result = await search(rawKeyword);
this.isSearching = false;
return result;
}
},
// 新数据设置的过期天数
NEW_DATA_EXPIRE_DAY_NUM:7,
// 搜索逻辑,可用来手动触发搜索
triggerSearchHandle: function (keyword){
if(keyword == null) keyword = $("#my_search_input").val()??"";
// 获取input元素
const inputEl = document.getElementById('my_search_input');
// 当视图没有初始化时调用该函数inputEl会为null
if(inputEl == null) return;
// 如果有传入搜索值,就要设置值
if(keyword != null) {
inputEl.value = keyword;
}
// 手动触发input事件
inputEl.dispatchEvent(new Event('input', { bubbles: true }));
// 维护全局搜索keyword
this.keyword = keyword;
},
// 数据改变事件
dataChangeEventListener: [],
// 缓存被删除事件
dataCacheRemoveEventListener:[],
onSearch: [],
// 新数据块处理完成事件
// 更新搜索数据的责任链
USDRC: getResponsibilityChain(),
onNewDataBlockHandleAfter: [],
// 新数据的tag
NEW_ITEMS_TAG: "[新]",
// 搜索的keyword
keyword: "",
// 持久化Key
SEARCH_DATA_KEY: "SEARCH_DATA_KEY",
SEARCH_NEW_ITEMS_KEY:"SEARCH_NEW_ITEMS_KEY",
// 搜索搜索出来的数据
searchData: [],
pos: 0,
clearUrlSearchTemplate(url) {
return url.replace(/\[\[[^\[\]]*\]\]/gm,"");
},
faviconSources: [
// "https://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") {
if(this.tmpVar != null) {
clearTimeout(this.tmpVar);
}
this.tmpVar = setTimeout(()=>{
$("#my_search_input").attr("placeholder",this.searchPlaceholder());
},duration)
let updateResult = placeholder==null?`🔁 数据库更新到 ${this.data==null?0:this.data.length}条`:placeholder;
$("#my_search_input").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: "[可搜索]"
},
script: {
// 当打开脚本项时,赋值
script_env_var: undefined, // 有哪些请看registry.script.script_env_var的赋值
tryRunTextViewHandler() {
const input = registry.view.element.input;
const rawKeyword = input.val();
if(registry.view.seeNowMode() === registry.view.modeEnum.SHOW_ITEM_DETAIL) {
// 当msg不为空发送msg消息到脚本
const subKeyword = registry.searchData.subSearch.getSubSearchKeyword()
if( subKeyword !== undefined) {
// 通知脚本回车send事件
const msg = subKeyword;
registry.script.script_env_var?.event.sendListener.forEach(listener=>listener(msg))
// 清理掉send msg内容
input.val(rawKeyword.replace(msg,""))
}
return true;
}
return false;
}
}
}
let dao = {}
// 网页脚本自动执行函数
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
if(! /^https?:\/\/.+/i.test(resource) ) return false;
return true;
}
/*cache.remove(registry.searchData.SEARCH_DATA_KEY);
cache.remove(registry.searchData.SEARCH_DATA_KEY+"2");
cache.remove(registry.searchData.SEARCH_NEW_ITEMS_KEY);
*/
// 设置远程可用Favicon源
let setFaviconSource = function () {
function startTestFaviconSources(sources,pos,setFaviconUrlTemplate) {
if(pos > sources.length - 1) return;
console.logout(`${pos}/${sources.length-1}: 正在测试 `+sources[pos])
checkUsability(sources[pos]).then(function(result) {
console.logout("使用的源:"+ sources[pos])
setFaviconUrlTemplate(result);
}).catch(function() {
startTestFaviconSources(sources,++pos,setFaviconUrlTemplate)
});
}
let cacheFaviconSourceData = cache.get(registry.searchData.CACHE_FAVICON_SOURCE_KEY);
let currentTime = new Date().getTime();
let timeout = registry.searchData.CACHE_FAVICON_SOURCE_TIMEOUT;
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: 0px;
box-sizing: border-box;
z-index: 10001;
position: relative;
display: flex;
justify-content: space-between;
align-items: center;
flex-wrap: nowrap;
}
#my_search_input {
text-align: left;
width: 100%;
height: 100%;
border: none;
outline: none;
font-size: 15px;
background: #fff;
padding: 0px 10px;
box-sizing: border-box;
color: rgba(0, 0, 0, .87);
font-weight: 400;
margin: 0px;
}
#matchResult {
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;
}
#text_show img {
width: 100%;
}
/*定义字体*/
@font-face {
font-family: 'HarmonyOS';
src: url('https://s1.hdslb.com/bfs/static/jinkela/long/font/HarmonyOS_Medium.a1.woff2');
}
#my_search_view {
font-family: 'HarmonyOS', sans-serif !important;
}
.searchItem {
background-image: url();
background-size: 100% 100%;
background-clip: content-box;
background-origin: content-box;
}
#my_search_input {
animation-duration: 1s;
animation-name: my_search_view;
}
.resultItem {
animation-duration: 0.5s;
animation-name: resultItem;
}
.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;
}
.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;
}
/*简述文本颜色为统一*/
#text_show p {
color: #202122;
}
/*自定义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;
}
/*让简述内容的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: 21%;
right:21%;
}
}
/*当视图小于等于1200px时*/
@media (max-width: 1200px) {
#my_search_box {
left: 18%;
right:18%;
}
}
/*当视图小于等于800px时*/
@media (max-width: 800px) {
#my_search_box {
left: 15%;
right:15%;
}
}
/*输入框右边按钮*/
#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;
outline: none;
}
#logoButton:active {
opacity: 0.4;
}
#logoButton img {
display: block;
width: 25px;
}
/*代码颜色*/
#text_show code,#text_show pre{
color:#5f6368;
padding: 5px;
}
/* 滚动条整体宽度 */
#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;
}
#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;
}
#matchResult .item_desc {
color: #4d5156;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
#matchResult img {
display: inline-block;
width: 24px;
height: 24px;
margin: 0 8px 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 = $("#my_search_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 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 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;
// 判断是否开始为附加内容
if(/^\s*-{3,}\s*$/gm.test(line)) {
appendTarget = "vassal"
// 分割行不添加
continue
}
// 向当前搜索项目容器追加当前行
if(appendTarget == "resource") {
current_build_search_item_resource += (line+"\\n");
}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(current_build_search_item_vassal != "") {
current_build_search_item.vassal = current_build_search_item_vassal;
}
// 打包装箱
search_data_lines.push(current_build_search_item);
// 重置资源添加目标 和 vassal
appendTarget = "resource"
current_build_search_item_vassal = ""
}
}
// 添加种类
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( ! /^(https?|ftp):\/\/[^\s/$.?#].[^\s]*$/i.test(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;
$.ajax({
url: `${url}?t=${+new Date().getTime()}`,
timeout: 5000, // 设置超时时间为 5 秒钟
success: function (result) {
resolve(result)
},
error: function(xhr, status, errorThrown){
console.log("cdn失败,不加速请求!");
// 反馈错误,调整请求顺序,避免错误还是访问
// 获取请求错误的根域名
let { domain } = parseUrl(url);
// 根据根域名从模板中找出完整域名
let templates = allCdns.filter(item=>item.includes(domain));
// 反馈
if(templates.length > 0 ) {
if(index > 0 || index <= cache.get(registry.other.UPDATE_CDNS_CACHE_KEY).length ) feedbackError(registry.other.UPDATE_CDNS_CACHE_KEY,templates[0]);
}
console.logout("反馈重调整后:",cache.get(registry.other.UPDATE_CDNS_CACHE_KEY)); // 反馈的结果只会在下次起作用
// 处理失败后的回调函数代码
rq(cdnPack({index,url,initUrl}));
}
});
}
rq(cdnPack({index:0,url:dataSourceUrl,initUrl:dataSourceUrl}));
});
}
// 下面的 dataSourceHandle 函数
let globalFetchFun = [];
// tis处理队列
let waitQueue = [];
// 缓存数据
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;
}
}
function dataSourceHandle(resourcePageUrl,tisMetaInfo = {}) { //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); // 记录
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);
}
}else {
// --> 是内容 <--
// 解析内容
if(tisTabFetchFunName === "") return;
let fetchFunStr = getFetchFunGetByName(tisTabFetchFunName);
let searchDataItems =(new Function('text', "return ( " + fetchFunStr + " )(`"+callBeforeParse.escape(text)+"`)"))();
// 处理并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,null,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;
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 searchInputDocument = null;
let matchItems = null;
let searchBox = null;
let isInitializedView = false;
let logoButton = null;
let textView = null;
let matchResult = 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";
view.innerHTML = (`
<div id="tis"></div>
<div id="my_search_view">
<div id="${searchBoxId}" >
<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="markdown-body" style="min-height:auto !important;">
</div>
</div>
`)
// 挂载到文档中
document.body.appendChild(view)
// 整个视图对象放在组件全局中/注册表中
registry.view.viewDocument = viewDocument = view;
// 想要追加请看下面registry.view.element是否已经包含,没有在那下面追加即可~
// 搜索框对象
searchInputDocument = $(document.getElementById(searchInputDocumentId))
matchItems = $(document.getElementById(matchItemsId));
searchBox = $(document.getElementById(searchBoxId))
logoButton = $(document.getElementById(logoButtonId))
textView = $(document.getElementById(textViewDocumentId))
matchResult = $(document.getElementById(matchResultDocumentId));
// 将视图对象放到注册表中
registry.view.element = {
input: searchInputDocument,
logoButton,
matchResult,
textView
}
// 菜单函数(点击输入框右边按钮时会调用)
function onClickLogo() {
registry.view.menuActive = true;
// alert("小彩蛋:可以搜索一下“系统项”了解脚本基本使用哦~");
// 调用手动触发搜索函数,如果已经搜索过,搜索空串(清理)
let keyword = "[系统项]";
registry.searchData.triggerSearchHandle(searchInputDocument.val()==keyword?'':keyword);
setTimeout(function(){ registry.view.menuActive = false;},registry.view.delayedHideTime+100);
// 重新聚焦搜索框
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(); // 输入框聚焦
}
});
/*// 图片放大/还原
textView.on("click","img",function(e) {
let target = e.target;
if(target.isEnlarge??false) {
$(this).animate({
width: "100%"
});
// 还原
target.isEnlarge = false;
}else {
$(this).animate({
width: "900px"
});
target.isEnlarge = true;
}
});*/
// 设置视图已经初始化
registry.view.initialized = true;
// 在搜索的结果集中上下选择移动然后回车(相当点击)
searchInputDocument.keyup(function(event){
let keyword = $(event.target).val().trim();
// 当不为空时,放到全局keyword中
if(keyword) {
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 ( ! 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){ // 回车选择的元素
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]);
// 设置活跃背景颜色
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;
*/
// 将数据放在指定搜索层级数据上
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) )
|| ( `${dataItem.resource}${dataItem.vassal}`.substring(0, 2048).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);
// 正确提取
let style = `
;${titleTagColorMatchHandler(tagCore)};
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;
`;
resultTitle = resultTitle.toReplaceAll(tag,`<span style="${style}">${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.resource}${item.vassal}`.substring(0, 2048).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
// 如果带#将加上删除线
let style = "color: #1a0dab;";
if( title.startsWith("#")) {
style = `text-decoration:line-through;color:#a8a8a8;`;
title = title.replace(/^#/,"")
}
return `<span style="${style}" class="item_title">${title}</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>`;
// 将符合的数据装载到视图
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>
${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 $("#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;
}
// 简述内容转markdown前
function sketchResourceToHtmlBefore(txtStr = "") {
// 1、“换行”转无意义中间值
txtStr = txtStr.replace(/<\s*br\s*\/\s*>/gm,"?br?"); // 单行简述下的换行,注意要在"<",">"转意前就要做了,注意顺序
// 2、特殊字符 转无意义中间值
txtStr = txtStr.replace(/</gm,"?lt?").replace(/>/gm,"?gt?").replace(/"/gm,"?quot?").replace(/'/gm,"?#39?");
return txtStr;
}
//简述内容转markdown
function sketchResourceToHtmlAfter(txtStr = "") {
// 1、链接变超链接,这里必需要使用“先匹配再替换”
const regexParam = /[^("?>]\s*(https?:\/\/[^\s()()\[\]<>"`]+)/gm;
let m;
let textStrClone = txtStr;
while ((m = regexParam.exec(textStrClone)) !== null) {
// 这对于避免零宽度匹配的无限循环是必要的
if (m.index === regexParam.lastIndex) {
regexParam.lastIndex++;
}
let match = m[0];
// 为简讯内容的url添加可链接
const regex = /(https?:\/\/[^\s()()\[\] `]+)/gm;
const subst = `<a href="$1" target="_blank">$1</a>`;
// 被替换的值将包含在结果变量中
let aTab = match.replace(regex, subst);
txtStr = txtStr.replace(match, aTab);
}
// 2、无意义中间值 转有意符
function revert(text) {
let obj = {
"?lt?":"<",
"?gt?":">",
"?quot?":""",
"?#39?":"'",
"?br?":"<br />"
}
for(let key in obj) {
text = text.toReplaceAll(key,obj[key]);
}
return text;
}
txtStr = revert(txtStr);
return txtStr;
}
$("#matchItems").on("click","li > a",function(e) {
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 />${sketchResourceToHtmlAfter(converter.makeHtml(sketchResourceToHtmlBefore( body )))} `)
}
if(hasVassal) {
showTextPage(itemData.title,"主项的相关/附加内容",itemData.vassal);
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() {
let viewHtml = itemData.resourceObj['view:html'];
let viewCss = itemData.resourceObj['view:css'];
let viewJs = itemData.resourceObj['view:js'];
// 将html挂载到视图中
if(this.beforeCallback != null) this.beforeCallback();
// 挂载MS_script_env_var,实现系统脚本API到视图
// registry.script.script_env_var是给脚本系统通知的 window.MS_script_env_var是view页获取的
actualWindow.MS_script_env_var = registry.script.script_env_var = {
fillKeyword: registry.searchData.subSearch.getSubSearchKeyword() || "",
event: {
sendListener : [] // (fillKeyword)=>{}
}
}
registry.view.textView.show(viewHtml,viewCss,viewJs);
// wait view complate alfter ...
waitViewRenderingComplete(()=>{
registry.script.tryRunTextViewHandler();
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)
// 初始化后将isInitializedView变量设置为true
isInitializedView = true;
}
let hideView = function () {
// 隐藏视图
// 如果视图还没有初始化,直接退出
if (!isInitializedView) 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";
// 将输入框内容置空,在置空前将值备份,好让未好得及的操作它
searchInputDocument.val("")
// 将之前搜索结果置空
matchItems.html("")
// 隐藏文本显示视图
textView.css({
"display":"none"
})
// 让搜索结果隐藏
matchResult.css({
"display":"none"
})
// 视图隐藏-清理旧数据
registry.searchData.clearData();
// 触发视图隐藏事件
registry.view.viewHideEventAfterListener.forEach(fun=>fun());
}
let showView = function () {
// 让视图可见
viewDocument.style.display = "block";
//聚焦
searchInputDocument.focus()
// 当输入框失去焦点时,隐藏视图
searchInputDocument.blur(function() {
if(registry.view.logo.isLogoButtonPressedRef.value) return;
setTimeout(function(){
if(registry.searchData.searchEven.isSearching) return;
// 判断输入框的内容是不是":debug"或是否正处于阅读模式,如果是,不隐藏
if(isInstructions("debug") || isInstructions("read")) return;
// 当前视图是否在展示数据,如搜索结果,简述内容?如果在展示不隐藏
let isNotExhibition = (($("#matchResult").css("display") == "none" || $("#matchItems > li").length == 0 ) && ($("#text_show").css("display") == "none" || $("#text_show").text().trim() == "") );
if(!isNotExhibition || registry.view.menuActive ) return;
registry.view.viewVisibilityController(false);
},registry.view.delayedHideTime)
});
}
// 返回给外界控制视图显示与隐藏
return function (isSetViewVisibility) {
if (isSetViewVisibility) {
// 让视图可见 >>>
// 如果还没初始化先初始化 // 初始化数据 initData();
if (!isInitializedView) {
// 初始化视图
initView();
// 初始化数据
// initData();
}
// 让视图可见
showView();
} else {
// 隐藏视图 >>>
if (isInitializedView) hideView();
}
}
})();
// 触发策略——快捷键
let useKeyTrigger = function (viewVisibilityController) {
let isFirstShow = true;
// 将视图与触发策略绑定
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
const searchButton = $("#search-tishub").click(async function() {
const keyword = selector(".tis-hub .keyword input").value;
// 搜索类型(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) {
const tisMetaInfo = new PageTextHandleChains(tis.body).parseAllDesignatedSingTags("tis")[0];
if(tisMetaInfo == null) continue;
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()
});
}).click();
// 单选框值改变时,搜索
const radioButtons = document.querySelectorAll('input[name="search-type"]');
radioButtons.forEach(radio => {
radio.addEventListener('change', function() {
if (this.checked) {
searchButton.click();
}
});
});
})
}
})(unsafeWindow);
// unsafeWindow是真实的window,作为参数传入