Greasy Fork

我的搜索

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

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

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

})();