// ==UserScript==
// @name 我的搜索
// @namespace http://tampermonkey.net/
// @version 2.2.1
// @description 订阅式搜索,打造属于自己的搜索引擎!
// @license MIT
// @author zhuangjie
// @match *://*/*
// @exclude http://127.0.0.1*
// @exclude http://localhost*
// @exclude http://192.168.*
// @icon 
// @require https://cdn.bootcdn.net/ajax/libs/jquery/3.6.2/jquery.min.js
// @grant window.onurlchange
// @grant GM_setValue
// @grant GM_getValue
// @grant GM_deleteValue
// @grant GM_registerMenuCommand
// @downloadURL none
// ==/UserScript==
(function() {
'use strict';
// 模块一:快捷键触发某一事件 (属于触发策略组)
// 模块二:搜索视图(显示与隐藏)(属于搜索视图组)
// 模块三:触发策略组触发策略触发搜索视图组视图
// 模块四:根据用户提供的策略(策略属于数据生成策略组)生成搜索项的数据库
// 模块五:视图接入数据库
// 【函数库】
// 数据缓存器
let cache = {
get(key) {
return GM_getValue(key);
},
set(key,value) {
GM_setValue(key,value);
}
}
//防抖函数模板
function debounce(fun, wait) {
let timer = null;
return function (...args) {
// 清除原来的定时器
if (timer) clearTimeout(timer)
// 开启一个新的定时器
timer = setTimeout(() => {
fun.apply(this, args)
}, wait)
}
}
// 判断是否为指定指令
function isInstructions(val,cmd) {
return val == ":"+cmd;
}
// 全局注册表
let ERROR = {
tell(info) {
console.error("ERROR " + info)
}
}
let registry = {
view: {
viewVisibilityController: async () => { ERROR.tell("视图未初始化,但你使用了它的未初始化的注册表信息!") },
viewDocument: null,
setButtonVisibility: async () => { ERROR.tell("按钮未初始化!") },
},
searchData: { //registry.searchData.subscribeKey
data: [],
subscribeKey: "subscribeKey"
}
}
let dao = {}
function showControlButton() {
// 会显示一个按钮
// 初始化按钮
let viewDocument = registry.view.viewDocument;
if(viewDocument == null) return;
// 视图已初始化,可以显示按钮
}
function hideControlButton() {
// 隐藏掉输入框右边按钮
}
// 实现模块一:使用快捷键触发指定事件
function triggerAndEvent(goKeys = "ctrl+alt+s", fun, isKeyCode = false) {
// 监听键盘按下事件
let handle = function (event) {
let isCtrl = goKeys.indexOf("ctrl") >= 0;
let isAlt = goKeys.indexOf("alt") >= 0;
let lastKey = goKeys.replaceAll("alt", "").replaceAll("ctrl", "").replaceAll("+", "").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);
}
// 数据源组
// 内置提取函数 (.+)[((](.*)[))]\s*[::]\s*(http.+)|(.+)[::](http.+) (.+)\s*[::]\s*(http.+)
let sFetchFun = `function(line) {
let baseReg = "([^::]+)[((](.*)[))]\s*[::]\s*(.+)";
let ifNotDescMatchReg = "([^::]+)\s*[::]\s*(.*)"
let title = "";
let desc = "";
let resource = "";
let captureResult = null;
if( !(/[()()]/.test(line))) {
// 兼容没有描述
captureResult = line.match(ifNotDescMatchReg);
if(captureResult == null ) return;
title = captureResult[1];
desc = "--无描述--";
resource = captureResult[2];
}else {
// 正常语法
captureResult = line.match(baseReg)
if(captureResult == null ) return;
title = captureResult[1];
desc = captureResult[2];
resource = captureResult[3];
}
return {
title: title,
desc: desc,
resource: resource
};
}`
/* let dataSources = [
{
url: "https://raw.githubusercontent.com/18476305640/xiaozhuang/dev/%E5%B0%8F%E5%BA%84%E7%9A%84%E7%BD%91%E7%AB%99%E6%94%B6%E8%97%8F%E5%AE%A4.md",
fetchFun: sysFetchFun
},
{
url: "https://raw.githubusercontent.com/18476305640/xiaozhuang/dev/%E5%B0%8F%E5%BA%84%E7%9A%84%E8%BD%AF%E4%BB%B6%E6%94%B6%E8%97%8F%E5%AE%A4.md",
fetchFun: sysFetchFun
}
];*/
// 获取存在的订阅信息
function getSubscribe() {
// 查看是否有订阅信息
let subscribeKey = registry.searchData.subscribeKey;
let subscribeInfo = cache.get(subscribeKey);
if(subscribeInfo == null ) {
// 初始化订阅信息(初次)
subscribeInfo = `
`;
cache.set(subscribeKey,subscribeInfo);
}
return subscribeInfo;
}
function editSubscribe(subscribe) {
// 判断导入的订阅是否有效
// 获取订阅信息(得到的值肯定不会为空)
let pageTextHandleChainsY = pageTextHandleChains.init(subscribe);
let tisHasFetchFun = pageTextHandleChainsY.parseSingleTab("tis","fetchFun");
let tisNotFetchFun = pageTextHandleChainsY.parseSingleTabValue("tis");
let tis = [...tisHasFetchFun, ...tisNotFetchFun];
// 生成订阅信息存储
let subscribeText = "\n";
for(let aTis of tisHasFetchFun) {
subscribeText += `\n`
}
for(let aTis of tisNotFetchFun) {
subscribeText += `\n`
}
// 持久化
let newSubscribeInfo = subscribeText.replace(/\n+/gm,"\n\n");
cache.set(registry.searchData.subscribeKey,newSubscribeInfo);
return tis.length;
}
// 存储订阅信息,当指定 sLineFetchFun 时,表示将解析“直接页”的配置,如果没有指定 sLineFetchFun 时,只解析内容
// 在提取函数中 \n 要改写为 \\n
let dataSources = getSubscribe()+ `
function(pageText) {
let type = "sketch"; // url sketch
let lines = pageText.split("\\n");
let search_data_lines = []; // 扫描的搜索数据 {},{}
let current_build_search_item = {};
let current_build_search_item_resource = "";
let point = 0; // 指的是上面的 current_build_search_item
let default_desc = "--无描述--"
function getTitleLineData(titleLine) {
const regex = /# ([^((\s]*)[((]?([^))\s]*)[))]?/gm;
// let test = /# ([^((\s]*)[((]?([^))\s]*)[))]?/gm.exec("# abc(abc)");
let matchData = regex.exec(titleLine)
return {
title: matchData[1],
desc: ((matchData[2]==null || matchData[2] == "")?default_desc:matchData[2])
}
}
for (let line of lines) {
if(line.indexOf("# ") == 0) {
// 处理上一个item的结尾工作
if(point > 0) { // 必须有前一个item才处理
// 加入resource,最后一项
current_build_search_item.resource = current_build_search_item_resource;
// 打包装箱
search_data_lines.push(current_build_search_item);
}
// 当前新的开始工作
point++;
// 创建新的搜索项目容器
current_build_search_item = {...getTitleLineData(line)}
// 重置resource
current_build_search_item_resource = "";
continue;
}
// 向当前搜索项目容器追加当前行
current_build_search_item_resource += (line+"\\n");
}
// 添加种类
for(let line of search_data_lines) {
line.type = type;
}
return search_data_lines;
}
function(pageText) {
let type = "url"; // url sketch
let lines = pageText.split("\\n");
let search_data_lines = []
for (let line of lines) {
let search_data_line = (function(line) {
let baseReg = "([^::]+)[((](.*)[))]\s*[::]\s*(.+)";
let ifNotDescMatchReg = "([^::]+)\s*[::]\s*(.*)"
let title = "";
let desc = "";
let resource = "";
let captureResult = null;
if( !(/[()()]/.test(line))) {
// 兼容没有描述
captureResult = line.match(ifNotDescMatchReg);
if(captureResult == null ) return;
title = captureResult[1];
desc = "--无描述--";
resource = captureResult[2];
}else {
// 正常语法
captureResult = line.match(baseReg)
if(captureResult == null ) return;
title = captureResult[1];
desc = captureResult[2];
resource = captureResult[3];
}
return {
title: title,
desc: desc,
resource: resource
};
})(line);
if (search_data_line == null || search_data_line.title == null) continue;
search_data_lines.push(search_data_line)
}
for(let line of search_data_lines) {
line.type = type;
}
return search_data_lines;
}
`;
// github CDN加速包装器
function cdnPack(githubResourceUrl) {
let githubUrlFlag = "raw.githubusercontent.com";
// 如何不满足github url ,不加速
if(githubResourceUrl.indexOf(githubUrlFlag) < 0) return githubResourceUrl;
return "https://proxy.zyun.vip/"+githubResourceUrl;
}
// 模块四:初始化数据源
// 使用责任链模式——对pageText进行操作的工具
const pageTextHandleChains = {
pageText: "",
setPageText(newPageText) {
this.pageText = newPageText;
},
getPageText() {
return this.pageText;
},
init(newPageText = "") {
// 深拷贝一份实例
let wo = {...this};
// 初始化
wo.setPageText(newPageText);
return wo;
},
// 解析双标签-获取指定标签下指定属性下的值
parseDoubleTab(tabName,attrName) {
// 返回指定标签下指定属性下的值
const regex = RegExp(`<\\s*${tabName}[^<>]*\\s*${attrName}="([^<>]*)"\\s*>([\\s\\S]*?)<\/\\s*${tabName}\\s*>`,"gm");
let m;
let tabNameArr = [];
let copyPageText = this.pageText;
// 注意下面的 copyPageText 不能改变
while ((m = regex.exec(copyPageText)) !== null) {
// 这对于避免零宽度匹配的无限循环是必要的
if (m.index === regex.lastIndex) {
regex.lastIndex++;
}
tabNameArr.push({
attrValue: m[1],
tabValue: m[2]
})
const newPageText =this.pageText.replace(m[0], "");
this.pageText = newPageText;
}
return tabNameArr;
},
// 解析双标签-只获取值
parseDoubleTabValue(tabName) {
// 返回指定标签下指定属性下的值
const regex = RegExp(`<\\s*${tabName}[^<>]*\\s*>([\\s\\S]*?)<\/\\s*${tabName}\\s*>`,"gm");
let m;
let tabNameArr = [];
let copyPageText = this.pageText;
while ((m = regex.exec(copyPageText)) !== null) {
// 这对于避免零宽度匹配的无限循环是必要的
if (m.index === regex.lastIndex) {
regex.lastIndex++;
}
tabNameArr.push({
tabValue: m[1]
})
const newPageText =this.pageText.replace(m[0], "");
this.pageText = newPageText;
}
return tabNameArr;
},
// 获取指定单标签指定属性与标签值(标签::值)
parseSingleTab(tabName,attrName) {
// 返回指定标签下指定属性下的值
const regex = RegExp(`<${tabName}::([^\\s<>]*)\\s*${attrName}="([^"<>]*)"\\s*\/>`,"gm");
let m;
let tabNameArr = []
let copyPageText = this.pageText;
while ((m = regex.exec(copyPageText)) !== null) {
// 这对于避免零宽度匹配的无限循环是必要的
if (m.index === regex.lastIndex) {
regex.lastIndex++;
}
tabNameArr.push({
tabValue: m[1],
attrValue: m[2]
})
const newPageText =this.pageText.replace(m[0], "");
this.pageText = newPageText;
}
return tabNameArr;
},
parseSingleTabValue(tabName) {
// 返回指定标签下指定属性下的值
const regex = RegExp(`<${tabName}::([^\\s<>]*)[^<>]*\/>`,"gm");
let m;
let tabNameArr = []
let copyPageText = this.pageText;
while ((m = regex.exec(copyPageText)) !== null) {
// 这对于避免零宽度匹配的无限循环是必要的
if (m.index === regex.lastIndex) {
regex.lastIndex++;
}
tabNameArr.push({
tabValue: m[1]
})
const newPageText =this.pageText.replace(m[0], "");
this.pageText = newPageText;
}
return tabNameArr;
},
// 清除指定单双标签
cleanTabByTabName(tabName) {
const regex = RegExp(`<\\s*${tabName}[^<>]*>([^<>]*)(<\/[^<>]*>)*`,"gm");
// 替换的内容
const subst = ``;
// 被替换的值将包含在结果变量中
const cleanedText = this.pageText.replace(regex, subst);
this.pageText = cleanedText;
}
}
// 从 订阅信息(或页) 中解析出配置(json)
function getConfigFromDataSource(pageText) {
let config = {
// {url、fetchFun属性}
tis: [],
// {name与fetchFun属性}
fetchFuns: []
}
// 从config中放在返回对象中
let pageTextHandleChainsX = pageTextHandleChains.init(pageText);
let fetchFunTabDatas = pageTextHandleChainsX.parseDoubleTab("fetchFun","name");
for(let fetchFunTabData of fetchFunTabDatas) {
config.fetchFuns.push( { name:fetchFunTabData.attrValue,fetchFun:fetchFunTabData.tabValue } )
}
// 获取tis
let tisHasFetchFun = pageTextHandleChainsX.parseSingleTab("tis","fetchFun");
let tisNotFetchFun = pageTextHandleChainsX.parseSingleTabValue("tis");
let tisArr = [...tisHasFetchFun, ...tisNotFetchFun]
for(let tis of tisArr) {
config.tis.push( { url:tis.tabValue, fetchFun:tis.attrValue } )
}
return config;
}
// 将url转为文本(url请求得到的就是文本),当下面的dataSourceUrl不是http的url时,就会直接返回,不作请求
function urlToText(dataSourceUrl) {
// dataSourceUrl 转text
return new Promise(function (resolve, reject) {
if((dataSourceUrl.trim().indexOf("http") != 0 ) ) return resolve(dataSourceUrl) ;
$.ajax({
url: cdnPack(dataSourceUrl),
success: function (result) {
resolve(result)
}
});
});
}
// 下面的 dataSourceHandle 函数
let globalFetchFun = [];
let waitQueue = [];
function dataSourceHandle(resourcePageUrl,tisTabFetchFunName) {
urlToText(resourcePageUrl).then(text => {
if(tisTabFetchFunName == null) {
// --> 是配置 <--
let data = []
// 解析配置,是一个json
let config = getConfigFromDataSource(text);
console.log("解析的配置:",config)
// 将FetchFun放到全局解析器中
globalFetchFun.push(...config.fetchFuns);
// 将tis放到处理队列中
waitQueue.push(...config.tis);
let tis = null;
while((tis = waitQueue.pop()) != undefined) {
// tis有两个url,第二是fetchFun
dataSourceHandle(tis.url,tis.fetchFun);
}
// 清理内容
pageTextHandleChains.setPageText("");
}else {
// --> 是内容 <--
// 解析内容
if(tisTabFetchFunName === "") return;
let fetchFunStr = getFetchFunGetByName(tisTabFetchFunName);
let search_data_line = (new Function('', "return (" + fetchFunStr + ")('" + text.replaceAll("\n","\\n") + "')"))();
// 将之前修改为 改为真正的换行符 \n
let replaceBefore = "\n";
let replaceAfter = "\\n";
for(let item of search_data_line) {
item.title = item.title.replaceAll(replaceAfter,replaceBefore);
item.desc = item.desc.replaceAll(replaceAfter,replaceBefore);
item.resource = item.resource.replaceAll(replaceAfter,replaceBefore);
}
registry.searchData.data.push(...search_data_line);
}
})
}
// 根据fetchFun名返回字符串函数
function getFetchFunGetByName(fetchFunName) {
for(let fetchFunData of globalFetchFun) {
if(fetchFunData.name == fetchFunName) {
return fetchFunData.fetchFun;
}
}
}
let initData = function () {
registry.searchData.data = [];
// 内部将使用递归,解析出信息
dataSourceHandle(dataSources,null);
console.log("总搜索数据:",registry.searchData.data)
}
// 模块二
registry.view.viewVisibilityController = (function () {
// 整个视图对象
let viewDocument = null;
let searchInputDocument = null;
let matchItems = null;
let searchBox = null;
let isInitializedView = false;
let viewName = "my_search_view"
let controlButton = null;
let initView = function () {
// 初始化视图
let view = document.createElement("div")
view.innerHTML = (`
`)
// 设置样式
view.style = `
position: fixed;left: 25%;right: 25%;top:50px;
border:2px solid #cecece;z-index:10000;
background: #ffffff;
overflow: hidden;
`;
// 挂载到文档中
document.body.appendChild(view)
// 整个视图对象放在组件全局中/注册表中
registry.view.viewDocument = viewDocument = view;
// 搜索框对象
searchInputDocument = $("#search_input")
matchItems = $("#matchItems");
searchBox = $("#searchBox")
controlButton = $("#controlButton")
searchBox.css({
"height": "45px",
"background": "#ffffff",
"padding": "0px 10px",
"box-sizing": " border-box",
"z-index": "10001",
"position":"relative"
})
searchInputDocument.css({
"width": "100%",
"height": "100%",
"border": "none",
"outline": "none",
"font-size": "15px"
})
$("#matchResult > ol").css({
"margin": "0px",
"padding": "0px 30px",
"overflow": "hidden"
})
controlButton.css({
"position": "absolute",
"font-size":"12px !important",
"right": "5px",
"margin":"10px 7px",
"padding":"3px 15px",
"border-radius":"13.5px",
"border":"none",
"display":"none", // 默认隐藏,由函数控制
})
// 将输入框的控制按钮设置可见性函数公开放注册表中
registry.view.setButtonVisibility = function (buttonVisibility = false) {
// registry.view.setButtonVisibility
controlButton.css({
"display": buttonVisibility?"block":"none"
})
}
// 给输入框加事件
// 执行 debounce 函数返回新函数
let handler = function (e) {
let key = e.target.value;
let searchResultData = []
let searchLevelData = [
[],[],[] // 分别是匹配标题/desc/url 的结果
]
// 如果为空时,不作搜索
if(key == "") {
// 置空搜索
matchItems.html("")
return;
}
// 前置处理函数,这里使用观察者模式
// searchPreFun(key);
// 搜索操作
let currentIndex = 0; // 数据项在总数据中的索引
for (let dataItem of registry.searchData.data) {
key = key.toUpperCase();
// 将数据放在指定搜索层级数据上
if ((dataItem.title.toUpperCase().indexOf(key) >= 0 && searchLevelData[0].push(dataItem) )
|| (dataItem.desc.toUpperCase().indexOf(key) >= 0 && searchLevelData[1].push(dataItem) )
|| (dataItem.resource.toUpperCase().indexOf(key) >= 0 && searchLevelData[2].push(dataItem) ) ) {
// 向满足条件的数据对象添加在总数据中的索引
dataItem.index = currentIndex;
}
currentIndex++;
}
console.log("层级数据:",searchLevelData)
// 将上面层级数据放在总容器中
searchResultData.push(...searchLevelData[0])
searchResultData.push(...searchLevelData[1])
searchResultData.push(...searchLevelData[2])
console.log("搜索总数据:",searchResultData)
// 放到视图上
// 置空内容
matchItems.html("")
// 最多显示条数
let show_item_number = 15;
for(let searchResultItem of searchResultData ) {
// 限制条数
if(show_item_number-- <= 0) {
break;
}
// 将符合的数据装载到视图
let item = `${searchResultItem.type=="sketch"?"📖":""} ${searchResultItem.title}(${searchResultItem.desc})`
matchItems.html(matchItems.html() + item)
currentIndex++;
}
// 设置li样式
$("#matchResult li").css({
"line-height": "30px",
"color": "#0088cc",
"list-style": "decimal",
"width":"100%",
"margin":"0px"
})
$("#matchResult li>a").css({
"display":"block",
"font-size":"15px",
"color": "#5094d5",
"text-decoration":"none",
"text-align":"left",
"overflow":"hidden", //超出的文本隐藏
"text-overflow":"ellipsis", //溢出用省略号显示
"white-space":"nowrap" //溢出不换行
})
}
// 给查询出来的结果添加事件 -- 设置事件委托
$("#matchItems").on("click","li > a",function(e) {
// 设置为阅读模式
$("#search_input").val(":read");
// 获取当前结果在搜索数组中的索引
let dataIndex = parseInt($(e.target).attr("index"));
let itemData = registry.searchData.data[dataIndex];
// 如果是简述搜索信息,那就取消a标签的默认跳转事件
if(itemData.resource.trim().indexOf("http") !== 0) {
// 取消默认事件
e.preventDefault();
alert("标题:"+itemData.title+"\n"+ "描述:"+itemData.desc+"\n"+"简述内容:\n"+itemData.resource);
return;
}
// 取消冒泡
window.event? window.event.cancelBubble = true : e.stopPropagation();
// 否则是URL跳转
})
const refresh = debounce(handler, 460)
// 第一次触发 scroll 执行一次 fn,后续只有在停止滑动 1 秒后才执行函数 fn
searchBox.on('input', refresh)
// 初始化后将isInitializedView变量设置为true
isInitializedView = true;
}
let hideView = function () {
// 隐藏视图
// 如果视图还没有初始化,直接退出
if (!isInitializedView) return;
// 让视图隐藏
viewDocument.style.display = "none";
// 将输入框内容置空
searchInputDocument.val("")
// 将之前搜索结果置空
matchItems.html("")
}
let showView = function () {
// 让视图可见
viewDocument.style.display = "block";
//聚焦
searchInputDocument.focus()
// 当输入框失去焦点时,隐藏视图
searchInputDocument.blur(function() {
// 判断输入框的内容是不是":debug"或是否正处于阅读模式,如果是,不隐藏
if(isInstructions(searchInputDocument.val(),"debug") || isInstructions(searchInputDocument.val(),"read")) return;
setTimeout(()=>{registry.view.viewVisibilityController(false)},200)
});
}
// 返回给外界控制视图显示与隐藏
return function (isSetViewVisibility) {
if (isSetViewVisibility) {
// 让视图可见 >>>
// 如果还没初始化先初始化 // 初始化数据 initData();
if (!isInitializedView) {
// 初始化视图
initView();
// 初始化数据
initData();
}
// 让视图可见
showView();
} else {
// 隐藏视图 >>>
if (isInitializedView) hideView();
}
}
})();
// 触发策略——快捷键
let useKeyTrigger = function (viewVisibilityController) {
// 将视图与触发策略绑定
triggerAndEvent("ctrl+alt+s", function () {
// 让视图可见
viewVisibilityController(true);
})
triggerAndEvent("Escape", function () {
// 让视图不可见
viewVisibilityController(false);
})
}
// 触发策略组
let trigger_group = [useKeyTrigger];
// 初始化入选的触发策略
(function () {
for (let trigger of trigger_group) {
trigger(registry.view.viewVisibilityController);
}
})();
// 打开视图进行配置
// 显示配置视图
// 是否显示进度 - 进度控制
GM_registerMenuCommand("订阅管理",function() {
showConfigView();
});
// 显示配置规则视图
function showConfigView() {
if($("#subscribe_save")[0] != null) return;
// 显示视图
var configViewContainer = document.createElement("div");
configViewContainer.style=`
width:450px; background:pink;
position: fixed;right: 0px; top: 0px;
z-index:10000;
padding: 20px;
border-radius: 14px;
`
configViewContainer.innerHTML = `
订阅总览:
`;
// 设置样式
document.body.appendChild(configViewContainer);
document.getElementById("title").style="margin-bottom: 10px; font-size: 16px;";
document.getElementById("all_subscribe").style="width:100%;height:150px";
document.getElementById("subscribe_save").style=" margin-top: 20px; border: none; border-radius: 3px; padding: 4px 20px; cursor: pointer;";
// 回显
document.getElementById("all_subscribe").value = getSubscribe();
// 保存
document.getElementById("subscribe_save").onclick=function() {
// 保存到对象
let allSubscribe = document.getElementById("all_subscribe").value;
let validCount = editSubscribe(allSubscribe);
// 清除视图
configViewContainer.remove();
alert("保存配置成功!有效订阅数:"+validCount);
}
}
})();