// ==UserScript==
// @name 搜索引擎工具栏(改)
// @namespace https://github.com/examplecode/useful-user-scripts
// @homepageURL https://scriptcat.org/zh-CN/users/157252
// @author examplecode && Aloazny && Gemini
// @version 4.6
// @description 二改@examplecode的搜索脚本,感谢@examplecode,添加MD3配色+高斯模糊+编辑/添加搜索引擎功能+搜索栏切换顶部底部+搜索框,保留原本的X浏览器搜索引擎读取,排序等功能。
// @match *://*/*
// @icon 
// @license MIT
// @run-at document-end
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_EX_getSearchEngines
// @grant GM_xmlhttpRequest
// @connect suggestion.baidu.com
// @connect suggestqueries.google.com
// @connect api.bing.com
// ==/UserScript==
(function() {
'use strict';
const STORAGE_KEY = 'quick_search_bar_final';
const THEME_KEY = 'quick_search_theme_v2';
const LAYOUT_KEY = 'quick_search_layout_v1';
const queryParams = ["q", "s", "p", "wd", "word", "keyword", "text", "query", "key", "result", "searchWord", "search-result" ];
const longSignals = ['search', 'search-result', 'category', 'find', 'query', 'result', 'tags', 'tag'];
const strictSignals = ['s', 'k', 'p'];
const defaultTheme = { primary: '#6750a4', opacity: 0.7, blur: 18, themeStyle: 'MIUIX' };
const defaultLayout = { pos: 'bottom', offset: 24, autoHide: false, showInput: false, inputStyle: 1, shrinkMode: false, historyLimit: 100, showSuggest: true };
const HISTORY_KEY = 'quick_search_history_v1';
const defaultEngines = [
{ name: '必应', host: 'bing.com', url: 'https://www.bing.com/search?q=%s' },
{ name: 'Yandex', host: 'yandex', url: 'https://yandex.com/search/touch/?text=%s' },
{ name: 'Google', host: 'google', url: 'https://www.google.com/search?q=%s' },
{ name: '百度', host: 'baidu.com', url: 'https://www.baidu.com/s?word=%s' },
{ name: 'ScriptCat', host: 'scriptcat.org', url: 'https://scriptcat.org/zh-CN/search?keyword=%s' },
{ name: '4Khd', host: '/(4khd|xxtt|ssuu|uuss)\\.(com|ink|uk)/', url: 'https://www.4khd.com/search/%s' },
{ name: '纳米AI', host: 'n.cn', url: 'https://www.n.cn/search/?q=%s&src=ec_vivo_1001' }
];
const storage = {
save: (engines, deletedIds) => {
const data = {
custom: engines.filter(e =>
!defaultEngines.some(d => storage.getStableId(d.name, d.host) === e.id) &&
!xEngines.some(x => storage.getStableId(x.name, x.host) === e.id)
),
order: engines.map(e => e.id),
hidden: engines.filter(e => !e.visible).map(e => e.id),
deleted: deletedIds || [],
pathRules: saved.pathRules || {},
cache: engines.map(e => ({ name: e.name, host: e.host, url: e.url, id: e.id, visible: e.visible, pathRule: (saved.pathRules && saved.pathRules[e.host]) || '' }))
};
if (typeof GM_setValue !== 'undefined') GM_setValue(STORAGE_KEY, data);
else localStorage.setItem(STORAGE_KEY, JSON.stringify(data));
},
load: () => {
let data;
if (typeof GM_getValue !== 'undefined') data = GM_getValue(STORAGE_KEY);
else { try { data = JSON.parse(localStorage.getItem(STORAGE_KEY)); } catch(e) {} }
return data || { custom: [], order: [], hidden: [], deleted: [], pathRules: {}, cache: null };
},
themeSave: (val) => {
if (typeof GM_setValue !== 'undefined') GM_setValue(THEME_KEY, val);
else localStorage.setItem(THEME_KEY, JSON.stringify(val));
applyTheme();
},
themeLoad: () => {
let data;
if (typeof GM_getValue !== 'undefined') data = GM_getValue(THEME_KEY);
else { try { data = JSON.parse(localStorage.getItem(THEME_KEY)); } catch(e) {} }
return data || defaultTheme;
},
layoutSave: (val) => {
if (typeof GM_setValue !== 'undefined') GM_setValue(LAYOUT_KEY, val);
else localStorage.setItem(LAYOUT_KEY, JSON.stringify(val));
},
layoutLoad: () => {
let data;
if (typeof GM_getValue !== 'undefined') data = GM_getValue(LAYOUT_KEY);
else { try { data = JSON.parse(localStorage.getItem(LAYOUT_KEY)); } catch(e) {} }
return data || defaultLayout;
},
getStableId: (name, host) => {
let str = (name || '') + '|' + (host || '');
let hash = 0;
for (let i = 0; i < str.length; i++) {
hash = ((hash << 5) - hash + str.charCodeAt(i)) | 0;
}
return 'se_' + Math.abs(hash).toString(36);
}
};
let theme = storage.themeLoad();
let layout = storage.layoutLoad();
function hexToRgb(hex) {
hex = hex.replace('#', '');
if (hex.length === 3) hex = hex[0] + hex[0] + hex[1] + hex[1] + hex[2] + hex[2];
let r = parseInt(hex.substring(0, 2), 16), g = parseInt(hex.substring(2, 4), 16), b = parseInt(hex.substring(4, 6), 16);
const brightness = (r * 299 + g * 587 + b * 114) / 1000;
const contrastColor = brightness > 150 ? '#000000' : '#ffffff';
return { rgb: `${r}, ${g}, ${b}`, contrast: contrastColor };
}
const shadowHost = document.createElement('div');
shadowHost.id = 'qs-shadow-root';
shadowHost.style.cssText = 'position:absolute;z-index:2147483647;';
const shadow = shadowHost.attachShadow({mode: 'open'});
function applyTheme() {
const styleId = 'qs-static-styles';
let el = shadow.getElementById(styleId) || document.createElement('style');
el.id = styleId;
const themeData = hexToRgb(theme.primary);
const isMIUI = theme.themeStyle === 'MIUIX';
const config = isMIUI ? { radius: '100px', btnRadius: '100px', cardRadius: '32px', border: '0.5px solid rgba(255,255,255,0.4)', saturate: '210%', shadow: '0 8px 32px rgba(0,0,0,0.1)' } : { radius: '16px', btnRadius: '12px', cardRadius: '24px', border: '1px solid rgba(0,0,0,0.1)', saturate: '150%', shadow: '0 4px 12px rgba(0,0,0,0.1)' };
el.innerHTML = `
:host {
--qs-pri: ${theme.primary};
--qs-pri-rgb: ${themeData.rgb};
--qs-on-pri: ${themeData.contrast};
--qs-bg: rgba(255, 255, 255, ${theme.opacity});
--qs-blur: ${theme.blur}px;
--qs-radius: ${config.radius};
--qs-btn-radius: ${config.btnRadius};
--qs-card-radius: ${config.cardRadius};
--qs-border: ${config.border};
--qs-saturate: ${config.saturate};
--qs-shadow: ${config.shadow};
}
#qs-manager-overlay { position:fixed;inset:0;background:rgba(0,0,0,0.3);backdrop-filter:blur(12px);-webkit-backdrop-filter:blur(12px);z-index:2147483647;display:flex;justify-content:center;align-items:center;font-family:system-ui,-apple-system,BlinkMacSystemFont,sans-serif; }
.qs-card { background:rgba(255,255,255,0.85);backdrop-filter:saturate(var(--qs-saturate)) blur(25px);-webkit-backdrop-filter:saturate(var(--qs-saturate)) blur(25px);width:92%;max-width:440px;max-height:80vh;border-radius:var(--qs-card-radius);box-shadow:0 20px 50px rgba(0,0,0,0.15);display:flex;flex-direction:column;overflow:hidden;border:var(--qs-border);color:#1c1b1f;transition: border-radius 0.3s ease;box-sizing:border-box; }
.qs-bar { position:fixed; left:50%; transform:translateX(-50%); z-index:2147483646; background:var(--qs-bg); backdrop-filter:saturate(var(--qs-saturate)) blur(var(--qs-blur)); -webkit-backdrop-filter:saturate(var(--qs-saturate)) blur(var(--qs-blur)); display:flex; overflow-x:auto; padding:10px 14px; gap:8px; border-radius:var(--qs-radius); box-shadow:var(--qs-shadow); border:var(--qs-border); max-width:92vw; scrollbar-width:none; align-items:center; transition:transform 0.3s, opacity 0.3s, border-radius 0.3s; scroll-behavior:smooth; box-sizing:border-box; }
.qs-bar::-webkit-scrollbar { display:none; }
.qs-btn { padding:10px 18px; border-radius:var(--qs-btn-radius); text-decoration:none; font-size:14px; white-space:nowrap; font-weight:600; flex-shrink:0; transition: transform 0.1s cubic-bezier(0.4, 0, 0.2, 1), opacity 0.2s; border:none; cursor:pointer; display:inline-flex; align-items:center; justify-content:center; box-sizing:border-box; }
.qs-btn:active { transform: scale(0.92); }
.qs-btn-active { background:var(--qs-pri); color:var(--qs-on-pri); }
.qs-btn-flat { background:rgba(0,0,0,0.04); color:#1d1b20; }
.qs-main-input { border:none; outline:none; font-size:13px; font-weight:500; transition: width 0.3s, background 0.3s; box-sizing:border-box; }
.se-item-container { margin-bottom: 10px; border-radius: 20px; border: 1px solid rgba(0,0,0,0.03); background: rgba(255,255,255,0.6); box-sizing:border-box; }
.se-item { display: flex; align-items: center; padding: 14px 16px; cursor: pointer; box-sizing:border-box; }
.btn-icon { width: 36px; height: 36px; border-radius: 12px; border: none; background: transparent; cursor: pointer; display: flex; align-items: center; justify-content: center; color: #49454f; font-size:18px; box-sizing:border-box; }
.toggle-box { width: 44px; height: 24px; background: #e7e0ec; border-radius: 20px; position: relative; transition: 0.3s; margin-right: 12px; flex-shrink:0; }
.toggle-box.active { background: var(--qs-pri); }
.toggle-circle { width: 18px; height: 18px; background: #fff; border-radius: 50%; position: absolute; top: 3px; left: 3px; transition: 0.3s; }
.toggle-box.active .toggle-circle { left: 23px; }
.edit-form { max-height: 0; opacity: 0; padding: 0 16px; overflow:hidden; transition: all 0.3s ease; box-sizing:border-box; }
.edit-form.open { max-height: 400px; opacity: 1; padding: 10px 16px 20px; }
.qs-input-field { width:100% !important; display:block !important; padding:14px; border-radius:12px; border:1px solid rgba(0,0,0,0.1) !important; background:#ffffff !important; color:#000000 !important; outline:none; font-size:14px; box-sizing:border-box !important; }
.qs-input-field::placeholder { color: #888 !important; }
#m-list::-webkit-scrollbar { display:none; }
#mask { position:fixed;inset:0;background:rgba(0,0,0,0.2);backdrop-filter:blur(${theme.blur}px);-webkit-backdrop-filter:blur(${theme.blur}px);z-index:2147483640;display:none; }
#box { position:fixed; ${layout.pos === 'top' ? 'top:'+(parseInt(layout.offset)+60)+'px' : 'bottom:'+(parseInt(layout.offset)+60)+'px'}; left:50%; transform:translateX(-50%); width:90vw; max-width:400px; z-index:2147483641; display:none; flex-wrap:wrap; gap:8px; padding:10px; box-sizing:border-box; }
.item { padding:8px 16px; background:rgba(255,255,255,0.8); backdrop-filter:blur(10px); border-radius:${isMIUI?'100px':'12px'}; font-size:13px; color:#1d1b20; cursor:pointer; border:1px solid rgba(0,0,0,0.05); transition:0.2s; box-shadow:0 2px 8px rgba(0,0,0,0.05); }
.item:hover { background:var(--qs-pri); color:var(--qs-on-pri); }
.item.his { border-color:rgba(var(--qs-pri-rgb), 0.3); }
`;
if (!el.parentNode) shadow.appendChild(el);
}
const updateSuggestStyle = () => {
applyTheme();
if(!shadow.getElementById('mask')){
const m = document.createElement('div'); m.id = 'mask';
const b = document.createElement('div'); b.id = 'box';
shadow.appendChild(m);
shadow.appendChild(b);
}
};
function normalizeUrl(url) {
return url ? url.replace(/%keywords%/g, '%s').trim() : '';
}
let xEngines = [];
if (typeof GM_EX_getSearchEngines === 'function') {
try {
const list = JSON.parse(GM_EX_getSearchEngines());
if (Array.isArray(list)) {
xEngines = list.map(e => ({ name: e.name || '未知', host: e.host || '', url: normalizeUrl(e.url || '') }));
}
} catch (e) {}
}
let saved = storage.load();
const suggestApi = {
baidu: (s) => `https://suggestion.baidu.com/su?wd=${encodeURIComponent(s)}&p=3`,
google: (s) => `https://suggestqueries.google.com/complete/search?client=chrome&q=${encodeURIComponent(s)}`,
bing: (s) => `https://api.bing.com/osjson.aspx?query=${encodeURIComponent(s)}`,
fetch: (s, type = 'bing') => {
return new Promise((resolve) => {
if (typeof GM_xmlhttpRequest === 'undefined') return resolve([]);
GM_xmlhttpRequest({
method: "GET", url: suggestApi[type](s), onload: (res) => {
try {
if (type === 'baidu') { const match = res.responseText.match(/\[.*\]/); resolve(match ? JSON.parse(match[0]) : []); }
else if (type === 'google' || type === 'bing') { const data = JSON.parse(res.responseText); resolve(data[1] || []); }
} catch (e) { resolve([]); }
}, onerror: () => resolve([])
});
});
}
};
const historyManager = {
get: () => { try { return JSON.parse(localStorage.getItem(HISTORY_KEY)) || []; } catch(e) { return []; } },
add: (s) => {
if (!s || s.startsWith('http')) return;
let h = historyManager.get();
h = [s, ...h.filter(i => i !== s)].slice(0, layout.historyLimit || 100);
localStorage.setItem(HISTORY_KEY, JSON.stringify(h));
}
};
function checkUrl(s) {
const reg = /^(https?:\/\/|ftp:\/\/|www\.)[^\s]+|^[a-z0-9]+([\-\.]{1}[a-z0-9]+)*\.[a-z]{2,5}(:[0-9]{1,5})?(\/.*)?$/i;
if (reg.test(s)) return s.startsWith('http') || s.startsWith('ftp') ? s : 'https://' + s;
return null;
}
function initEngines() {
if (saved.cache && saved.cache.length > 0) {
return saved.cache;
}
if (!saved.pathRules) saved.pathRules = {};
let pool = [...(saved.custom || []), ...defaultEngines, ...xEngines];
let map = new Map();
const seenUrls = new Set(), seenRegs = new Set();
pool.forEach(e => {
const urlKey = (e.url || '').toLowerCase();
const host = (e.host || '').trim();
const isRegExp = host.startsWith('/') && host.endsWith('/');
let shouldAdd = false;
if (isRegExp) {
if (!seenRegs.has(host)) {
seenRegs.add(host);
shouldAdd = true;
}
} else if (urlKey && !seenUrls.has(urlKey)) {
seenUrls.add(urlKey);
shouldAdd = true;
}
if (shouldAdd) {
let id = e.id || storage.getStableId(e.name, e.host);
let finalId = id, suffix = 1;
while (map.has(finalId)) {
finalId = id + '_' + (suffix++);
}
map.set(finalId, { ...e, id: finalId, visible: !(saved.hidden || []).includes(finalId) });
}
});
let list = Array.from(map.values()).filter(e => !(saved.deleted || []).includes(e.id));
if (saved.order && saved.order.length > 0) {
const orderMap = new Map(saved.order.map((id, i) => [id, i]));
list.sort((a, b) => (orderMap.get(a.id) ?? 999) - (orderMap.get(b.id) ?? 999));
}
storage.save(list, saved.deleted);
return list;
}
let allEngines = initEngines().map(e => { const h = (e.host || '').trim(); if (h.startsWith('/') && h.endsWith('/')) { try { const p = h.slice(1, -1); e._matchFn = s => new RegExp(p, 'i').test(s); } catch(err) { e._matchFn = s => s.toLowerCase().includes(h.toLowerCase()); } } else { e._matchFn = s => s.toLowerCase().includes(h.toLowerCase()); } return e; });
function showManager() {
if (shadow.getElementById('qs-manager-overlay')) return;
const overlay = document.createElement('div');
overlay.id = 'qs-manager-overlay';
const card = document.createElement('div');
card.className = 'qs-card';
card.innerHTML = `
${['#6750a4', '#1d1b20', '#0061a4', '#006d39', '#ba1a1a', '#ffb400'].map(c => `
`).join('')}
✨ 当前风格: ${theme.themeStyle || 'MIUIX'}
透明度${Math.round(theme.opacity*100)}%
高斯模糊${theme.blur}px
布局位置
🔍搜索框样式 ${layout.inputStyle || 1}
历史记录上限
发现冲突,请选择导入方式:
• 合并:保留当前所有配置,仅追加文件中独有的搜索引擎,且不改变当前主题颜色。
• 覆盖:清空当前所有配置,完全还原为文件中的状态(含引擎列表、排序及主题配色)。
`;
overlay.appendChild(card);
shadow.appendChild(overlay);
let pendingData = null;
const applyImport = (isMerge) => {
if (!pendingData) return;
if (isMerge) {
if (pendingData[STORAGE_KEY]) {
const local = storage.load();
const remote = pendingData[STORAGE_KEY];
const localCustomIds = new Set(local.custom.map(e => e.id));
const newCustom = remote.custom.filter(nc => !localCustomIds.has(nc.id));
const mergedCustom = [...local.custom, ...newCustom];
const mergedOrder = [ ...local.order, ...remote.order.filter(id => !local.order.includes(id))];
const mergedHidden = [...new Set([...(local.hidden || []), ...(remote.hidden || [])])];
const mergedDeleted = [...new Set([...(local.deleted || []), ...(remote.deleted || [])])];
const mergedPathRules = { ...(local.pathRules || {}), ...(remote.pathRules || {}) };
const merged = {
custom : mergedCustom,
order : mergedOrder,
hidden : mergedHidden,
deleted : mergedDeleted,
pathRules: mergedPathRules,
cache : null
};
if (typeof GM_setValue !== 'undefined') GM_setValue(STORAGE_KEY, merged);
else localStorage.setItem(STORAGE_KEY, JSON.stringify(merged));
}
}
else {
if (pendingData[THEME_KEY]) storage.themeSave(pendingData[THEME_KEY]);
if (pendingData[LAYOUT_KEY]) storage.layoutSave(pendingData[LAYOUT_KEY]);
if (pendingData[STORAGE_KEY]) {
const d = pendingData[STORAGE_KEY];
d.cache = null;
if (typeof GM_setValue !== 'undefined') GM_setValue(STORAGE_KEY, d);
else localStorage.setItem(STORAGE_KEY, JSON.stringify(d));
}
}
location.reload();
};
const themeBtn = card.querySelector('#m-theme-btn'), themePanel = card.querySelector('#m-theme-panel');
const ocdBtn = card.querySelector('#m-ocd-btn'), ocdPanel = card.querySelector('#m-ocd-panel');
const hexInput = card.querySelector('#t-hex-input'), colorPicker = card.querySelector('#t-custom-color');
const opacitySlider = card.querySelector('#t-opacity'), blurSlider = card.querySelector('#t-blur');
const styleCycleBtn = card.querySelector('#m-style-cycle');
const styleSwitch = card.querySelector('#t-style-switch');
const updateAll = () => { storage.themeSave(theme); applyTheme(); };
const dataBtn = card.querySelector('#m-data-btn'), dataPanel = card.querySelector('#m-data-panel');
const fileInput = card.querySelector('#d-file'), importOpts = card.querySelector('#d-import-opts');
hexInput.oninput = (e) => { if (/^#[0-9A-Fa-f]{6}$/.test(e.target.value)) { theme.primary = e.target.value; colorPicker.value = e.target.value; updateAll(); } };
hexInput.onkeydown = (e) => { if (e.key === 'Enter') { e.preventDefault(); hexInput.blur(); updateAll(); } };
colorPicker.oninput = (e) => { theme.primary = e.target.value; hexInput.value = e.target.value; updateAll(); };
opacitySlider.oninput = (e) => { theme.opacity = e.target.value / 100; card.querySelector('#v-opacity').innerText = e.target.value + '%'; updateAll(); };
blurSlider.oninput = (e) => { theme.blur = e.target.value; card.querySelector('#v-blur').innerText = e.target.value + 'px'; updateAll(); };
fileInput.onchange = (e) => { const file = e.target.files[0]; if (!file) return; const reader = new FileReader(); reader.onload = (ev) => { try { pendingData = JSON.parse(ev.target.result); importOpts.style.display = 'block'; } catch (err) { alert('无效的JSON文件'); } }; reader.readAsText(file); };
themeBtn.onclick = () => { [ocdPanel, dataPanel].forEach(p => p.style.display = 'none'); themePanel.style.display = themePanel.style.display === 'none' ? 'block' : 'none'; };
ocdBtn.onclick = () => { [themePanel, dataPanel].forEach(p => p.style.display = 'none'); ocdPanel.style.display = ocdPanel.style.display === 'none' ? 'block' : 'none'; };
dataBtn.onclick = () => { [themePanel, ocdPanel].forEach(p => p.style.display = 'none'); dataPanel.style.display = dataPanel.style.display === 'none' ? 'block' : 'none'; };
styleCycleBtn.onclick = () => { layout.inputStyle = ((parseInt(layout.inputStyle) || 1) % 3) + 1; storage.layoutSave(layout); styleCycleBtn.innerText = `🔍搜索框样式 ${layout.inputStyle}`; styleCycleBtn.style.transform = 'scale(0.95)'; styleCycleBtn.style.transition = 'transform 0.1s'; setTimeout(() => { styleCycleBtn.style.transform = 'scale(1)'; }, 100); };
styleSwitch.onclick = () => { theme.themeStyle = (theme.themeStyle === 'MIUIX' ? 'MD3' : 'MIUIX'); styleSwitch.innerText = `✨ 当前风格: ${theme.themeStyle}`; updateAll(); applyTheme(); styleSwitch.style.transform = 'scale(0.95)'; styleSwitch.style.transition = 'transform 0.2s'; setTimeout(() => styleSwitch.style.transform = 'scale(1)', 150); };
card.querySelectorAll('.theme-dot').forEach(dot => { dot.onclick = () => { theme.primary = dot.dataset.color; hexInput.value = theme.primary; colorPicker.value = theme.primary; updateAll(); overlay.remove(); showManager(); }; });
card.querySelector('#t-reset').onclick = () => { theme = {...defaultTheme}; updateAll(); overlay.remove(); showManager(); };
card.querySelector('#l-pos-toggle').onclick = (e) => { layout.pos = layout.pos === 'top' ? 'bottom' : 'top'; storage.layoutSave(layout); e.target.innerText = layout.pos === 'top' ? '↑靠上' : '靠下↓'; };
card.querySelector('#l-offset').oninput = (e) => { layout.offset = e.target.value; card.querySelector('#l-offset-val').innerText = layout.offset; storage.layoutSave(layout); };
card.querySelector('#d-merge').onclick = () => applyImport(true);
card.querySelector('#d-cover').onclick = () => applyImport(false);
card.querySelector('#d-import').onclick = () => fileInput.click();
card.querySelector('#d-export').onclick = () => { const config = { [STORAGE_KEY]: storage.load(), [THEME_KEY]: storage.themeLoad(), [LAYOUT_KEY]: storage.layoutLoad() }; const blob = new Blob([JSON.stringify(config, null, 2)], { type: 'application/json' }); const a = document.createElement('a'); a.href = URL.createObjectURL(blob); a.download = `search_bar_config_${new Date().getTime()}.json`; a.click(); };
card.querySelector('#l-his-limit').onchange = (e) => { layout.historyLimit = parseInt(e.target.value) || 100; storage.layoutSave(layout); };
card.querySelector('#l-suggest').onclick = (e) => { layout.showSuggest = !layout.showSuggest; storage.layoutSave(layout); updateLayoutBtn('#l-suggest', layout.showSuggest, ' 搜索建议'); e.target.style.transform = 'scale(0.95)'; setTimeout(() => e.target.style.transform = 'scale(1)', 100); };
const updateLayoutBtn = (id, active, text) => {
const btn = card.querySelector(id);
btn.innerText = (active ? '✅' : '🚫') + text;
btn.style.background = active ? 'var(--qs-pri)' : 'rgba(0,0,0,0.05)';
btn.style.color = active ? 'var(--qs-on-pri)' : '#49454f';
};
card.querySelector('#l-input').onclick = () => { layout.showInput = !layout.showInput; storage.layoutSave(layout); updateLayoutBtn('#l-input', layout.showInput, ' 搜索框'); };
card.querySelector('#l-hide').onclick = () => { layout.autoHide = !layout.autoHide; if (layout.autoHide) layout.shrinkMode = false; storage.layoutSave(layout); updateLayoutBtn('#l-hide', layout.autoHide, ' 滑动隐藏'); updateLayoutBtn('#l-shrink', layout.shrinkMode, ' 贴边收缩模式'); };
card.querySelector('#l-shrink').onclick = () => { layout.shrinkMode = !layout.shrinkMode; if (layout.shrinkMode) layout.autoHide = false; storage.layoutSave(layout); updateLayoutBtn('#l-shrink', layout.shrinkMode, ' 贴边收缩模式'); updateLayoutBtn('#l-hide', layout.autoHide, ' 滑动隐藏'); };
const render = () => {
const list = card.querySelector('#m-list');
list.innerHTML = '';
allEngines.forEach((e, i) => {
const currentRule = (saved.pathRules && saved.pathRules[e.host]) ? saved.pathRules[e.host] : '';
const container = document.createElement('div');
container.className = 'se-item-container';
container.innerHTML = `
`;
container.querySelector('.se-item').onclick = (ev) => { if (!ev.target.closest('.btn-icon') && !ev.target.closest('.toggle-box')) container.querySelector('.edit-form').classList.toggle('open'); };
container.querySelector('.toggle-box').onclick = () => { e.visible = !e.visible; storage.save(allEngines, saved.deleted); render(); };
container.querySelector('.up').onclick = () => { if (i > 0) { [allEngines[i], allEngines[i - 1]] = [allEngines[i - 1], allEngines[i]]; storage.save(allEngines, saved.deleted); render(); } };
container.querySelector('.down').onclick = () => { if (i < allEngines.length - 1) { [allEngines[i], allEngines[i + 1]] = [allEngines[i + 1], allEngines[i]]; storage.save(allEngines, saved.deleted); render(); } };
container.querySelector('.del').onclick = () => { saved.deleted.push(e.id); allEngines.splice(i, 1); storage.save(allEngines, saved.deleted); render(); };
container.querySelector('.ed-save').onclick=()=>{const n=container.querySelector('.ed-name').value.trim(),h=container.querySelector('.ed-host').value.trim(),u=normalizeUrl(container.querySelector('.ed-url').value.trim()),p=container.querySelector('.ed-path').value.trim().replace(/%(s|keywords)%?/gi, '');if(n&&h&&u.includes('%s')){if(p){if(!saved.pathRules)saved.pathRules={};saved.pathRules[h]=p}else if(saved.pathRules){delete saved.pathRules[h]}const newId=(n===e.name&&h===e.host)?e.id:storage.getStableId(n,h);Object.assign(e,{name:n,host:h,url:u,id:newId});storage.save(allEngines,saved.deleted);render()}else{alert('请检查输入')}};
list.appendChild(container);
});
};
const addForm = card.querySelector('#m-add-form'), inName = card.querySelector('#in-name'), inHost = card.querySelector('#in-host'), inUrl = card.querySelector('#in-url');
inUrl.oninput=()=>{try{const u=inUrl.value.trim();if(!u)return;const f=new URL(u.startsWith('http')?u:'https://'+u).hostname.replace(/^www\./i,'');if(!inHost.value)inHost.value=f;if(!inName.value){const r=f.split('.')[0];inName.value=r.charAt(0).toUpperCase()+r.slice(1)}}catch(e){}};
card.querySelector('#in-confirm').onclick=()=>{const u=normalizeUrl(inUrl.value);let n=inName.value.trim(),h=inHost.value.trim();if(!u.includes('%s'))return alert('URL必须包含%s');if(allEngines.some(e=>e.host===h&&e.name===n))return alert('已存在');try{const p1=u.split('%s')[0],m=p1.match(/([\/|?|&][^\/|?|&]+)$/);if(m){const s=m[1].replace(/%(s|keywords)%?/gi, '');if(s.length>1&&!longSignals.some(l=>s.includes(l))&&!strictSignals.some(t=>s==='/'+t)){if(!saved.pathRules)saved.pathRules={};saved.pathRules[h]=s}}else{const tU=new URL(u.replace('%s','pk')),pA=tU.pathname.split('/'),i=pA.indexOf('pk');if(i>0){const s=pA[i-1];if(s&&!longSignals.includes(s)&&!strictSignals.includes(s)){if(!saved.pathRules)saved.pathRules={};saved.pathRules[h]='/'+s+'/'}}}}catch(e){}const ne={name:n,host:h,url:u,id:storage.getStableId(n,h),visible:true};if(h.startsWith('/')&&h.endsWith('/')){try{const p=h.slice(1,-1);ne._matchFn=s=>new RegExp(p,'i').test(s)}catch(e){ne._matchFn=s=>s.toLowerCase().includes(h.toLowerCase())}}else{ne._matchFn=s=>s.toLowerCase().includes(h.toLowerCase())}allEngines.push(ne);storage.save(allEngines,saved.deleted);addForm.style.display='none';render()};
card.querySelector('#m-add-btn').onclick = () => { addForm.style.display = 'block'; inName.value = inHost.value = inUrl.value = ''; };
card.querySelector('#in-cancel').onclick = () => addForm.style.display = 'none';
card.querySelector('#m-reset-btn').onclick=()=>{if(confirm('确定还原?')){saved={custom:[],order:[],hidden:[],deleted:[]};allEngines=initEngines();storage.save(allEngines,[]);render()}};render();
card.querySelector('#m-close').onclick=()=>{location.reload()};
overlay.onclick = (e) => { if (e.target === overlay) overlay.remove(); };
}
const currentEngine = allEngines.filter(e => e.visible).find(e => e.host && e._matchFn(location.host.toLowerCase()));
const isMatchedHost = !!currentEngine;
const query = (() => {
if (!isMatchedHost) return null;
try {
if (location.pathname === '/' && !location.search && !location.hash) return null;
const url = new URL(location.href);
const fullPath = url.pathname + url.search + url.hash;
const customRule = saved.pathRules ? saved.pathRules[currentEngine.host] : null;
if (customRule) {
const cleanRule = customRule.replace(/%(s|keywords)%?/gi, '');
const ruleIdx = fullPath.toLowerCase().indexOf(cleanRule.toLowerCase());
if (ruleIdx !== -1) {
let part = fullPath.substring(ruleIdx + cleanRule.length).split(/[&?]/)[0];
if (part) {
if (part.includes('/')) {
const allSignals = [...longSignals, ...strictSignals];
const pieces = part.split('/').filter(p => p && !allSignals.includes(p.toLowerCase()));
part = pieces[pieces.length - 1] || part;
}
let res = decodeURIComponent(part).replace(/^[?&=/]+/g, '').replace(/\.(html|php|jsp|asp|htm)$/i, '').replace(/^-+|-+$/g, '');
if (res) return res;
}
}
}
for (const p of queryParams) {
const reg = new RegExp(`[?&]${p}=([^&]+)`, 'i');
const match = (location.search + location.hash).match(reg);
if (match && match[1]) return decodeURIComponent(match[1].replace(/\+/g, ' ')).trim();
}
const pathLower = url.pathname.toLowerCase();
const hasSearchSignal = longSignals.some(sig => pathLower.includes('/' + sig + '/') || pathLower.endsWith('/' + sig)) || strictSignals.some(sig => pathLower.includes('/' + sig + '/'));
if (hasSearchSignal) {
const pathParts = url.pathname.split('/').filter(p => p.length > 0);
if (pathParts.length > 0) {
const allSignals = [...longSignals, ...strictSignals];
const sigIdx = pathParts.findIndex(p => allSignals.includes(p.toLowerCase()));
let targetPart = (sigIdx !== -1 && pathParts[sigIdx + 1]) ? pathParts[sigIdx + 1] : pathParts[pathParts.length - 1];
let lastPart = decodeURIComponent(targetPart).trim().replace(/\.(html|php|jsp|asp|htm)$/i, '').replace(/^-+|-+$/g, '');
if (lastPart.length >= 1 && !allSignals.includes(lastPart.toLowerCase()) && !/^\d+$/.test(lastPart) && !lastPart.includes('.')) return lastPart;
}
}
} catch (e) { console.warn("关键词提取错误:", e); }
return null;
})();
if (query && isMatchedHost) {
applyTheme();
updateSuggestStyle();
document.body.appendChild(shadowHost);
requestAnimationFrame(() => {
const bar = document.createElement('div');
bar.className = 'qs-bar';
bar.style[layout.pos === 'top' ? 'top' : 'bottom'] = `${layout.offset}px`;
const getBestMatched = () => allEngines.filter(e => e.visible && e.host && e._matchFn(location.host.toLowerCase())).map(e => { let s = (e.host === location.host.toLowerCase() ? 1000 : (!e.host.includes('/') ? 500 : 0)), u; try { u = new URL(e.url); s += (u.pathname !== '/' && location.pathname.toLowerCase().includes(u.pathname.toLowerCase()) ? 200 : 0); u.searchParams.forEach(v => { const f = v.replace(/%s|%keywords%/g, '').trim().toLowerCase(); if (f) s += decodeURIComponent(location.href.toLowerCase()).includes(f) ? 600 : -800; }); } catch(e) {} return { e, s: s + (allEngines.length - allEngines.indexOf(e)) }; }).sort((a, b) => b.s - a.s)[0]?.e;
const bestMatched = getBestMatched();
const showSuggest = async val => { if (!layout.showSuggest) return; const mask = shadow.getElementById('mask'), box = shadow.getElementById('box'); if (!val) { mask.style.display = box.style.display = 'none'; return; } mask.style.display = 'block'; box.style.display = 'flex'; let his = historyManager.get().filter(h => h.includes(val)).slice(0,8); let apiType = 'bing'; if (bestMatched) { const h = bestMatched.host.toLowerCase(); if (h.includes('google')) apiType = 'google'; else if (h.includes('baidu')) apiType = 'baidu'; else if (h.includes('bing')) apiType = 'bing'; } let sug = await suggestApi.fetch(val, apiType); box.innerHTML=''; [...his.map(h=>({t:h,c:'his'})), ...sug.map(s=>({t:s,c:''}))].slice(0,26).forEach(item=>{ const div=document.createElement('div'); div.className='item '+item.c; div.textContent=item.t; div.onclick=(e)=>{ e.preventDefault(); e.stopPropagation(); doSearch(bestMatched||allEngines[0],item.t) }; box.appendChild(div) }); };
const syncInputs = (val) => { bar.querySelectorAll('.qs-main-input').forEach(el => { if (el.value !== val) el.value = val; });};
const doSearch = (targetEngine, keyword) => { if (!keyword) return; const urlLink = checkUrl(keyword); if (urlLink) { window.location.href = urlLink; return; } historyManager.add(keyword); let targetUrl = targetEngine.url; if (targetUrl.includes('custom_browser_search')) { const p = new URLSearchParams(new URL(targetUrl).search); targetUrl = decodeURIComponent(p.get('url')); } window.location.href = targetUrl.replace(/%s/g, encodeURIComponent(keyword)).replace(/%keywords%/g, encodeURIComponent(keyword)); };
const curStyle = layout.inputStyle || 1;
if (curStyle === 3 && layout.showInput) {
const input = document.createElement('input');
input.className = 'qs-main-input';
input.value = query;
input.placeholder = "搜索...";
input.style.cssText = `width:80px;padding:8px 14px;border-radius:100px;background:rgba(var(--qs-pri-rgb),0.08);color:#1d1b20;flex-shrink:0;`;
input.onfocus = () => { input.style.width = '130px'; input.style.background = 'rgba(var(--qs-pri-rgb),0.15)'; showSuggest(input.value); };
input.onblur = () => { setTimeout(() => { shadow.getElementById('mask').style.display = shadow.getElementById('box').style.display = 'none'; input.style.width = '80px'; input.style.background = 'rgba(var(--qs-pri-rgb),0.08)'; }, 200); };
input.oninput = (ev) => { syncInputs(ev.target.value); showSuggest(ev.target.value); };
input.onkeydown = (e) => { if (e.key === 'Enter') { e.preventDefault(); e.stopImmediatePropagation(); doSearch(bestMatched || allEngines[0], e.target.value.trim()); } };
bar.appendChild(input);
}
allEngines.filter(e => e.visible).forEach(e => {
const isBest = bestMatched && e.id === bestMatched.id;
if (isBest && layout.showInput) {
if (curStyle === 1) {
const wrapper = document.createElement('div');
wrapper.style.cssText = `display:flex;align-items:center;background:var(--qs-pri);border-radius:100px;padding:3px;gap:4px;flex-shrink:0;`;
wrapper.setAttribute('data-active', 'true');
const input = document.createElement('input');
input.className = 'qs-main-input';
input.value = query;
input.style.cssText = `width:90px;padding:7px 12px;border-radius:100px;background:rgba(255,255,255,0.2);color:var(--qs-on-pri);`;
input.oninput = (ev) => { syncInputs(ev.target.value); showSuggest(ev.target.value); };
input.onfocus = () => { input.style.width = '140px'; showSuggest(input.value); };
input.onblur = () => { setTimeout(() => { shadow.getElementById('mask').style.display = shadow.getElementById('box').style.display = 'none'; input.style.width = '90px'; }, 200); };
input.onkeydown = (e_ev) => { if (e_ev.key === 'Enter') { e_ev.preventDefault(); e_ev.stopImmediatePropagation(); doSearch(e, input.value.trim()); } };
const label = document.createElement('span');
label.textContent = e.name;
label.style.cssText = `padding:0 14px 0 8px;color:var(--qs-on-pri);font-size:14px;font-weight:600;white-space:nowrap;cursor:pointer;`;
label.onclick = (ev) => { ev.preventDefault(); ev.stopImmediatePropagation(); const val = input.value.trim(); doSearch(e, val); };
wrapper.append(input, label);
bar.appendChild(wrapper);
return;
} else if (curStyle === 2) {
const input = document.createElement('input');
input.className = 'qs-main-input';
input.value = query;
input.style.cssText = `width:80px;padding:8px 14px;border-radius:100px;border:1.5px solid var(--qs-pri);background:rgba(var(--qs-pri-rgb),0.1);color:#1d1b20;flex-shrink:0;margin-right:2px;`;
input.oninput = (ev) => { syncInputs(ev.target.value); showSuggest(ev.target.value); };
input.onfocus = () => { input.style.width = '130px'; showSuggest(input.value); };
input.onblur = () => { setTimeout(() => { shadow.getElementById('mask').style.display = shadow.getElementById('box').style.display = 'none'; input.style.width = '80px'; }, 200); };
input.onkeydown = (e) => { if (e.key === 'Enter') { e.preventDefault(); e.stopImmediatePropagation(); doSearch(bestMatched || allEngines[0], e.target.value.trim()); } };
bar.appendChild(input);
}
}
const btn = document.createElement('a');
btn.className = `qs-btn ${isBest ? 'qs-btn-active' : 'qs-btn-flat'}`;
btn.textContent = e.name;
btn.href = 'javascript:void(0);';
if (isBest) btn.setAttribute('data-active', 'true');
btn.onclick = (ev) => { ev.preventDefault(); ev.stopImmediatePropagation(); const inputEl = bar.querySelector('.qs-main-input'); const currentVal = inputEl ? inputEl.value.trim() : query; doSearch(e, currentVal); };
bar.appendChild(btn);
});
const setBtn = document.createElement('div');
setBtn.innerHTML = '⚙️';
setBtn.style.cssText = 'width:40px;height:40px;display:flex;align-items:center;justify-content:center;cursor:pointer;background:rgba(0,0,0,0.04);border-radius:50%;flex-shrink:0;margin-left:4px;';
setBtn.onclick = showManager;
bar.append(setBtn);
shadow.appendChild(bar);
setTimeout(() => {
const activeEl = bar.querySelector('[data-active="true"]');
if (activeEl) bar.scrollLeft = activeEl.offsetLeft - (bar.offsetWidth / 2) + (activeEl.offsetWidth / 2);
}, 100);
if (layout.autoHide || layout.shrinkMode) {
let lastY = window.scrollY, isShrunk = false;
const applyState = (s) => { isShrunk = s; if (layout.shrinkMode) {
bar.style.cssText += `transition:all .4s cubic-bezier(0.4,0,0.2,1);max-width:${s?'120px':'92vw'};padding-right:${s?'12px':'8px'};transform:translateX(calc(-50% + ${s?'200px':'0px'}));border-radius:${s?'30px 0 0 30px':'var(--qs-radius)'};`;
Array.from(bar.children).forEach(item => { const isCore = item.classList.contains('qs-main-input') || item.classList.contains('qs-btn-active') || item.getAttribute('data-active') === 'true';
item.style.display = s ? (isCore ? '' : 'none') : (item.innerText === '⚙️' ? 'flex' : '');
if (item.classList.contains('qs-main-input') || item.tagName === 'INPUT') item.style.width = item.style.minWidth = s ? '70px' : ''; });
if (s) bar.scrollLeft = 0; else setTimeout(() => { const a = bar.querySelector('[data-active="true"]'); if (a) bar.scrollTo({left: a.offsetLeft - bar.offsetWidth/2 + a.offsetWidth/2, behavior: 'smooth'}); }, 400);
} else { bar.style.transform = s ? `translateX(-50%) translateY(${layout.pos === 'bottom' ? '180%' : '-180%'})` : 'translateX(-50%) translateY(0)'; bar.style.opacity = s ? '0' : '1'; }};
bar.onclick = (e) => { if (isShrunk && layout.shrinkMode) { applyState(false); e.stopPropagation(); }};
window.addEventListener('scroll', () => { const curY = window.scrollY; if (Math.abs(curY - lastY) > 15) { applyState(curY > lastY && curY > 60); lastY = curY; } }, { passive: true });
}
});
}
})();