聚合搜索V4

整合百度、F搜、Google、微信、Bing、知乎、知网空间搜索,提高搜索效率,可以自由添加和删除搜索引擎

// ==UserScript==
// @name         聚合搜索V4
// @namespace    http://tampermonkey.net/
// @version      1.0.1
// @description  整合百度、F搜、Google、微信、Bing、知乎、知网空间搜索,提高搜索效率,可以自由添加和删除搜索引擎
// @author       Liao Brant
// @match        *://*/*
// @grant        unsafeWindow
// @grant        window.onload
// @grant        GM_registerMenuCommand
// @grant        GM_getValue
// @grant        GM_setValue
// @run-at       document-body

// @license     MIT
// ==/UserScript==

const defaultUrlMap = [
  {
    name: '百度',
    searchUrl: 'https://www.baidu.com/s?wd=%s',
    keyName: 'wd',
  },
  {
    name: 'F搜',
    searchUrl: 'https://fsoufsou.com/search?q=%s',
    keyName: 'q',
  },
  {
    name: 'Google',
    searchUrl: 'https://www.google.com/search?q=%s',
    keyName: 'q',
  },
  {
    name: '开发者',
    searchUrl: 'https://kaifa.baidu.com/searchPage?wd=%s&module=SEARCH',
    keyName: 'wd',
  },
  {
    name: '微信文章',
    searchUrl: 'https://weixin.sogou.com/weixin?type=2&s_from=input&query=%s',
    keyName: 'query',
  },
  {
    name: 'Bing',
    searchUrl: 'https://www.bing.com/search?ensearch=0&q=%s',
    keyName: 'q',
  },
  {
    name: '知乎',
    searchUrl: 'https://www.zhihu.com/search?type=content&q=%s',
    keyName: 'q',
  },
  {
    name: '知网空间',
    searchUrl: 'https://search.cnki.com.cn/Search/Result?content=%s',
    keyName: 'q',
  },
];

// 获取搜索配置
function getSearchConfig() {
  const urlConfig = GM_getValue('SearchList');
  if (!urlConfig || !urlConfig.length) {
    // 默认的网址配置
    GM_setValue('SearchList', defaultUrlMap);
    return defaultUrlMap;
  } else {
    return urlConfig;
  }
}

// 搜索网址配置
let urlMapping = getSearchConfig();

// JS获取url参数
function getQueryVariable(variable) {
  let query = window.location.search.substring(1);
  let pairs = query.split('&');
  for (let pair of pairs) {
    let [key, value] = pair.split('=');
    if (key == variable) {
      return decodeURIComponent(value);
    }
  }
  return null;
}

// 从url中获取搜索关键词
function getKeywords() {
  let keywords = '';
  for (let item of urlMapping) {
    // 判断url是否符合搜索配置的条件
    if (item.searchUrl.includes(window.location.hostname + window.location.pathname)) {
      keywords = getQueryVariable(item.keyName);
      break;
    }
  }
  console.log(keywords);
  return keywords;
}

// 域名
const hostname = window.location.hostname;

let isBlank = GM_getValue('isBlank');

console.log('新标签页打开?', isBlank);
if (isBlank === undefined) {
  GM_setValue('isBlank', false);
  isBlank = false;
}

// 改变打开搜索引擎的方式
const engine = document.getElementsByClassName('search-engine-a');
function triggerAttribute(value) {
  for (const item of engine) {
    item.target = value;
  }
}

// 适配火狐浏览器的百度搜索
const isFirefox = () => {
  if (navigator.userAgent.indexOf('Firefox') > 0) {
    console.warn('[ Firefox ] 🚀');
    urlMapping[0].searchUrl = 'https://www.baidu.com/baidu?wd=%s';
  } else {
  }
};

