Greasy Fork is available in English.

Safari Search Redirector

A Safari-focused userscript that redirects searches to other engines using keywords, providing search engine customization that Safari lacks

질문, 리뷰하거나, 이 스크립트를 신고하세요.
// ==UserScript==
// @name         Safari Search Redirector
// @name:zh-CN   Safari 搜索引擎重定向器
// @name:zh-TW   Safari 搜尋引擎重定向器
// @namespace    https://github.com/hxueh
// @version      0.3
// @description  A Safari-focused userscript that redirects searches to other engines using keywords, providing search engine customization that Safari lacks
// @description:zh-cn 这是一个将 Safari 重定向到其他搜索引擎的脚本。主要是解决 Safari 自定义搜索引擎能力缺失的问题。
// @description:zh-tw 這是一個將 Safari 重定向到其他搜尋引擎的腳本。主要是解決 Safari 自定義搜尋引擎能力缺失的問題。
// @author       https://github.com/hxueh
// @match        *://www.google.com/*
// @match        *://www.ecosia.org/*
// @compatible   safari
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_addStyle
// @grant        GM_registerMenuCommand
// @run-at       document-start
// @license      MIT
// @icon         
// ==/UserScript==

const STORAGE_KEY = "safari_search_redirector_default";
const searchEngines = {
  g: `https://www.google.com/search?q=%s`, // Google
  google: `https://www.google.com/search?q=%s`, // Google
  sp: `https://www.startpage.com/search?q=%s`, // Startpage
  startpage: `https://www.startpage.com/search?q=%s`, // Startpage
  ecosia: `https://www.ecosia.org/search?q=%s`, // Ecosia
  kagi: `https://kagi.com/search?q=%s`, // Kagi
  caixin: `https://search.caixin.com/newsearch/caixinsearch?keyword=%s`, // Caixin
  brave: `https://search.brave.com/search?q=%s`, // Brave Search
  gpt: `https://chatgpt.com/?temporary-chat=true&q=%s`, // ChatGPT
  chatgpt: `https://chatgpt.com/?temporary-chat=true&q=%s`, // ChatGPT
  claude: `https://claude.ai/new?q=%s`, // Claude
  dict: `https://www.collinsdictionary.com/us/dictionary/english/%s`, // Collins Dictionary
  ndb: `https://neodb.social/search?q=%s`, // NeoDB
  neodb: `https://neodb.social/search?q=%s`, // NeoDB
  podcast: `https://www.listennotes.com/search/?q=%s`, // Listen Notes
  pp: `https://www.perplexity.ai/search?q=%s`, // Perplexity
  perplexity: `https://www.perplexity.ai/search?q=%s`, // Perplexity
  reddit: `https://www.reddit.com/search?q=%s`, // Reddit
  so: `https://stackoverflow.com/search?q=%s`, // Stack Overflow
  twitter: `https://twitter.com/search?q=%s`, // Twitter
  x: `https://www.x.com/search?q=%s`, // X
  yt: `https://youtube.com/results?search_query=%s`, // YouTube
  youtube: `https://youtube.com/results?search_query=%s`, // YouTube
  taobao: `https://s.taobao.com/search?q=%s`, // Taobao
  jd: `https://search.jd.com/Search?keyword=%s`, // JD
  bili: `https://search.bilibili.com/video?keyword=%s`, // Bilibili
  xhs: `https://www.xiaohongshu.com/search_result?keyword=%s`, // Xiaohongshu
  weibo: `https://s.weibo.com/weibo/%s`, // Weibo
  youdao: `https://dict.youdao.com/search?q=%s`, // Youdao
  zhihu: `https://www.zhihu.com/search?type=content&q=%s`, // Zhihu
  douban: `https://www.douban.com/search/?q=%s`, // Douban
  hn: `https://hn.algolia.com/?q=%s`, // Hacker News
};

const defaultSearchEngineRefer = {
  "www.ecosia.org": "tts",
  "www.google.com": "client",
};

