js 片段实现目录树自动展开 V0.0.2 版

看到论坛里有很多人表示希望有这个功能,今天也在看帖子讨论的过程中研究了一番,有一点思路,就试着实现了下。

由于刚接触思源,插件还不会写,api 也不熟,所以选择用 js 片段实现,基本是纯原生 js 实现。

主要功能: 记住上次已展开的目录,下次打开或刷新时自动展开上次已展开的目录。

实现原理:

仅记录已展开目录的 data-path, 笔记以 data-url 区分,以 object 嵌套的方式记录展开目录所在的层级关系。

加载时,会根据记录的目录层级关系,依次展开。

展开时,会记录目录的层级关系,如果不存在则添加。

折叠时,会根据目录的层级关系,删除对应的 Object 嵌套层级。

数据默认存储在/data/storage/tree-auto-extend.json 中,object 嵌套格式可参考这里的数据。

版本更新:

目前是 V0.0.2,支持 pc 客户端,web 版和手机版。但注意,pc 发布版是只读模式,无法写入数据,因此无法使用,请使用伺服版。

V0.0.2 更新内容: 更新了目录被删除或移动后可能报错的问题,兼容手机版和 web 版。

能力有限,这里仅仅是抛砖引玉,完善和优化还得靠各位大佬的鼎力支持!

完整代码如下:

// 目录树自动展开
// 功能:记住上次已展开的目录,下次打开或刷新时自动展开上次已展开的目录
// Version 0.0.2
// By Wilson 2024.7.24 晚,修改于 2024.7.26 晚
// MIT License
// 0.0.2 更新了目录被删除或移动后可能报错的问题,兼容手机版和web版

// 实现原理:
// 仅记录已展开目录的data-path, 笔记以data-url区分,以object嵌套的方式记录展开目录所在的层级关系
// 加载时,会根据记录的目录层级关系,依次展开
// 展开时,会记录目录的层级关系,如果不存在则添加
// 折叠时,会根据目录的层级关系,删除对应的Object嵌套层级
// 目前支持pc客户端,web版和手机版,但注意,pc发布版是只读模式,无法写入数据,因此无法使用,请使用伺服版。

// 配置存储路径(这个存储路径可以根据你自己的需要进行修改)
const storagePath = "/data/storage/tree-auto-extend.json";
// 是否启用自动定位当前文件所在的目录,启用后当前焦点文档所在的目录也会自动展开
const isShowFilePosition = true;

