// ==UserScript==
// @name 在必应(Bing)中搜索思源(SiYuan)
// @namespace https://fradeet.top/
// @supportURL https://github.com/Fradeet/SearchEngine-with-SiYuan-serach
// @version 2025-02-14
// @description 在搜索引擎侧栏(目前是必应)展示相同的关键词在思源笔记中的结果。
// @author Fradeet
// @license MIT
// @icon 
// @match https://*.bing.com/search*
// @connect 127.0.0.1
// @connect *
// @grant GM_setValue
// @grant GM_getValue
// @grant GM_xmlhttpRequest
// @downloadURL none
// ==/UserScript==
(function () {
'use strict';
// console.log("GM_info", GM_info);
// 获取本地存储配置
// console.log("GM_getValue", GM_getValue("config"));
let config = JSON.parse(GM_getValue("config", `{
"Location": "local",
"SiYuan": {
"Endpoint": "http://127.0.0.1:6806/",
"Token": ""
}
}`));
// 插入样式
const style = document.createElement('style');
style.innerHTML = `
#siyuan-search {
margin-top: 1em;
}
#siyuan-search-list {
list-style: none;
padding: 0;
margin: 0;
}
#siyuan-search-list li {
margin-bottom: 0.5em;
}
#siyuan-search li:hover {
background-color: #f5f5f5;
}
#siyuan-search li a {
text-decoration: none;
color: #0078d7;
}
#siyuan-search li a:hover {
text-decoration: underline;
}
#donate-button {
margin-left: 1em;
}
#setting-button {
margin-left: 1em;
}
.siyuan-search-info {
display: flex;
}
.siyuan-updated {
margin-left: auto;
}
.siyuan-search-item {
background-color: white;
border-radius: 5px;
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
margin-bottom: 10px;
padding: 10px;
position: relative;
}
.siyuan-search-title {
display: -webkit-box;
-webkit-box-orient: vertical;
-webkit-line-clamp: 2; /* 设置你想要的行数 */
overflow: hidden;
margin-bottom: 0.5em;
}
.siyuan-hpath {
background-color: rgba(55, 53, 47, 0.06);
border-radius: 1em;
max-width: 70%;
white-space: nowrap; /* 防止文本换行 */
overflow: hidden; /* 隐藏溢出的文本 */
text-overflow: ellipsis; /* 使用省略号表示被隐藏的文本 */
direction: rtl; /* 文本从右到左排列 */
text-align: left; /* 将文本对齐方式设置为左对齐 */
font-size: 0.8em;
}
#siyuan-search-header img {
width: 2em;
height: 2em;
align-items: center; /* 垂直居中 */
}
#siyuan-search-header div {
align-items: center; /* 垂直居中 */
}
#siyuan-related-list li {
margin-bottom: 0.5em;
}
`;
document.head.appendChild(style);
// 创建一个 li
let li = document.createElement('li');
li.style.cssText = 'margin-bottom: 2em;';
li.innerHTML = `
`;
const sidePanel = document.getElementById("b_context");
sidePanel.insertBefore(li, sidePanel.firstChild);
const title_ul = document.querySelector("#siyuan-search-list")
const related_ul = document.querySelector("#siyuan-related-list")
let setting_button = document.getElementById("setting-button");
let setting = document.querySelector("#siyuan-setting");
setting_button.onclick = () => {
setting.style.display = 'inline';
}
// 将现有配置填入
if (config.SiYuan !== undefined) {
if (config.Location === "local") {
document.getElementById("location").value = "local";
document.getElementById("url").disabled = true;
} else {
document.getElementById("location").value = "remote";
}
document.getElementById("url").value = config.SiYuan.Endpoint;
document.getElementById("token").value = config.SiYuan.Token;
}
// TODO 选择远程的时候弹出提示,提示可能需要 Tampermonkey 允许跨域
// 监听设置,如果是本地则服务器输入框禁用
document.getElementById("location").onchange = () => {
if (document.getElementById("location").value === "local") {
document.getElementById("url").disabled = true;
} else {
document.getElementById("url").disabled = false;
}
}
document.getElementById("save-button").onclick = () => {
let location = document.getElementById("location").value;
if (location === "local") {
config.Location = "local";
config.SiYuan.Endpoint = "http://127.0.0.1:6806";
} else {
config.Location = "remote";
config.SiYuan.Endpoint = document.getElementById("url").value;
}
config.SiYuan.Token = document.getElementById("token").value;
GM_setValue("config", JSON.stringify(config));
document.getElementById("setting-log").innerHTML = `
保存成功,刷新生效。
`;
setTimeout(() => {
document.getElementById("setting-log").innerHTML = '';
}, 2000);
}
const ErrorPage = (code) => {
document.getElementById("middle-line").style.display = 'none';
title_ul.appendChild(document.createTextNode(`连接 SiYuan 失败。${code ? `错误码: ${code}` : ''}`));
}
if (config.SiYuan.Token !== "") {
console.log("[SiYuan Search] CheckConnect");
GM_xmlhttpRequest({
method: 'POST',
url: config.SiYuan.Endpoint + '/api/system/version',
responseType: 'json',
onload: function (res) {
if (res.status == 200 && res.response.code === 0) {
console.log("[SiYuan Search] Connect success, SiYuan version: ", res.response.data);
// 获取搜索框内容
const searchInput = document.getElementById("sb_form_q").value;
// 可显示笔记块
const sql2 = `SELECT id, content, hpath, updated, root_id
FROM blocks
WHERE content
LIKE '%${searchInput}%'
ORDER BY
sort ASC,
updated DESC;`
// SQL 搜索
GM_xmlhttpRequest({
method: 'POST',
url: config.SiYuan.Endpoint + '/api/query/sql',
data: JSON.stringify({
"stmt": sql2
}),
Headers: {
"Authorization": "Token " + config.SiYuan.Token
},
responseType: 'json',
onload: function (res) {
if (res.status === 200 && res.response.code === 0) {
console.log(`[SiYuan Search] Search '${searchInput}' success, result: `, res.response.data);
// 以标题成功匹配的列表
let note_list = [];
// 以块成功匹配的对象,用于存放查询到的笔记标题
let block_list = [];
// 渲染页面
res.response.data.forEach((e) => {
let li = document.createElement('li');
// 判断是否文档块
if (e.root_id === e.id) {
li.classList.add('siyuan-search-item');
note_list.push(e.root_id);
if (config.Location === "local") {
li.innerHTML = `
${e.content}
时间:${e.updated.substring(0, 8)}
`;
} else if (config.Location === "remote") {
li.innerHTML = `
${e.content}
时间:${e.updated.substring(0, 8)}
`;
}
title_ul.appendChild(li);
} else {
if (note_list.includes(e.root_id) === false && block_list.includes(e.root_id) === false) {
// 获取块的笔记名字
GM_xmlhttpRequest({
method: 'POST',
url: config.SiYuan.Endpoint + '/api/attr/getBlockAttrs',
data: JSON.stringify({
"id": e.root_id,
}),
Headers: {
"Authorization": "Token " + config.SiYuan.Token
},
responseType: 'json',
onload: function (res) {
if (res.status === 200 && res.response.code === 0) {
// 渲染到页面
li.innerHTML = `
${res.response.data.title}
`;
related_ul.appendChild(li);
}
},
});
// 标记这个笔记已经显示了,放里面会有异步
block_list.push(e.root_id);
}
}
});
} else {
console.error("[SiYuan Search] Response issue: ", res);
ErrorPage(res.response.code);
}
},
onerror: function () {
console.error("[SiYuan Search] Search failed.");
}
});
} else {
console.error("[SiYuan Search] Response issue: ", res);
ErrorPage(res.response.code);
}
},
onerror: function () {
console.error("[SiYuan Search] Connect failed.");
ErrorPage();
}
});
} else {
console.log("[SiYuan Search] Please configure the connection information.");
// 展开设置页面
setting.style.display = 'inline';
document.getElementById("middle-line").style.display = 'none';
title_ul.appendChild(document.createTextNode(`请先登录。`));
}
})();