[js] 快捷键移动列表节点

image.png

各位以后要注意先看编辑选项里有没有想要的功能 😂

没仔细看直接想当然的让 AI 搓了 10 分钟搓了个弱化版,当反向示例了这下doge


类似代码编辑器中移动代码行

移动块演示.gif

可能有神奇的特性,个人暂时只打算用于列表节点的临时交换。

(function() { const listSwapper = { // 获取当前文档编辑区 getActiveEditor: function() { return document.querySelector('.protyle:not(.fn__none) .protyle-wysiwyg'); }, // 判断节点是否为列表项 isListItem: function(node) { return node && node.classList && node.classList.contains('li'); }, // 判断节点是否为列表容器 isList: function(node) { return node && node.classList && node.classList.contains('list'); }, // 从当前选择的节点查找最近的列表项节点 findListItemNode: function(node) { // 如果是文本节点,先获取其父元素 if (node.nodeType === 3) { node = node.parentElement; } // 向上查找,直到找到列表项节点 while (node) { if (this.isListItem(node)) { return node; } node = node.parentElement; // 如果超出编辑区范围则停止 if (!node || node.classList.contains('protyle-wysiwyg')) { return null; } } return null; }, // 获取同级的前一个列表项 getPreviousListItem: function(listItem) { if (!this.isListItem(listItem)) return null; // 直接获取前一个兄弟元素 let prev = listItem.previousElementSibling; // 确保是列表项 if (prev && this.isListItem(prev)) { return prev; } return null; }, // 获取同级的后一个列表项 getNextListItem: function(listItem) { if (!this.isListItem(listItem)) return null; // 直接获取后一个兄弟元素 let next = listItem.nextElementSibling; // 确保是列表项 if (next && this.isListItem(next)) { return next; } return null; }, // 交换两个列表项节点的位置 swapListItems: function(item1, item2) { if (!item1 || !item2 || !this.isListItem(item1) || !this.isListItem(item2)) { return false; } try { const parent = item1.parentNode; // 确保两个节点在同一个父列表中 if (parent !== item2.parentNode) { console.log('列表项不在同一个父容器中,无法交换'); return false; } // 保存当前项的ID用于后续恢复焦点 const itemId = item1.getAttribute('data-node-id'); // 判断相对位置并交换 const next = item1.nextElementSibling; if (item2 === next) { // item2紧跟在item1后面,直接调整顺序 parent.insertBefore(item2, item1); } else { // 创建节点引用的副本,用于判断顺序 const item1Clone = item1; const item2Clone = item2; // 标记item2的下一个节点 const item2Next = item2.nextElementSibling; // 如果item1在item2之前 if (Array.from(parent.children).indexOf(item1Clone) < Array.from(parent.children).indexOf(item2Clone)) { // 将item2移到item1之前 parent.insertBefore(item2, item1); // 将item1移到item2原来的位置 if (item2Next) { parent.insertBefore(item1, item2Next); } else { parent.appendChild(item1); } } else { // item2在item1之前 // 先将item1移到item2的位置 parent.insertBefore(item1, item2); // 再将item2移到item1原来的位置 if (next) { parent.insertBefore(item2, next); } else { parent.appendChild(item2); } } } console.log('列表项交换成功'); return itemId; } catch (e) { console.error('交换列表项失败:', e); return false; } }, // 向上移动列表项 moveListItemUp: function() { // 获取当前选中内容所在的列表项 const selection = window.getSelection(); if (!selection.rangeCount) return; const listItem = this.findListItemNode(selection.getRangeAt(0).startContainer); if (!listItem) { console.log('当前光标不在列表项内'); return; } console.log('当前列表项:', listItem); // 获取同级的前一个列表项 const prevItem = this.getPreviousListItem(listItem); if (!prevItem) { console.log('已经是第一个列表项,无法上移'); return; } console.log('前一个列表项:', prevItem); // 交换列表项 const itemId = this.swapListItems(listItem, prevItem); // 恢复焦点 if (itemId) { setTimeout(() => { this.focusListItemById(itemId); }, 10); } }, // 向下移动列表项 moveListItemDown: function() { // 获取当前选中内容所在的列表项 const selection = window.getSelection(); if (!selection.rangeCount) return; const listItem = this.findListItemNode(selection.getRangeAt(0).startContainer); if (!listItem) { console.log('当前光标不在列表项内'); return; } console.log('当前列表项:', listItem); // 获取同级的后一个列表项 const nextItem = this.getNextListItem(listItem); if (!nextItem) { console.log('已经是最后一个列表项,无法下移'); return; } console.log('后一个列表项:', nextItem); // 交换列表项 const itemId = this.swapListItems(listItem, nextItem); // 恢复焦点 if (itemId) { setTimeout(() => { this.focusListItemById(itemId); }, 10); } }, // 通过ID聚焦到列表项 focusListItemById: function(itemId) { if (!itemId) return; // 查找具有特定ID的列表项 const item = document.querySelector(`[data-node-id="${itemId}"]`); if (!item) return; // 查找列表项中的可编辑段落 const paragraph = item.querySelector('[contenteditable="true"]'); if (!paragraph) return; // 聚焦并将光标设置到段落末尾 paragraph.focus(); const range = document.createRange(); const selection = window.getSelection(); // 将光标设置到段落内容末尾 if (paragraph.childNodes.length > 0) { const lastChild = paragraph.childNodes[paragraph.childNodes.length - 1]; if (lastChild.nodeType === Node.TEXT_NODE) { range.setStart(lastChild, lastChild.length); range.setEnd(lastChild, lastChild.length); } else { range.selectNodeContents(lastChild); range.collapse(false); } } else { range.selectNodeContents(paragraph); range.collapse(false); } selection.removeAllRanges(); selection.addRange(range); }, // 初始化键盘事件监听 init: function() { document.addEventListener('keydown', (event) => { // 检查是否是Alt+箭头组合键 if (event.altKey && !event.ctrlKey && !event.shiftKey && !event.metaKey) { // Alt + 上箭头:向上移动列表项 if (event.key === 'ArrowUp') { event.preventDefault(); this.moveListItemUp(); return false; } // Alt + 下箭头:向下移动列表项 else if (event.key === 'ArrowDown') { event.preventDefault(); this.moveListItemDown(); return false; } } }); console.log('思源笔记列表项交换脚本已加载,使用 Alt+↑/↓ 来交换列表项位置'); } }; // 初始化脚本 listSwapper.init(); })();
  • 思源笔记

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

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

    26373 引用 • 109677 回帖
  • 代码片段

    代码片段分为 CSS 与 JS 两种代码,添加在 [设置 - 外观 - 代码片段] 中,这些代码会在思源笔记加载时自动执行,用于改善笔记的样式或功能。

    用户在该标签下分享代码片段时需在帖子标题前添加 [css] [js] 用于区分代码片段类型。

    204 引用 • 1477 回帖
1 操作
xqh042 在 2025-03-23 16:30:31 更新了该帖

相关帖子

欢迎来到这里!

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

注册 关于
请输入回帖内容 ...
  • 话说代码片段除了不会保存,跟思源内置的功能有什么区别吗?

    image.png

    1 回复
  • xqh042

    甚至还更弱了 😂 主打一个没仔细看设置乱折腾


    好吧,其实我还是搜了一下设置的,但是关键词(移动、块)都没命中,就以为没这功能了doge

推荐标签 标签

  • Linux

    Linux 是一套免费使用和自由传播的类 Unix 操作系统,是一个基于 POSIX 和 Unix 的多用户、多任务、支持多线程和多 CPU 的操作系统。它能运行主要的 Unix 工具软件、应用程序和网络协议,并支持 32 位和 64 位硬件。Linux 继承了 Unix 以网络为核心的设计思想,是一个性能稳定的多用户网络操作系统。

    955 引用 • 944 回帖
  • 七牛云

    七牛云是国内领先的企业级公有云服务商,致力于打造以数据为核心的场景化 PaaS 服务。围绕富媒体场景,七牛先后推出了对象存储,融合 CDN 加速,数据通用处理,内容反垃圾服务,以及直播云服务等。

    29 引用 • 230 回帖 • 124 关注
  • 书籍

    宋真宗赵恒曾经说过:“书中自有黄金屋,书中自有颜如玉。”

    84 引用 • 414 回帖
  • NetBeans

    NetBeans 是一个始于 1997 年的 Xelfi 计划,本身是捷克布拉格查理大学的数学及物理学院的学生计划。此计划延伸而成立了一家公司进而发展这个商用版本的 NetBeans IDE,直到 1999 年 Sun 买下此公司。Sun 于次年(2000 年)六月将 NetBeans IDE 开源,直到现在 NetBeans 的社群依然持续增长。

    78 引用 • 102 回帖 • 713 关注
  • Bug

    Bug 本意是指臭虫、缺陷、损坏、犯贫、窃听器、小虫等。现在人们把在程序中一些缺陷或问题统称为 bug(漏洞)。

    76 引用 • 1742 回帖 • 1 关注
  • HTML

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

    108 引用 • 295 回帖 • 2 关注
  • 工具

    子曰:“工欲善其事,必先利其器。”

    300 引用 • 768 回帖
  • 叶归
    13 引用 • 59 回帖 • 22 关注
  • Vim

    Vim 是类 UNIX 系统文本编辑器 Vi 的加强版本,加入了更多特性来帮助编辑源代码。Vim 的部分增强功能包括文件比较(vimdiff)、语法高亮、全面的帮助系统、本地脚本(Vimscript)和便于选择的可视化模式。

    29 引用 • 66 回帖
  • 倾城之链
    23 引用 • 66 回帖 • 167 关注
  • 大疆创新

    深圳市大疆创新科技有限公司(DJI-Innovations,简称 DJI),成立于 2006 年,是全球领先的无人飞行器控制系统及无人机解决方案的研发和生产商,客户遍布全球 100 多个国家。通过持续的创新,大疆致力于为无人机工业、行业用户以及专业航拍应用提供性能最强、体验最佳的革命性智能飞控产品和解决方案。

    2 引用 • 14 回帖
  • 游戏

    沉迷游戏伤身,强撸灰飞烟灭。

    187 引用 • 832 回帖
  • Google

    Google(Google Inc.,NASDAQ:GOOG)是一家美国上市公司(公有股份公司),于 1998 年 9 月 7 日以私有股份公司的形式创立,设计并管理一个互联网搜索引擎。Google 公司的总部称作“Googleplex”,它位于加利福尼亚山景城。Google 目前被公认为是全球规模最大的搜索引擎,它提供了简单易用的免费服务。不作恶(Don't be evil)是谷歌公司的一项非正式的公司口号。

    49 引用 • 192 回帖
  • Vditor

    Vditor 是一款浏览器端的 Markdown 编辑器,支持所见即所得、即时渲染(类似 Typora)和分屏预览模式。它使用 TypeScript 实现,支持原生 JavaScript、Vue、React 和 Angular。

    374 引用 • 1859 回帖
  • CodeMirror
    2 引用 • 17 回帖 • 173 关注
  • Sphinx

    Sphinx 是一个基于 SQL 的全文检索引擎,可以结合 MySQL、PostgreSQL 做全文搜索,它可以提供比数据库本身更专业的搜索功能,使得应用程序更容易实现专业化的全文检索。

    1 引用 • 225 关注
  • 钉钉

    钉钉,专为中国企业打造的免费沟通协同多端平台, 阿里巴巴出品。

    15 引用 • 67 回帖 • 266 关注
  • Ubuntu

    Ubuntu(友帮拓、优般图、乌班图)是一个以桌面应用为主的 Linux 操作系统,其名称来自非洲南部祖鲁语或豪萨语的“ubuntu”一词,意思是“人性”、“我的存在是因为大家的存在”,是非洲传统的一种价值观,类似华人社会的“仁爱”思想。Ubuntu 的目标在于为一般用户提供一个最新的、同时又相当稳定的主要由自由软件构建而成的操作系统。

    127 引用 • 169 回帖
  • 支付宝

    支付宝是全球领先的独立第三方支付平台,致力于为广大用户提供安全快速的电子支付/网上支付/安全支付/手机支付体验,及转账收款/水电煤缴费/信用卡还款/AA 收款等生活服务应用。

    29 引用 • 347 回帖
  • Dubbo

    Dubbo 是一个分布式服务框架,致力于提供高性能和透明化的 RPC 远程服务调用方案,是 [阿里巴巴] SOA 服务化治理方案的核心框架,每天为 2,000+ 个服务提供 3,000,000,000+ 次访问量支持,并被广泛应用于阿里巴巴集团的各成员站点。

    60 引用 • 82 回帖 • 617 关注
  • PostgreSQL

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

    22 引用 • 22 回帖 • 3 关注
  • 以太坊

    以太坊(Ethereum)并不是一个机构,而是一款能够在区块链上实现智能合约、开源的底层系统。以太坊是一个平台和一种编程语言 Solidity,使开发人员能够建立和发布下一代去中心化应用。 以太坊可以用来编程、分散、担保和交易任何事物:投票、域名、金融交易所、众筹、公司管理、合同和知识产权等等。

    34 引用 • 367 回帖 • 1 关注
  • 星云链

    星云链是一个开源公链,业内简单的将其称为区块链上的谷歌。其实它不仅仅是区块链搜索引擎,一个公链的所有功能,它基本都有,比如你可以用它来开发部署你的去中心化的 APP,你可以在上面编写智能合约,发送交易等等。3 分钟快速接入星云链 (NAS) 测试网

    3 引用 • 16 回帖 • 2 关注
  • ActiveMQ

    ActiveMQ 是 Apache 旗下的一款开源消息总线系统,它完整实现了 JMS 规范,是一个企业级的消息中间件。

    19 引用 • 13 回帖 • 683 关注
  • SOHO

    为成为自由职业者在家办公而努力吧!

    7 引用 • 55 回帖 • 3 关注
  • 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.

    7 引用 • 69 回帖 • 5 关注
  • 印象笔记
    3 引用 • 16 回帖 • 2 关注