// 监听笔记列表渲染完成
whenElementExist("ul.b3-list[data-url]").then(async myElement => {
    // 防止更多笔记列表尚未渲染完成(笔记目录只渲染第一级,一般20毫秒足够了)
    await sleep(20);

    // 获取所有展开元素(这里变量命名有点问题,就这样吧)
    let collapsedEls = await getCollapsedEls();

    // 监听折叠事件
    document.querySelectorAll("ul.b3-list[data-url]").forEach(async book => {
        // 当前笔记的 data-url
        const bookDataUrl = book.getAttribute("data-url");

        // 自动展开
        autoExpandElements(collapsedEls[bookDataUrl], bookDataUrl);

        // 监听折叠事件
        book.addEventListener("click", async event => {
            //等待箭头按钮渲染完成
            // await sleep(40);
            let arrowBtn = event.target;
            if (event.target.tagName == "SPAN") {
                arrowBtn = event.target.firstElementChild;
            }
            // 如果是箭头按钮
            if (arrowBtn && arrowBtn.classList.contains("b3-list-item__arrow")) {
                //等待箭头按钮切换完成
                const currentClassList = arrowBtn.classList.toString();
                await whenElementExist(()=>{
                    return arrowBtn.classList.toString() !== currentClassList;
                });
                const isOpen = arrowBtn.classList.contains("b3-list-item__arrow--open");
                // 存储当前元素的折叠状态
                storeElement(collapsedEls, bookDataUrl, arrowBtn, isOpen);
            }
        });
    });

    // 定位当前打开的文档
    if(isShowFilePosition){
        await sleep(40);
        document.querySelector(".layout-tab-container .block__icons span[data-type=focus]")?.click();
    }
});
// 自动展元素
async function autoExpandElements(collapsedEls, bookDataUrl, path = []) {
    if (!collapsedEls) {
        return;
    }
    // 循环所有存储的展开元素并自动展开
    for (let key in collapsedEls) {
        if (key && collapsedEls.hasOwnProperty(key)) {
            const newPath = [...path, key];
            const joinPath = newPath.join('/');
            // 意外情况跳过
            if (joinPath === "") continue;
            const elPath = joinPath === "/" ? "/" : `/${joinPath.replace(/^\/+/, '')}.sy`;
            const elArrowBtn = document.querySelector("ul.b3-list[data-url='" + bookDataUrl + "'] li[data-path='" + elPath + "'] span.b3-list-item__toggle");
            if (elArrowBtn && !elArrowBtn.firstElementChild.classList.contains("b3-list-item__arrow--open")) {
                elArrowBtn.click();
                // 等待子元素渲染完毕
                //await sleep(40);
                await whenElementExist(()=>{
                    return elArrowBtn.closest("li").nextElementSibling?.tagName === 'UL';
                });
            }
            // 如果当前值也是一个对象,递归遍历
            if (typeof collapsedEls[key] === 'object' && collapsedEls[key] !== null && !Array.isArray(collapsedEls[key])) {
                autoExpandElements(collapsedEls[key], bookDataUrl, newPath);
            }
        }
    }
}
// 存储当前展开的元素
function storeElement(collapsedEls, bookDataUrl, arrowBtn, isOpen) {
    const nodePath = arrowBtn.parentElement.parentElement.getAttribute("data-path");
    const nodePathArr = nodePath === "/" ? ["/"] : nodePath.replace(/.sy$/i, '').split("/");
    nodePathArr[0] = nodePathArr[0] === "" ? "/" : nodePathArr[0];
    if (!collapsedEls[bookDataUrl]) {
        collapsedEls[bookDataUrl] = {};
    }
    if (isOpen) {
        // 如果是展开,则创建嵌套对象
        createNestedObject(nodePathArr, collapsedEls[bookDataUrl]);
    } else {
        // 如果是折叠,则删除最后一个key
        removeNestedKey(collapsedEls, bookDataUrl, nodePathArr);
    }
    // 存储展开元素的数据
    storeCollapsedEl(collapsedEls);
}
// 获取存储的展开元素(这里函数命名有点问题,就这样吧)
async function getCollapsedEls() {
    let data = await getFileContent(storagePath||"/data/storage/tree-auto-extend.json") || "{}";
    data = JSON.parse(data);
    if(data.code === 404) data = {};
    return data;
}
// 存储展开元素(这里函数命名有点问题,就这样吧)
function storeCollapsedEl(data) {
    putFileContent(storagePath||"/data/storage/tree-auto-extend.json", JSON.stringify(data))
}
// 创建嵌套对象
function createNestedObject(pathArray, objRef) {
    if (pathArray.length === 0) return; // 如果数组为空,退出递归
    const currentNode = pathArray.shift(); // 取出数组的第一个元素
    if (!objRef[currentNode]) {
        objRef[currentNode] = {}; // 创建一个新的空对象
    }
    if (pathArray.length > 0) {
        createNestedObject(pathArray, objRef[currentNode]); // 递归调用,传入新的路径数组和当前节点的子对象
    }
}
// 删除嵌套对象的最后一个key
function removeNestedKey(obj, urlKey, pathArr) {
    if (!obj || !urlKey || !pathArr || pathArr.length === 0) {
        return;
    }
    let currentLevel = obj[urlKey];
    if (!currentLevel) {
        return;
    }
    // Traverse through the object based on the path array
    for (let i = 0; i < pathArr.length - 1; i++) {
        const key = pathArr[i];
        if (!currentLevel[key]) {
            return;
        }
        currentLevel = currentLevel[key];
    }
    // Delete the last key specified in the path array
    const lastKeyToDelete = pathArr[pathArr.length - 1];
    if (lastKeyToDelete in currentLevel) {
        delete currentLevel[lastKeyToDelete];
    }
}
// 延迟执行
function sleep(ms) {
    return new Promise(resolve => setTimeout(resolve, ms));
}
// 等待元素渲染完成后执行
function whenElementExist(selector) {
    return new Promise(resolve => {
        const checkForElement = () => {
            let isExist = false;
            if (typeof selector === 'function') {
                isExist = selector();
            } else {
                isExist = document.querySelector(selector);
            }
            if (isExist) {
                resolve(true);
            } else {
                requestAnimationFrame(checkForElement);
            }
        };
        checkForElement();
    });
}
// 写入文件内容
async function putFileContent(path, content) {
    const formData = new FormData();
    formData.append("path", path);
    formData.append("file", new Blob([content]));
    return fetch("/api/file/putFile", {
        method: "POST",
        body: formData,
    })
    .then((response) => {
        if (response.ok) {
            //console.log("File saved successfully");
        }
        else {
            throw new Error("Failed to save file");
        }
    })
    .catch((error) => {
        console.error(error);
    });
}
// 获取文件内容
async function getFileContent(path) {
    return fetch("/api/file/getFile", {
        method: "POST",
        headers: {
            "Content-Type": "application/json",
        },
        body: JSON.stringify({
            path,
        }),
    })
    .then((response) => {
        if (response.ok) {
            return response.text();
        }
        else {
            throw new Error("Failed to get file content");
        }
    })
    .catch((error) => {
        console.error(error);
    });
}

