Greasy Fork

我的搜索

打造属于自己的搜索引擎!

目前为 2022-12-24 提交的版本。查看 最新版本

// ==UserScript==
// @name         我的搜索
// @namespace    http://tampermonkey.net/
// @version      1.1.0
// @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)
        }
    }
    // 判断是否为指定指令
    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: {
            data: []
        }
    }
    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 sysFetchFun = `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",
            targetDocument: "#readme > .markdown-body",
            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",
            targetDocument: "#readme > .markdown-body",
            fetchFun: sysFetchFun
        }
    ];
    // 模块四:初始化数据源
    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 || 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 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 = (`
             <div id="${viewName}">
                <div id="searchBox">
                    <input placeholder="我的搜索" id="search_input" />
                    <button id="controlButton">Bug</button>
                </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: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);
                // 搜索操作
                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",
                    "text-decoration":"none",
                    "text-align":"left",
                    "overflow":"hidden", //超出的文本隐藏
                    "text-overflow":"ellipsis", //溢出用省略号显示
                    "white-space":"nowrap" //溢出不换行
                })



            }
            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")) 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);
        }
    })();

})();