// Add CSS for the settings window
GM_addStyle(`
  .search-settings-overlay {
    position: fixed;
    top: 0;
    left: 0;
    right: 0;
    bottom: 0;
    background: rgba(0, 0, 0, 0.5);
    z-index: 999999;
    display: flex;
    justify-content: center;
    align-items: center;
  }

  .search-settings-window {
    background: white;
    padding: 20px;
    border-radius: 8px;
    box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
    width: 300px;
  }

  .search-settings-window h2 {
    margin: 0 0 20px 0;
    font-size: 18px;
    color: #333;
  }

  .search-settings-window select {
    width: 100%;
    padding: 8px;
    border: 1px solid #ddd;
    border-radius: 4px;
    margin-bottom: 20px;
  }

  .search-settings-window .buttons {
    display: flex;
    justify-content: flex-end;
    gap: 10px;
  }

  .search-settings-window button {
    padding: 8px 16px;
    border: none;
    border-radius: 4px;
    cursor: pointer;
  }

  .search-settings-window .save-btn {
    background: #4CAF50;
    color: white;
  }

  .search-settings-window .cancel-btn {
    background: #f5f5f5;
    color: #333;
  }

  .search-settings-window button:hover {
    opacity: 0.9;
  }
`);

function getDefaultEngine() {
  return GM_getValue(STORAGE_KEY, "kagi");
}

function showSettingsWindow() {
  const overlay = document.createElement("div");
  overlay.className = "search-settings-overlay";

  const window = document.createElement("div");
  window.className = "search-settings-window";
  window.innerHTML = `
    <h2>Default Search Engine Settings</h2>
    <select id="default-engine">
      ${Object.entries(searchEngines)
        .map(
          ([key]) => `
          <option value="${key}" ${
            key === getDefaultEngine() ? "selected" : ""
          }>
            ${key}
          </option>
        `
        )
        .join("")}
    </select>
    <div class="buttons">
      <button class="cancel-btn">Cancel</button>
      <button class="save-btn">Save</button>
    </div>
  `;

  function closeSettings() {
    document.body.removeChild(overlay);
  }

  // Event Handlers
  window.querySelector(".save-btn").addEventListener("click", () => {
    const engine = window.querySelector("#default-engine").value;
    GM_setValue(STORAGE_KEY, engine);
    closeSettings();
  });

  window.querySelector(".cancel-btn").addEventListener("click", closeSettings);
  overlay.addEventListener("click", (e) => {
    if (e.target === overlay) closeSettings();
  });

  overlay.appendChild(window);
  document.body.appendChild(overlay);
}

// Register settings menu command
GM_registerMenuCommand("⚙️ Default Search Engine Settings", showSettingsWindow);

(function () {
  "use strict";
  // Get the current hostname
  const hostname = window.location.hostname;

  // Get the URL parameters
  const urlParams = new URLSearchParams(window.location.search);

  // Get the search query from the URL
  let query = urlParams.get("q");
  // Exit if query is not exists or empty
  if (!query || query.trim().length === 0) return;
  // get Safari only search engine refer
  const referKey = defaultSearchEngineRefer[hostname];
  // Exit if the refer is not exists
  if (!referKey || referKey.trim().length === 0) return;
  const referValue = urlParams.get(referKey);
  if (!referValue || referValue.trim().length === 0) return;

  // Check if the query starts with a keyword in the searchEngines mapping
  const keyword = Object.keys(searchEngines).find((k) =>
    query.toLowerCase().startsWith(k + " ")
  );

  let searchEngineUrl;
  if (keyword) {
    // Get the search term after the keyword
    query = query.slice(keyword.length).trim();
    searchEngineUrl = searchEngines[keyword];
  } else {
    // Default search engine
    searchEngineUrl = searchEngines[getDefaultEngine()];
  }
  // Skip redirection if already on the target search engine
  const searchEngineHostname = new URL(searchEngineUrl).hostname;
  if (hostname === searchEngineHostname) {
    return;
  }

  // Clear the document content
  window.stop();
  document.documentElement.innerHTML = "<html></html>";
  // Redirect to the search engine
  window.location.href = searchEngineUrl.replace(
    "%s",
    encodeURIComponent(query)
  );
})();