使用方法:

  1. 设置 -> 外观 -> 代码片段 -> JS 中添加新的 js 代码片段即可。

  2. 输入上述代码到你新建的代码片段中

  3. 开启代码片段即可(如果有问题重启思源试试)

    image.png

注意: 如果你开启了 const isShowFilePosition = true,当前焦点文档所在的目录也会自动展开。


说明: 可能有时事情较多,可能无法及时看到大家的问题,不过在空闲时,会第一时间回复,请大家积极踊跃提出宝贵意见!


你觉得这个 js 片段怎么样?

单选 公开 永不结束 11 票
非常棒 👍
100% 11 票
一般般啦
0% 0 票
不好用
0% 0 票

  • 思源笔记

    思源笔记是一款隐私优先的个人知识管理系统,支持完全离线使用,同时也支持端到端加密同步。

    融合块、大纲和双向链接,重构你的思维。

    20644 引用 • 80672 回帖 • 1 关注
  • 插件
    85 引用 • 448 回帖 • 3 关注
6 操作
wilsons 在 2024-07-27 23:06:47 更新了该帖
wilsons 在 2024-07-27 19:10:51 更新了该帖
wilsons 在 2024-07-27 18:58:52 更新了该帖
wilsons 在 2024-07-27 17:04:21 更新了该帖 wilsons 在 2024-07-26 23:14:17 更新了该帖 wilsons 在 2024-07-24 06:29:40 更新了该帖

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...
  • EmberSky

    我试了下, 这个功能好像只有: 保存展开情况, 刷新后恢复

    但我感觉论坛讨论的还有几个需求

    1. 展开多级, 然后点击中间层级的折叠, 最后再点击展开, 最底层的还是展开的
    2. 新建文档之后, 不要折叠底层的文档

    不过思路很有意思, 我准备抄一抄, 周末按照我的需求, 改吧改吧

    1 回复
  • wilsons

    嗯嗯,期待ing,大家一起完善~

    或许这种实现方式不是最好的,但暂且可用。

    1 回复
  • 推荐楼主也在 https://sypai.cc/ 的 JS 板块发一篇分享

    1 回复
  • wilsons 1

    感谢建议!已发。

  • EmberSky 1 1 赞同

    哈哈哈, 黑猫白猫, 能抓老鼠的就是好猫

    我的文件数不多, 随便搞

  • daloo

    能够实现预想的功能,非常棒,向大佬和热心肠的同好致敬

  • EmberSky

    我研究了一下代码, 写的好骚啊, 看得出来, 非常专业

    image.png

    1 回复
  • wilsons

    不敢当,写的太匆忙,当时边写边构思,变量命名上也没认真思考,可读性差了点,可参考注释去理解。自动展开使用了递归,加上嵌套对象,可能有点不易理解。读代码通常比写代码的人理解起来更难,因为不清楚对方的思路,还可能因为命名,注释等被误导。

    我觉得可以根据原理自己去实现也许更容易些,毕竟目前实现方式也不是最好的,当时也没仔细构思,想到哪写到哪,以实现功能为目的。😄

  • wilsons

    各位大佬,我在 Mac 上用 pc 客户端开启发布服务后,在浏览器访问正常,输入账号认证也正常。

    但当用 js 访问 api 写入文件时却报错没权限,怎么回事?有大佬知道吗?如图,详细代码可参考楼主的帖子中的读写文件的代码和楼主说的已知问题。

    image.png

    但,手机端的发布服务并没有这个问题。

    好在 pc 端的发布服务不常用,毕竟有客户端了。

    不知道仅仅我遇到了这个问题,还是大家也有这个问题,有知道的,麻烦告知下,先谢谢啦!

  • wilsons

    pc 客户端开启发布服务后,当用 js 访问 api 写入文件时却报错没权限,怎么回事?

    已找到原因了,是因为 pc 端的发布服务是只读模式。

    image.png

    1 回复
  • 手机端的发布服务并没有这个问题

    那按理来说移动端的发布服务是一样的,不应该能写入啊

    1 回复
  • wilsons 1

    不一样,手机版并没有这个特别说明,而且手机端的发布服务可读写。

    我想可能官方认为 pc 版的发布一般是用来分享的,所以默认只读。

    其实,我觉得 pc 版可以给用户一个选择,是否只读模式,这样更好。

    记得之前看过一个帖子,手机端的媒体文件等快速传到 pc 版的思源,如果 pc 版发布支持写的话,估计那个问题就能解决了,毕竟通过局域网直接传到 pc 版,还是挺方便的。

    1 回复
  • 移动端目前没有发布服务功能,你用的那个可以写入的是伺服功能,桌面端也有。

    先记录 Issue #12113 · siyuan-note/siyuan

    2 回复
  • wilsons

    哦哦,明白了,之前没发现,还以为发布就是伺服呢,学习了!

  • wilsons

    请教下,这个怎么通过 192 开头的 ip 访问?好像只能以 127 开头的 ip 访问,192 无法访问。

    image.png

    1 回复
  • 哪个能连通,用那个就行了

    2 回复
  • wilsons

    主要是 127 开头的不能局域网访问,必须 192 开头的才行,有些场景能局域网访问还是很方便的。

  • wilsons

    可以了,重启电脑,又重新设置一遍好了。

    但感觉 pc 版的伺服,UI 不会自适应手机版;而手机版的伺服,UI 可以同时兼容手机版和 pc 版。

    1 回复
  • 不管是哪个客户端伺服都应该是一样的,因为代码是相同的

    1 回复
  • wilsons 1

    明白了,确实两者都能自适应,不过,必须从 url 根路径访问,比如:htttp://192.163.0.103:6806/

    我之前是因为,先 pc 打开地址,然后手机扫描二维码,那么地址就会变为:htttp://192.163.0.103:6806/stage/build/desktop/ ,然后,因为有了 desktop 后缀,这时候无论手机或 pc 访问都是访问的是桌面版的 UI,同理手机版也是,后缀是/mobile/。

    所以,手动改下后缀再访问也是一样的,比如:htttp://192.163.0.103:6806/stage/build/mobile/ 访问手机版。

