// ==UserScript==
// @name 我的搜索
// @namespace http://tampermonkey.net/
// @version 1.00
// @description 打造属于自己的搜索引擎!
// @license MIT
// @author zhuangjie
// @match *://*/*
// @icon 
// @grant window.onurlchange
// @require https://cdn.bootcdn.net/ajax/libs/jquery/3.6.2/jquery.min.js
// ==/UserScript==
(function() {
'use strict';
// 模块一:快捷键触发某一事件 (属于触发策略组)
// 模块二:搜索视图(显示与隐藏)(属于搜索视图组)
// 模块三:触发策略组触发策略触发搜索视图组视图
// 模块四:根据用户提供的策略(策略属于数据生成策略组)生成搜索项的数据库
// 模块五:视图接入数据库
// 【函数库】
//防抖函数模板
function debounce(fun, wait) {
let timer = null;
return function (...args) {
// 清除原来的定时器
if (timer) clearTimeout(timer)
// 开启一个新的定时器
timer = setTimeout(() => {
fun.apply(this, args)
}, wait)
}
}
// 全局注册表
let ERROR = {
tell(info) {
console.error("ERROR " + info)
}
}
let registry = {
view: {
viewVisibilityController: async () => { ERROR.tell("视图未初始化,但你使用了它的未初始化的注册表信息!") }
},
searchData: {
data: []
}
}
let dao = {}
// 实现模块一:使用快捷键触发指定事件
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);
}
// 数据源组
let dataSources = [
{
url: "https://raw.githubusercontent.com/18476305640/xiaozhuang/144f6988c983ae0a480a60c83e647d1e2e230b17/%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",
targetDocument: "#readme > .markdown-body",
fetchFun: `function(line) {
let captureResult = line.match("(.+)[((](.*)[))]\s*[::]\s*(http.+)|(.+)[::](http.+)")
if(captureResult == null) return null;
return {
title: captureResult[1],
desc: captureResult[2],
resource: captureResult[3]
};
}`
},
{
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",
targetDocument: "#readme > .markdown-body",
fetchFun: `function(line) {
let captureResult = line.match("(.+)[((](.*)[))]\s*[::]\s*(http.+)|(.+)[::](http.+)")
if(captureResult == null) return null;
return {
title: captureResult[1],
desc: captureResult[2],
resource: captureResult[3]
};
}`
}
];
// 模块四:初始化数据源
let initData = function () {
registry.searchData.data = []
for (let dataSource of dataSources) {
new Promise(function (resolve, reject) {
$.ajax({
url: dataSource.url,
success: function (result) {
resolve(result)
}
});
}).then((result) => {
// 交给handler
//dataSource.fetchFun()
let lines = result.split("\n");
let search_data_lines = []
for (let line of lines) {
let search_data_line = (new Function('', "return (" + dataSource.fetchFun + ")('" + line + "')"))();
if (search_data_line == null) continue;
if (search_data_line.title == null) continue;
search_data_lines.push(search_data_line)
}
registry.searchData.data.push(...search_data_lines)
});
}
}
// 模块二
registry.view.viewVisibilityController = (function () {
let isInitializedView = false;
let viewName = "my_search_view"
let searchInputDocument = null //搜索框对象
let viewDocument = null; // 整个视图对象
let matchItems = null;
let initView = function () {
// 初始化视图
let view = document.createElement("div")
view.innerHTML = (`
<div id="${viewName}">
<div id="search">
<input placeholder="我的搜索,打造属于个人的搜索引擎" id="search_input" />
</div>
<div id="matchResult">
<ol id="matchItems">
<!-- <li><a href="#">webGL水族馆(测试电脑性能)</a></li> -->
</ol>
</div>
</div>
`)
// 设置样式
view.style = `
position: fixed;left: 25%;right: 25%;top:50px;
border:1px solid #cecece;z-index:10000;
background: #ffffff;
overflow: hidden;
`;
// 挂载到文档中
document.body.appendChild(view)
// 挂载整个视图对象到组件全局变量
viewDocument = view;
// 挂载输入框到组件全局变量
searchInputDocument = $("#search_input")
matchItems = $("#matchItems");
$("#search").css({
"height": "45px",
"background": "#ffffff",
"padding": "0px 10px",
"box-sizing": " border-box",
"z-index": "10001"
})
searchInputDocument.css({
"width": "100%",
"height": "100%",
"border": "none",
"outline": "none",
"font-size": "15px"
})
$("#matchResult > ol").css({
"margin": "0px",
"padding": "0px 30px",
"overflow": "hidden"
})
// 给输入框加事件
// 执行 debounce 函数返回新函数
let handler = function (e) {
let key = e.target.value;
let searchResultData = []
let searchLevelData = [
[],[],[]
]
// 如果为空时,不作搜索
if(key == "") {
// 置空搜索
matchItems.html("")
return;
}
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) ) ) {
}
}
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 = `<li><a href="${searchResultItem.resource}" target="_blank">${searchResultItem.title}(${searchResultItem.desc})</a></li>`
matchItems.html(matchItems.html() + item)
}
// 设置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",
"overflow":"hidden", //超出的文本隐藏
"text-overflow":"ellipsis", //溢出用省略号显示
"white-space":"nowrap" //溢出不换行
})
}
const refresh = debounce(handler, 460)
// 第一次触发 scroll 执行一次 fn,后续只有在停止滑动 1 秒后才执行函数 fn
$("#search").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() {
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);
}
})();
})();