// 适配cn.bing.com的必应域名
const cnBing = {
  name: 'Bing',
  searchUrl: 'https://cn.bing.com/search?q=%s',
  keyName: 'q',
};
// 匹配到cn.bing就修改必应配置对象
if (window.location.hostname === 'cn.bing.com') {
  for (let item of urlMapping) {
    if (item.name === 'Bing') {
      item = cnBing;
    }
  }
}

// 添加自定义样式
function addStyle() {
  const styleSheet = document.createElement('style');
  document.head.appendChild(styleSheet);

  styleSheet.textContent = `
  .search-container {
    position: fixed;
    top: 160px;
    left: 20px;
    width: 100px;
    background-color: #EEEEEE;
    font-size: 12px;
    z-index: 99999;
  }

  .search-title {
    display: block;
    text-align: center;
    margin-top: 10px;
    margin-bottom: 5px;
    font-size: 14px;
    font-weight: bold;
    -webkit-user-select:none;
    -moz-user-select:none;
    -ms-user-select:none;
    user-select:none;
  }

  .search-list__item {
    display: flex;
    justify-content: space-between;
    align-items: center;
    color: #333333 !important;
    padding-right: 6px;
  }
  .search-list__item:hover {
    color: #ffffff !important;
    background-color: #666666;
  }

  .search-list__item .search-engine-a {
    color: inherit !important;
    padding: 10px 6px 10px 20px;
    text-decoration: none;
    flex-grow: 2;
  }

  .search-list__close {
    width: 12px;
    height: 12px;
    opacity: 0;
    z-index: 10;
  }
  .search-list__close:hover {
    opacity: 1;
  }

  #search-black_over {
    content: "";
    position: absolute;
    top: 0;
    left: 0;
    width: 100vw;
    height: 100%;
    // background-color: #F0F5FF;
    background-color: #f0f5ffe3;
    z-index: 999;
  }
  #search-popup {
    position: fixed;
    top: 50vh;
    left: 50vw;
    background-color: #FFFFFF;
    width: 460px;
    transform: translate(-50%, -50%);
    z-index: 1000;
  }
  #setting-engine-ul {
    padding: 20px;
    height: 60vh;
    min-height: 400px;
    overflow: auto;
  }
  #setting-engine-ul::-webkit-scrollbar {
    width: 10px;
  }

  #setting-engine-ul::-webkit-scrollbar-thumb {
    background-color: #d9d9d9;
    border-radius: 9999em;
  }
  .setting-engine-item{
    margin-bottom: 10px;
    padding: 10px 0;
    border-radius: 4px !important;
    box-shadow: 0 2px 8px rgb(22 93 255 / 8%) !important;
  }
  .engine-info{
    margin: 5px;
  }
  .engine-label{
    width: 70px;
    display: inline-block;
  }
  .engine-info input {
    width: 275px;
  }

  .setting-engine-header, .setting-engine-footer{
    display: flex;
    padding: 15px;
    justify-content: space-between;
    align-items: center;
    border-top: 1px solid #e8e8e8;
  }
`;
}

