diff --git a/plugin.json b/plugin.json index 1478472..7a4b96d 100644 --- a/plugin.json +++ b/plugin.json @@ -2,7 +2,7 @@ "name": "siyuan-plugin-hsr-mdzz2048-fork", "author": "TCOTC", "url": "https://github.com/TCOTC/siyuan-plugin-hsr-mdzz2048-fork", - "version": "0.4.5", + "version": "0.4.6", "minAppVersion": "2.9.8", "backends": ["windows", "linux", "darwin"], "frontends": ["desktop", "browser-desktop"], diff --git a/src/Search.vue b/src/Search.vue index 77855ad..79fcd8f 100644 --- a/src/Search.vue +++ b/src/Search.vue @@ -4,6 +4,7 @@ 28.0, 可以使用这个 API) -function highlightHitResult(value: string, change: boolean) { // 搜索并高亮结果 +function highlightHitResult0(value: string, change: boolean) { // 搜索并高亮结果 // 如果文本框内容改变,搜索结果和索引计数都立刻清零 if (change == true) { @@ -152,6 +153,85 @@ function highlightHitResult(value: string, change: boolean) { // 搜索并高亮 // 滚动页面 // scroollIntoRanges(resultIndex.value) } +function highlightHitResult(value: string, change: boolean) { // 搜索并高亮结果 + + // 如果文本框内容改变,搜索结果和索引计数都立刻清零 + if (change == true) { + resultIndex.value = 0 + resultCount.value = 0 + } + + // 首先,选取所有符合条件的元素 + // const elements = document.querySelectorAll('.layout-tab-container > div:not(.fn__none) .protyle-wysiwyg [data-node-id]'); + // 获取文档根,后续直接对全文档文本进行搜索, + const docRoot = document.querySelector('.layout-tab-container > div:not(.fn__none) .protyle-wysiwyg') as HTMLElement; + const docText=docRoot.textContent.toLowerCase(); + const docLen=docText.length; + + // 准备一个数组来保存所有文本节点 + const allTextNodes = []; + let incr_lens = []; + let cur_len0=0; + + const treeWalker = document.createTreeWalker(docRoot, NodeFilter.SHOW_TEXT); + let currentNode = treeWalker.nextNode(); + while (currentNode) { + allTextNodes.push(currentNode); + cur_len0+=currentNode.textContent.length + incr_lens.push(cur_len0); + currentNode = treeWalker.nextNode(); + } + + // 清除上个高亮 + CSS.highlights.clear() + + // 为空判断 + const str = value.trim().toLowerCase() + if (!str) return + let textNodeCnt=allTextNodes.length + let cur_nodeIdx=0; + let txtNode + // 查找所有文本节点是否包含搜索词,并创建对应的 Range 对象 + // 把全局匹配索引转换为文本节点的索引和offset,使得range可以跨多个文本节点 + let ranges = []; + let startIndex = 0; + let endIndex=0; + while ((startIndex = docText.indexOf(str, startIndex)) !== -1) { + const range = document.createRange(); + endIndex=startIndex + str.length + // console.log(`开始结束索引:${startIndex}-${endIndex}`) + try { + while (cur_nodeIdx