请输入回帖内容 ...

推荐标签 标签

  • API

    应用程序编程接口(Application Programming Interface)是一些预先定义的函数,目的是提供应用程序与开发人员基于某软件或硬件得以访问一组例程的能力,而又无需访问源码,或理解内部工作机制的细节。

    76 引用 • 429 回帖 • 5 关注
  • Netty

    Netty 是一个基于 NIO 的客户端-服务器编程框架,使用 Netty 可以让你快速、简单地开发出一个可维护、高性能的网络应用,例如实现了某种协议的客户、服务端应用。

    49 引用 • 33 回帖 • 21 关注
  • 思源笔记

    思源笔记是一款隐私优先的个人知识管理系统,支持完全离线使用,同时也支持端到端加密同步。

    融合块、大纲和双向链接,重构你的思维。

    20643 引用 • 80672 回帖 • 1 关注
  • MongoDB

    MongoDB(来自于英文单词“Humongous”,中文含义为“庞大”)是一个基于分布式文件存储的数据库,由 C++ 语言编写。旨在为应用提供可扩展的高性能数据存储解决方案。MongoDB 是一个介于关系数据库和非关系数据库之间的产品,是非关系数据库当中功能最丰富,最像关系数据库的。它支持的数据结构非常松散,是类似 JSON 的 BSON 格式,因此可以存储比较复杂的数据类型。

    90 引用 • 59 回帖
  • V2EX

    V2EX 是创意工作者们的社区。这里目前汇聚了超过 400,000 名主要来自互联网行业、游戏行业和媒体行业的创意工作者。V2EX 希望能够成为创意工作者们的生活和事业的一部分。

    17 引用 • 236 回帖 • 371 关注
  • OpenResty

    OpenResty 是一个基于 NGINX 与 Lua 的高性能 Web 平台,其内部集成了大量精良的 Lua 库、第三方模块以及大多数的依赖项。用于方便地搭建能够处理超高并发、扩展性极高的动态 Web 应用、Web 服务和动态网关。

    17 引用 • 45 关注
  • 持续集成

    持续集成(Continuous Integration)是一种软件开发实践,即团队开发成员经常集成他们的工作,通过每个成员每天至少集成一次,也就意味着每天可能会发生多次集成。每次集成都通过自动化的构建(包括编译,发布,自动化测试)来验证,从而尽早地发现集成错误。

    14 引用 • 7 回帖
  • Webswing

    Webswing 是一个能将任何 Swing 应用通过纯 HTML5 运行在浏览器中的 Web 服务器,详细介绍请看 将 Java Swing 应用变成 Web 应用

    1 引用 • 15 回帖 • 621 关注
  • PostgreSQL

    PostgreSQL 是一款功能强大的企业级数据库系统,在 BSD 开源许可证下发布。

    22 引用 • 22 回帖
  • TensorFlow

    TensorFlow 是一个采用数据流图(data flow graphs),用于数值计算的开源软件库。节点(Nodes)在图中表示数学操作,图中的线(edges)则表示在节点间相互联系的多维数据数组,即张量(tensor)。

    20 引用 • 19 回帖 • 3 关注
  • 开源

    Open Source, Open Mind, Open Sight, Open Future!

    405 引用 • 3557 回帖
  • CAP

    CAP 指的是在一个分布式系统中, Consistency(一致性)、 Availability(可用性)、Partition tolerance(分区容错性),三者不可兼得。

    11 引用 • 5 回帖 • 590 关注
  • 禅道

    禅道是一款国产的开源项目管理软件,她的核心管理思想基于敏捷方法 scrum,内置了产品管理和项目管理,同时又根据国内研发现状补充了测试管理、计划管理、发布管理、文档管理、事务管理等功能,在一个软件中就可以将软件研发中的需求、任务、bug、用例、计划、发布等要素有序的跟踪管理起来,完整地覆盖了项目管理的核心流程。

    6 引用 • 15 回帖 • 169 关注
  • Flutter

    Flutter 是谷歌的移动 UI 框架,可以快速在 iOS 和 Android 上构建高质量的原生用户界面。 Flutter 可以与现有的代码一起工作,它正在被越来越多的开发者和组织使用,并且 Flutter 是完全免费、开源的。

    39 引用 • 92 回帖 • 8 关注
  • Hprose

    Hprose 是一款先进的轻量级、跨语言、跨平台、无侵入式、高性能动态远程对象调用引擎库。它不仅简单易用,而且功能强大。你无需专门学习,只需看上几眼,就能用它轻松构建分布式应用系统。

    9 引用 • 17 回帖 • 606 关注
  • App

    App(应用程序,Application 的缩写)一般指手机软件。

    90 引用 • 383 回帖
  • 倾城之链
    23 引用 • 66 回帖 • 125 关注
  • CentOS

    CentOS(Community Enterprise Operating System)是 Linux 发行版之一,它是来自于 Red Hat Enterprise Linux 依照开放源代码规定释出的源代码所编译而成。由于出自同样的源代码,因此有些要求高度稳定的服务器以 CentOS 替代商业版的 Red Hat Enterprise Linux 使用。两者的不同在于 CentOS 并不包含封闭源代码软件。

    238 引用 • 224 回帖 • 2 关注
  • FreeMarker

    FreeMarker 是一款好用且功能强大的 Java 模版引擎。

    23 引用 • 20 回帖 • 444 关注
  • C

    C 语言是一门通用计算机编程语言,应用广泛。C 语言的设计目标是提供一种能以简易的方式编译、处理低级存储器、产生少量的机器码以及不需要任何运行环境支持便能运行的编程语言。

    83 引用 • 165 回帖 • 3 关注
  • danl
    98 关注
  • Flume

    Flume 是一套分布式的、可靠的,可用于有效地收集、聚合和搬运大量日志数据的服务架构。

    9 引用 • 6 回帖 • 616 关注
  • 分享

    有什么新发现就分享给大家吧!

    246 引用 • 1780 回帖
  • HTML

    HTML5 是 HTML 下一个的主要修订版本,现在仍处于发展阶段。广义论及 HTML5 时,实际指的是包括 HTML、CSS 和 JavaScript 在内的一套技术组合。

    103 引用 • 294 回帖
  • RYMCU

    RYMCU 致力于打造一个即严谨又活泼、专业又不失有趣,为数百万人服务的开源嵌入式知识学习交流平台。

    4 引用 • 6 回帖 • 42 关注
  • OpenShift

    红帽提供的 PaaS 云,支持多种编程语言,为开发人员提供了更为灵活的框架、存储选择。

    14 引用 • 20 回帖 • 617 关注
  • Logseq

    Logseq 是一个隐私优先、开源的知识库工具。

    Logseq is a joyful, open-source outliner that works on top of local plain-text Markdown and Org-mode files. Use it to write, organize and share your thoughts, keep your to-do list, and build your own digital garden.

    5 引用 • 62 回帖 • 10 关注