// 添加节点
function addBox() {
  // 主元素
  const div = document.createElement('div');
  div.id = 'search-app-box';
  div.className = 'search-container';
  document.body.insertAdjacentElement('afterBegin', div);

  // 标题
  let title = document.createElement('span');
  title.innerText = '聚合搜索';
  title.className = 'search-title';
  title.style.textDecoration = isBlank ? 'underline' : '';
  title.ondblclick = () => {
    title.style.textDecoration = !isBlank ? 'underline' : '';
    GM_setValue('isBlank', !isBlank);
    isBlank = !isBlank;
    triggerAttribute(isBlank ? '_blank' : '');
  };
  div.appendChild(title);

  // 搜索列表
  for (let index in urlMapping) {
    const item = urlMapping[index];
    // 单个搜索引擎
    const searchItem = document.createElement('div');
    searchItem.className = 'search-list__item';
    const a_target = !item.searchUrl.includes(hostname) && isBlank ? '_blank' : '';

    searchItem.innerHTML = `
      <a class="search-engine-a" href="${item.searchUrl.replace('%s', getKeywords())}" ${a_target}>
        ${item.name}
      </a>
    `;
    // 移除按钮
    const closeImg = document.createElement('img');
    closeImg.className = 'search-list__close';
    closeImg.src =
      '';
    // 点击按钮移除节点
    closeImg.onclick = function () {
      urlMapping.splice(index, 1);
      searchItem.remove();
      GM_setValue('SearchList', urlMapping);
    };

    searchItem.appendChild(closeImg);
    div.appendChild(searchItem);
  }
}
// 重新生成节点
function addBoxAgain() {
  document.getElementById('search-app-box')?.remove();
  addBox();
}
// 关闭面板
closePanel = () => {
  document.getElementById('search-black_over')?.remove();
  document.getElementById('search-popup')?.remove();
};
// 新增搜索引擎
addEngine = () => {
  const engineUl = document.getElementById('setting-engine-ul');
  const li = document.createElement('li');
  li.className = 'setting-engine-item';
  li.innerHTML = `
    <div class="engine-info">
      <label class="engine-label">引擎名称</label>
      <input class="GF-engine-name__input" />
    </div>
    <div class="engine-info">
      <label class="engine-label">引擎地址</label>
      <input class="GF-engine-url__input"/>
    </div>
    <div class="engine-info">
      <label class="engine-label">搜索参数</label>
      <input class="GF-engine-key__input" />
    </div>
  `;
  engineUl.appendChild(li);
};
// 重置配置
resetEngineConfig = () => {
  urlMapping = defaultUrlMap;
  addBoxAgain();
  closePanel();
};
// 保存配置
saveEngineConfig = () => {
  const list = [];
  const names = document.getElementsByClassName('GF-engine-name__input');
  const urls = document.getElementsByClassName('GF-engine-url__input');
  const keys = document.getElementsByClassName('GF-engine-key__input');

  for (let index = 0; index < names.length; index++) {
    list.push({
      name: names[index]?.value,
      searchUrl: urls[index]?.value,
      keyName: keys[index]?.value,
    });
  }

  urlMapping = list;
  GM_setValue('SearchList', list);
  addBoxAgain();
  closePanel();
};
// 创建设置面板
function createSettingPanel() {
  // 遮罩
  const over = document.createElement('div');
  over.id = 'search-black_over';
  // 弹窗
  const popup = document.createElement('div');
  popup.id = 'search-popup';

  let engineList = '';
  urlMapping.forEach((item) => {
    engineList += `<li class="setting-engine-item">
      <div class="engine-info">
        <label class="engine-label">引擎名称</label>
        <input class="GF-engine-name__input" value="${item.name}" />
      </div>
      <div class="engine-info">
        <label class="engine-label">引擎地址</label>
        <input class="GF-engine-url__input" value="${item.searchUrl}" />
      </div>
      <div class="engine-info">
        <label class="engine-label">搜索参数</label>
        <input class="GF-engine-key__input" value="${item.keyName}" />
      </div>
    </li>`;
  });

  popup.innerHTML = `
    <div class="setting-engine-header">
      <button id="setting-engine-reset" onclick="closePanel()">关闭</button>
      <button id="setting-engine-save" onclick="addEngine()">新增</button>
    </div>
    <ul id="setting-engine-ul">${engineList}</ul>
    <div class="setting-engine-footer">
      <button id="setting-engine-reset" onclick="resetEngineConfig()">重置</button>
      <button id="setting-engine-save" onclick="saveEngineConfig()">保存</button>
    </div>
  `;

  document.body.insertAdjacentElement('afterBegin', popup);
  document.body.insertAdjacentElement('afterBegin', over);
}
// 添加油猴设置菜单
function addUserSetting() {
  GM_registerMenuCommand('设置', createSettingPanel);
}
// 初始化
function init() {
  if (!getKeywords()) return;
  addStyle();
  isFirefox();
  addBox();
  addUserSetting();
}

(function () {
  'use strict';
  window.onload = init();
})();