1、基本使用:ctrl+alt+s呼出 Esc 隐藏
2、内容类型说明:有链接内容(点击后跳转),与简述内容,点击后直接可查看的简述文本
3、制作订阅,请查看内置订阅,按着来。
4、“搜索PRO”模式,搜索语法: "<搜索出可搜索,如baidu><tab键><baidu搜索的内容,如 今日热点>"
取消搜索PRO模式:shift+tab
5、新添加/修改的内容,都会有“NEW”的标签,快搜索“NEW”试试吧~ (查看最近7天)
6、呼出搜索框直接按tab
键,输入问题再回车即可体现AI简单问答功能。
作者: Zhuang Jie 联系信息: [email protected] 脚本简述: 我一直热爱收集各类软件和网站,因此积累了大量的信息。之前,我常常通过文字搜索(使用 Ctrl+F 方式)在记录中找到所需信息。然而,这种方式并不总是高效而直观。因此,我开发了这款脚本,它可以帮助我更快地检索和导航到我需要的信息。通过这款脚本,我能够更有效地管理并使用我的收集的所有资源。 更新日志 | 意见反馈/内容提交 | 官方微信公众-订阅号(“我的搜索-搜我想搜”)
基本素养:
- 国际网络教程 | 我的搜索福利机场:>>一键导入<<
脚本技术支持:
作者作品:
- 记住阅读进度-脚本(油猴脚本,记住页面的滚动进度)
- TabAutoClose谷歌浏览器插件-根据规则自动关闭标签
- 脚本用户专享 Boose找工作脚本Auto Batch Push
支持作者 如果你觉得我们的产品有价值,并希望我们能持续改进和开发新的功能,我们真诚的请求你考虑支持我们。你可以选择捐款。每一次的支持,对我们都非常重要。
-- env -- _icon  -- script -- function ( {registry} ) { registry.searchData.triggerSearchHandle(registry.searchData.specialKeyword.new); }
-- env --
-- script -- function ( {registry} ) { registry.searchData.triggerSearchHandle(registry.searchData.specialKeyword.history); }
-- env --
-- script -- function ( {registry} ) { registry.searchData.triggerSearchHandle(registry.searchData.specialKeyword.highFrequency); }
-- env -- _icon 
_describe 使用的公益chatGPTApi来支持的内置AI助手 -- script -- function main({ cache, $, view }) { view.mount(); }
-- view:html --
-- view:css --
#chat-control { display: flex; align-items: stretch; height: 30px; } #chat-control select, #chat-control input, #chat-control button { padding: 5px 10px; margin-right: 2px; font-size: 14px; }
#conversation-history { margin-bottom: 1rem; border: 1px solid #ccc; padding: 0.5rem; min-height: 50px; } pre { display: flex; flex-direction: column; margin: 3px 0; } pre .code-type { display: block; margin: 2px 5px; color: #bcbcbc; /代码类型不可选中/ -webkit-touch-callout: none; -webkit-user-select: none; -khtml-user-select: none; -moz-user-select: none; -ms-user-select: none; user-select: none;
}
.你-message { color: #007BFF; }
.AI-message { color: #28A745; }
#loading { display: inline-block; width: 20px; height: 20px; border: 4px solid rgba(0, 0, 0, 0.1); border-top-color: #000; border-radius: 50%; animation: spin 1s linear infinite; } #prompt { margin: 10px 0; height: 28px; display: flex; } #prompt > .optional { background: #ededed; border-radius: 8px; color: #7a7a7a; padding: 0 6px; cursor: pointer; margin-right: 4px; } #prompt input { border-radius: 8px; font-size: 10px; padding: 0px 10px; border: 1px solid #6e6e6e; } @keyframes spin { to { transform: rotate(360deg); } }
-- view:js --
console.log("进入viewScript")
const headers = new Headers();
headers.append("Content-Type", "application/json");
function getAnyOne(keys = []) {
const randomIndex = Math.floor(Math.random() * keys.length);
return keys[randomIndex];
}
function trimString(str, trimStr) {
// 使用正则表达式去除前后指定的字符串
const regex = new RegExp(^${trimStr}|${trimStr}$
, 'g');
return str.replace(regex, '');
}
// 编码:window.btoa("raw")
// 解码:window.atob("processed")
const providerList = [
'ewogICAgImJhc2VVcmwiOiBbImh0dHBzOi8vZnJlZS52MzYuY20vdjEiXSwKICAgICJrZXlzIjogWwogICAgICAic2stNElJcFRPdDgwTmR0STZHOTEyRGExZkY4N2YzYTRhMGJCYjA2MmVBNDQ3N2NDMjUyIiwKICAgICAgInNrLWxSdFo5dTdyYTcwcjJZQjhCODBlOWNEMDE4MDA0MUI1QWM0MjcwQzBBMzgwMWUwMCIsCiAgICAgICJzay12cnI1WWVQMWlmYkp2S3VVMzNDOERkQjMwM0I5NDczM0E3Q2RDYzE0RTJEY0U4QWYiCiAgICBdLAogICAgIm1vZGVsIjogWyJncHQtNG8tbWluaSIsImdwdC0zLjUtdHVyYm8tMDEyNSIsImdwdC0zLjUtdHVyYm8tMTEwNiIsInNlbGVjdGVkOmdwdC0zLjUtdHVyYm8iLCJncHQtMy41LXR1cmJvLTE2ayIsIm5ldC1ncHQtMy41LXR1cmJvIiwid2hpc3Blci0xIiwiZGFsbC1lLTIiXQogIH0=',
'ewogICAgImJhc2VVcmwiOiBbImh0dHBzOi8vYXBpLmNoYXRhbnl3aGVyZS50ZWNoL3YxIiwiaHR0cHM6Ly9hcGkuY2hhdGFueXdoZXJlLm9yZy92MSJdLAogICAgImtleXMiOiBbCiAgICAgICJzay1hYVU4dkM3Sk93dXpQRUpVaXJENXVqb3cyWndTTUNIc3lGT2pTbUM1Z2EwY0hWNTQiLAogICAgICAic2stM2p6Y0FhUGg2NGVvUENzQWREMmI3V0RESVZPcW0yQ2FUc0dLQVdFTFVaYXJNSXRpIiwKICAgICAgInNrLUpjdDltUVJXMEpqdzZ2OG9qSU9Qb3J1MklkbEd3ZlVoRmFwZW4xSnQxemZUazN5SyIKICAgIF0sCiAgICAibW9kZWwiOiBbImdwdC00Iiwic2VsZWN0ZWQ6Z3B0LTMuNS10dXJibyIsImdwdC00LWNhIiwiZ3B0LTMuNS10dXJiby1jYSJdCiAgfQ==' ] // 选择一个服务提供者信息 let sessionProvider = { ...(typeof providerList[0] === "string" ? JSON.parse(atob(getAnyOne(providerList))) : getAnyOne(providerList)), chooseOut: function() { return { baseUrl: getAnyOne(sessionProvider.baseUrl), key: getAnyOne(sessionProvider.keys), defaultChooseModel: trimString(this.model.find(m => m.startsWith("selected:")) || this.model[0],"selected:") } } } let sessionProviderSpecific = sessionProvider.chooseOut();
// 历史对话记录
let conversationHistory = [{ role: "system", content: "你是乐于助人AI助理" }];
let isLoading = false;
const inputDocument = document.getElementById("input");
const sendBtn = document.querySelector("#chat-btn");
const selectElement = document.getElementById("model-select");
headers.append("Authorization", Bearer ${sessionProviderSpecific.key}
);
// 将模型放到下拉框中
sessionProvider.model.forEach(currentModel => {
const option = document.createElement("option");
option.selected = (currentModel = trimString(currentModel,"selected:")) === sessionProviderSpecific.defaultChooseModel; option.value = option.textContent = currentModel;
selectElement.appendChild(option); });
function chatClickAction() { openLoading(); inputDocument.value = '' // disable 按钮 sendBtn.disabled = true; }
function chat() { if(sendBtn.disabled) return;
const userInput = inputDocument.value; const messageId = "user-" + Date.now();
conversationHistory.push({ role: "user", content: userInput }); appendToConversationHistory("你", userInput, messageId); chatClickAction();
const requestOptions = { method: 'POST', headers, body: JSON.stringify({ model: selectElement.value, // 下拉框ai框架选择值 messages: conversationHistory, stream: true }), };
fetch(${sessionProviderSpecific.baseUrl}/chat/completions
, requestOptions)
.then(async response => {
const reader = response.body.getReader();
const decoder = new TextDecoder();
let partialMessage = ""; // Partial message from the stream
let aiMessage = ''
// 消息id,用于标识消息div
const messageId = "assistant-" + Date.now();
while (true) {
const { done, value } = await reader.read();
if (done) {
console.log("Stream complete");
conversationHistory.push({ role: "assistant", content: aiMessage });
document.querySelector("#chat-btn").disabled = false;
return;
}
if (isLoading) closeLoading();
const chunk = decoder.decode(value, { stream: true });
let messages = (partialMessage + chunk).split("\n");
partialMessage = messages.pop();
// 过滤数组里的空字符串
messages = messages.filter(message => message);
for (const message of messages) {
if (!message) continue; // 忽略空字符串
if (message === "data: [DONE]") {
break;
}
const jsonStartIndex = message.indexOf("{");
const jsonData = message.slice(jsonStartIndex);
const dataObject = JSON.parse(jsonData);
const aiResponse = dataObject.choices[0].delta.content;
if (aiResponse) {
aiMessage += aiResponse;
console.log(aiResponse);
appendToConversationHistory("AI", aiMessage, messageId);
}
}
}
})
.catch(error => console.log('error', error));
}
function appendToConversationHistory(role, content, messageElementId) { const historyContainer = document.getElementById("conversation-history");
let messageElement = document.getElementById(messageElementId); if (!messageElement) { messageElement = document.createElement("div"); messageElement.id = messageElementId; messageElement.classList.add(role + "-message"); historyContainer.appendChild(messageElement); }
let formattedContent = nextLineMedials(true,content);
// 检查内容中是否包含代码块标记
const codeBlockRegex = /(\w+)?([\s\S]*?)
|([^
]+)/g; formattedContent = formattedContent.replace(codeBlockRegex, (match, lang, blockContent, inlineContent) => { if (blockContent) { blockContent = trimString( nextLineMedials(false,blockContent,"\n"), "\n") return
${lang || "plaintext"}
${escapeHtml(blockContent)}
; } else if (inlineContent) { inlineContent = nextLineMedials(false,inlineContent,"\n") // 处理行内代码,将HTML标签字符转义 return
${escapeHtml(inlineContent)}
`;
}
return match;
});
formattedContent = nextLineMedials(false,formattedContent,"")
messageElement.innerHTML = role + " : " + formattedContent; }
// \n => 中间字符 =>
| \n
function nextLineMedials(isToMedials = true, content = "", target = "") {
if(isToMedials) {
return content.replace(/(\n)+/g, "[[$换行$]]")
} else {
return content.replace(/[[$换行$]]/g, target);
}
}
// 去除两边的\n
function trimNewlines(str) {
return str.replace(/^\n+|\n+$/g, '');
}
// 转义 HTML 标签字符(代码块特殊字符也直接显示,相当innerText)
function escapeHtml(str) {
return str.replace(/[&<>"'/]/g, function (match) {
const escapeMap = {
'&': '&',
'<': '<',
'>': '>',
'"': '"',
"'": ''',
'/': '/'
};
return escapeMap[match];
});
}
function openLoading() { isLoading = true; // 创建loading 节点 const loadingElement = document.createElement("div"); loadingElement.id = "loading"; loadingElement.style.display = "inline-block"; // 挂在到historyContainer document.getElementById("conversation-history").appendChild(loadingElement); }
function closeLoading() { isLoading = false; // 删除loading节点 document.getElementById("loading").remove(); }
// 给发送按钮绑定点击事件
sendBtn.addEventListener("click", () => chat());
// 监听msg push 事件(新版本使用MS_SCRIPT_ENV替换原来的MS_script_env_var,下面写有兼容代码)
(window.MS_SCRIPT_ENV || window.MS_script_env_var).event.sendListener.push((fillKeyword)=> {
console.log("已接收到:", fillKeyword);
if (${fillKeyword}
.trim().length === 0) return;
inputDocument.value = fillKeyword; // 手动设置值
chat(); // 点击send
});
// 回车发送事件
inputDocument.addEventListener('keypress', function(event) {
// 检查是否按下了 "Enter" 键
if (event.key === 'Enter') {
// 阻止默认行为(防止在表单中触发提交等)
event.preventDefault();
// 触发按钮点击
chat();
}
});