[js] 在面包屑旁添加全屏按钮,快捷切换全屏

本贴最后更新于 274 天前,其中的信息可能已经时移世改

效果是在面包屑旁添加一个全屏按钮,快捷切换页签的全屏状态:

JS 片段:

// 在面包屑旁添加全屏按钮,快捷切换全屏 JS片段 // author by JeffreyChen https://ld246.com/article/1731698559408 // 参考 @wilsons 的方法进行了优化 https://ld246.com/article/1731683038390/comment/1731693866270#comments ,不再需要全屏切换快捷键 (function() { function addFullScreenButton(protyleElement) { // 检查该 protyle 是否已经有了 fullScreen_simulate 按钮 if (protyleElement.querySelector('.fullScreen_simulate')) { return; // 如果已存在,直接返回 } let mode = protyleElement.querySelector('.protyle-breadcrumb .block__icon[data-type="readonly"]'); if (mode) { mode.insertAdjacentHTML( "beforebegin", '<button class="fullScreen_simulate block__icon fn__flex-center ariaLabel" aria-label="全屏切换"></button>' ); let fullScreenBtn = protyleElement.querySelector(".fullScreen_simulate"); fullScreenBtn.innerHTML = `<svg><use xlink:href="#iconFullscreen"></use></svg>`; fullScreenBtn.addEventListener("click", function (e) { // 获取 .layout-tab-container > .protyle .protyle-breadcrumb__space 元素 const breadcrumbSpace = protyleElement.querySelector('.protyle-breadcrumb__space'); // 如果元素存在,则模拟点击,聚焦当前页签 if (breadcrumbSpace) { breadcrumbSpace.click(); } toggleFullScreen(protyleElement, fullScreenBtn); // 切换全屏状态 }); } } // 切换全屏状态的函数 function toggleFullScreen(protyle, fullScreenBtn) { if (!window.siyuan.editorIsFullscreen) { enterFullScreen(protyle, fullScreenBtn); } else { exitFullScreen(protyle, fullScreenBtn); } } function enterFullScreen(protyle, fullScreenBtn) { protyle.classList.add("fullscreen"); window.siyuan.editorIsFullscreen = true; updateFullScreenButton(fullScreenBtn, true); // 更新按钮 } function exitFullScreen(protyle, fullScreenBtn) { protyle.classList.remove("fullscreen"); window.siyuan.editorIsFullscreen = false; updateFullScreenButton(fullScreenBtn, false); // 更新按钮 } function updateFullScreenButton(fullScreenBtn, isFullScreen) { const iconUse = fullScreenBtn.querySelector('use'); // 切换图标 iconUse.setAttribute('xlink:href', isFullScreen ? '#iconFullscreenExit' : '#iconFullscreen'); fullScreenBtn.setAttribute('aria-label', isFullScreen ? '退出全屏' : '全屏'); } // 定期检查 .layout__center 是否存在于 DOM 中 function checkForLayoutCenter() { const targetNode = document.querySelector('.layout__center'); if (targetNode) { startObserving(targetNode); // 立即检查一次 protyle-breadcrumb checkProtyleElements(targetNode); } else { setTimeout(checkForLayoutCenter, 200); } } function checkProtyleElements(targetNode) { const protyles = targetNode.querySelectorAll('.layout-tab-container > .protyle'); protyles.forEach(protyle => { addFullScreenButton(protyle); }); } function startObserving(targetNode) { const observer = new MutationObserver((mutations) => { mutations.forEach(mutation => { if (mutation.type === 'childList') { mutation.addedNodes.forEach(node => { // 检查添加的节点是否是 .protyle 元素 if (node.nodeType === 1 && node.matches('.layout-tab-container > .protyle')) { addFullScreenButton(node); } }); } else if (mutation.type === 'attributes' && mutation.target.matches('.layout-tab-container > .protyle')) { // 如果已有的 protyle 的类名发生变化,尝试添加全屏按钮 addFullScreenButton(mutation.target); } }); }); // 配置并开始观察 const config = { childList: true, subtree: true, attributes: true }; observer.observe(targetNode, config); } checkForLayoutCenter(); })();
旧方案(全屏切换需要设置快捷键)
// 在面包屑旁添加全屏按钮,快捷切换全屏 JS片段 // author by JeffreyChen https://ld246.com/article/1731698559408 // 注意:需要为全屏切换设置一个快捷键 (function() { function addFullScreenButton(protyleElement) { // 检查该 protyle 是否已经有了 fullScreen_simulate 按钮 if (protyleElement.querySelector('.fullScreen_simulate')) { return; // 如果已存在,直接返回 } let mode = protyleElement.querySelector('.protyle-breadcrumb .block__icon[data-type="readonly"]'); if (mode) { mode.insertAdjacentHTML( "beforebegin", '<button class="fullScreen_simulate block__icon fn__flex-center ariaLabel" aria-label="全屏切换"></button>' ); let fullScreenBtn = protyleElement.querySelector(".fullScreen_simulate"); fullScreenBtn.innerHTML = `<svg><use xlink:href="#iconFullscreen"></use></svg>`; fullScreenBtn.addEventListener("click", function (e) { // 获取 .layout-tab-container > .protyle .protyle-breadcrumb__space 元素 const breadcrumbSpace = protyleElement.querySelector('.protyle-breadcrumb__space'); // 如果元素存在,则模拟点击,聚焦当前页签 if (breadcrumbSpace) { breadcrumbSpace.click(); } dispatchKeyEvent(); // 切换图标 const iconUse = fullScreenBtn.querySelector('use'); if (iconUse.getAttribute('xlink:href') === '#iconFullscreen') { iconUse.setAttribute('xlink:href', '#iconFullscreenExit'); } else { iconUse.setAttribute('xlink:href', '#iconFullscreen'); } }); } } function dispatchKeyEvent() { let keyInit = parseHotKeyStr(window.top.siyuan.config.keymap.editor.general.fullscreen.custom); keyInit["bubbles"] = true; let keydownEvent = new KeyboardEvent('keydown', keyInit); document.getElementsByTagName("body")[0].dispatchEvent(keydownEvent); let keyUpEvent = new KeyboardEvent('keyup', keyInit); document.getElementsByTagName("body")[0].dispatchEvent(keyUpEvent); } /** * @param {*} hotkeyStr 思源hotkey格式 Refer: https://github.com/siyuan-note/siyuan/blob/d0f011b1a5b12e5546421f8bd442606bf0b5ad86/app/src/protyle/util/hotKey.ts#L4 * @returns KeyboardEventInit Refer: https://developer.mozilla.org/zh-CN/docs/Web/API/KeyboardEvent/KeyboardEvent */ function parseHotKeyStr(hotkeyStr) { let result = { ctrlKey: false, altKey: false, metaKey: false, shiftKey: false, key: 'A', keyCode: 0 } if (hotkeyStr == "" || hotkeyStr == undefined || hotkeyStr == null) { console.error("解析快捷键设置失败", hotkeyStr); throw new Error("解析快捷键设置失败"); } let onlyKey = hotkeyStr; if (hotkeyStr.indexOf("⌘") != -1) { result.ctrlKey = true; onlyKey = onlyKey.replace("⌘", ""); } if (hotkeyStr.indexOf("⌥") != -1) { result.altKey = true; onlyKey = onlyKey.replace("⌥", ""); } if (hotkeyStr.indexOf("⇧") != -1) { result.shiftKey = true; onlyKey = onlyKey.replace("⇧", ""); } // 未处理 windows btn (MetaKey) result.key = onlyKey; // 在https://github.com/siyuan-note/siyuan/commit/70acd57c4b4701b973a8ca93fadf6c003b24c789#diff-558f9f531a326d2fd53151e3fc250ac4bd545452ba782b0c7c18765a37a4e2cc // 更改中,思源改为使用keyCode判断快捷键按下事件,这里进行了对应的转换 // 另请参考该提交中涉及的文件 result.keyCode = keyCodeList[result.key]; console.assert(result.keyCode != undefined, `keyCode转换错误,key为${result.key}`); switch (result.key) { case "→": { result.key = "ArrowRight"; break; } case "←": { result.key = "ArrowLeft"; break; } case "↑": { result.key = "ArrowUp"; break; } case "↓": { result.key = "ArrowDown"; break; } case "⌦": { result.key = "Delete"; break; } case "⌫": { result.key = "Backspace"; break; } case "↩": { result.key = "Enter"; break; } } return result; } const keyCodeList = { "⌫": 8, "⇥": 9, "↩": 13, "⇧": 16, "⌘": 91, "⌥": 18, "Pause": 19, "CapsLock": 20, "Escape": 27, " ": 32, "PageUp": 33, "PageDown": 34, "End": 35, "Home": 36, "←": 37, "↑": 38, "→": 39, "↓": 40, "PrintScreen": 44, "Insert": 45, "⌦": 46, "0": 48, "1": 49, "2": 50, "3": 51, "4": 52, "5": 53, "6": 54, "7": 55, "8": 56, "9": 57, "A": 65, "B": 66, "C": 67, "D": 68, "E": 69, "F": 70, "G": 71, "H": 72, "I": 73, "J": 74, "K": 75, "L": 76, "M": 77, "N": 78, "O": 79, "P": 80, "Q": 81, "R": 82, "S": 83, "T": 84, "U": 85, "V": 86, "W": 87, "X": 88, "Y": 89, "Z": 90, "ContextMenu": 93, "MyComputer": 182, "MyCalculator": 183, ";": 186, "=": 187, ",": 188, "-": 189, ".": 190, "/": 191, "`": 192, "[": 219, "\\": 220, "]": 221, "'": 222, "*": 106, "+": 107, "-": 109, ".": 110, "/": 111, "F1": 112, "F2": 113, "F3": 114, "F4": 115, "F5": 116, "F6": 117, "F7": 118, "F8": 119, "F9": 120, "F10": 121, "F11": 122, "F12": 123, "NumLock": 144, "ScrollLock": 145 }; // 定期检查 .layout__center 是否存在于 DOM 中 function checkForLayoutCenter() { const targetNode = document.querySelector('.layout__center'); if (targetNode) { startObserving(targetNode); // 立即检查一次 protyle-breadcrumb checkProtyleElements(targetNode); } else { setTimeout(checkForLayoutCenter, 200); } } function checkProtyleElements(targetNode) { const protyles = targetNode.querySelectorAll('.layout-tab-container > .protyle'); protyles.forEach(protyle => { addFullScreenButton(protyle); }); } function startObserving(targetNode) { const observer = new MutationObserver((mutations) => { mutations.forEach(mutation => { if (mutation.type === 'childList') { mutation.addedNodes.forEach(node => { // 检查添加的节点是否是 .protyle 元素 if (node.nodeType === 1 && node.matches('.layout-tab-container > .protyle')) { addFullScreenButton(node); } }); } else if (mutation.type === 'attributes' && mutation.target.matches('.layout-tab-container > .protyle')) { // 如果已有的 protyle 的类名发生变化,尝试添加全屏按钮 addFullScreenButton(mutation.target); } }); }); // 配置并开始观察 const config = { childList: true, subtree: true, attributes: true }; observer.observe(targetNode, config); } checkForLayoutCenter(); })();
打赏 50 积分后可见
50 积分 • 6 打赏
  • 思源笔记

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

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

    26774 引用 • 111596 回帖
  • 代码片段

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

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

    227 引用 • 1638 回帖
2 操作
JeffreyChen 在 2024-11-16 03:51:10 更新了该帖
JeffreyChen 在 2024-11-16 03:49:59 更新了该帖

相关帖子

欢迎来到这里!

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

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

    建议加上其他事件切换时(比如按快捷键,文档菜单全屏等)也自动切换按钮,类似我下面这样的代码

    // 监听其他元素全屏事件 observeClassAddition(protyle, 'fullscreen', (eventType) => { if(eventType === 'fullscreen'){ fullScreenBtn.innerHTML = exitFullscreenSvg; fullScreenBtn.setAttribute('aria-label', '退出全屏'); } else { fullScreenBtn.innerHTML = fullscreenSvg; fullScreenBtn.setAttribute('aria-label', '全屏'); } });
  • wilsons 2

    刚才试了下,通过监听 window.siyuan.editorIsFullscreen 对象的变化也可以判断是否全屏了,这样就不需要监听 protyle 样式的变化了

    // 定义一个可观察的属性 window.siyuan._editorIsFullscreen = window.siyuan.editorIsFullscreen || false; Object.defineProperty(window.siyuan, 'editorIsFullscreen', { get: function() { return this._editorIsFullscreen; }, set: function(value) { const oldValue = this._editorIsFullscreen; this._editorIsFullscreen = value; // value true是全屏,false是退出全屏 console.log(`editorIsFullscreen changed from ${oldValue} to ${value}`); }, configurable: true, enumerable: true });
  • wujianzhong via macOS

    有 Bug,,在 macOS 下,全屏后右上角的退出全屏按钮(整排按钮)都不可用,无论鼠标如何点都不起作用

    1 回复
  • 试试新建工作空间,只使用这个代码片段,看看会不会有问题

    1 回复
  • wujianzhong via macOS

    试了,,在空白的新空间可以使用,,但安装了插件之后就出问题了

    1 回复
  • 你排查一下跟哪个插件冲突

    1 回复
  • wujianzhong

    跟 移除按钮 和 移除按钮-经典 两个插件冲突

    1 回复
  • 你是两个插件都打开了吗?还是只打开了一个

请输入回帖内容 ...
JeffreyChen
思源笔记同步教程:https://ld246.com/article/1692089679062 爱发电:https://afdian.com/a/JeffreyChen

推荐标签 标签

  • ActiveMQ

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

    19 引用 • 13 回帖 • 687 关注
  • HHKB

    HHKB 是富士通的 Happy Hacking 系列电容键盘。电容键盘即无接点静电电容式键盘(Capacitive Keyboard)。

    5 引用 • 74 回帖 • 522 关注
  • H2

    H2 是一个开源的嵌入式数据库引擎,采用 Java 语言编写,不受平台的限制,同时 H2 提供了一个十分方便的 web 控制台用于操作和管理数据库内容。H2 还提供兼容模式,可以兼容一些主流的数据库,因此采用 H2 作为开发期的数据库非常方便。

    11 引用 • 54 回帖 • 676 关注
  • OAuth

    OAuth 协议为用户资源的授权提供了一个安全的、开放而又简易的标准。与以往的授权方式不同之处是 oAuth 的授权不会使第三方触及到用户的帐号信息(如用户名与密码),即第三方无需使用用户的用户名与密码就可以申请获得该用户资源的授权,因此 oAuth 是安全的。oAuth 是 Open Authorization 的简写。

    36 引用 • 103 回帖 • 35 关注
  • Openfire

    Openfire 是开源的、基于可拓展通讯和表示协议 (XMPP)、采用 Java 编程语言开发的实时协作服务器。Openfire 的效率很高,单台服务器可支持上万并发用户。

    6 引用 • 7 回帖 • 117 关注
  • IDEA

    IDEA 全称 IntelliJ IDEA,是一款 Java 语言开发的集成环境,在业界被公认为最好的 Java 开发工具之一。IDEA 是 JetBrains 公司的产品,这家公司总部位于捷克共和国的首都布拉格,开发人员以严谨著称的东欧程序员为主。

    181 引用 • 400 回帖 • 1 关注
  • MongoDB

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

    91 引用 • 59 回帖 • 2 关注
  • SSL

    SSL(Secure Sockets Layer 安全套接层),及其继任者传输层安全(Transport Layer Security,TLS)是为网络通信提供安全及数据完整性的一种安全协议。TLS 与 SSL 在传输层对网络连接进行加密。

    70 引用 • 193 回帖 • 415 关注
  • NGINX

    NGINX 是一个高性能的 HTTP 和反向代理服务器,也是一个 IMAP/POP3/SMTP 代理服务器。 NGINX 是由 Igor Sysoev 为俄罗斯访问量第二的 Rambler.ru 站点开发的,第一个公开版本 0.1.0 发布于 2004 年 10 月 4 日。

    315 引用 • 547 回帖
  • CSDN

    CSDN (Chinese Software Developer Network) 创立于 1999 年,是中国的 IT 社区和服务平台,为中国的软件开发者和 IT 从业者提供知识传播、职业发展、软件开发等全生命周期服务,满足他们在职业发展中学习及共享知识和信息、建立职业发展社交圈、通过软件开发实现技术商业化等刚性需求。

    14 引用 • 155 回帖 • 1 关注
  • Chrome

    Chrome 又称 Google 浏览器,是一个由谷歌公司开发的网页浏览器。该浏览器是基于其他开源软件所编写,包括 WebKit,目标是提升稳定性、速度和安全性,并创造出简单且有效率的使用者界面。

    63 引用 • 289 回帖
  • 星云链

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

    3 引用 • 16 回帖
  • 职场

    找到自己的位置,萌新烦恼少。

    127 引用 • 1708 回帖 • 2 关注
  • 宕机

    宕机,多指一些网站、游戏、网络应用等服务器一种区别于正常运行的状态,也叫“Down 机”、“当机”或“死机”。宕机状态不仅仅是指服务器“挂掉了”、“死机了”状态,也包括服务器假死、停用、关闭等一些原因而导致出现的不能够正常运行的状态。

    13 引用 • 82 回帖 • 74 关注
  • RYMCU

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

    4 引用 • 6 回帖 • 65 关注
  • 分享

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

    248 引用 • 1795 回帖
  • Sandbox

    如果帖子标签含有 Sandbox ,则该帖子会被视为“测试帖”,主要用于测试社区功能,排查 bug 等,该标签下内容不定期进行清理。

    443 引用 • 1238 回帖 • 599 关注
  • Firefox

    Mozilla Firefox 中文俗称“火狐”(正式缩写为 Fx 或 fx,非正式缩写为 FF),是一个开源的网页浏览器,使用 Gecko 排版引擎,支持多种操作系统,如 Windows、OSX 及 Linux 等。

    7 引用 • 30 回帖 • 368 关注
  • ReactiveX

    ReactiveX 是一个专注于异步编程与控制可观察数据(或者事件)流的 API。它组合了观察者模式,迭代器模式和函数式编程的优秀思想。

    1 引用 • 2 回帖 • 184 关注
  • 智能合约

    智能合约(Smart contract)是一种旨在以信息化方式传播、验证或执行合同的计算机协议。智能合约允许在没有第三方的情况下进行可信交易,这些交易可追踪且不可逆转。智能合约概念于 1994 年由 Nick Szabo 首次提出。

    1 引用 • 11 回帖
  • Typecho

    Typecho 是一款博客程序,它在 GPLv2 许可证下发行,基于 PHP 构建,可以运行在各种平台上,支持多种数据库(MySQL、PostgreSQL、SQLite)。

    12 引用 • 67 回帖 • 445 关注
  • PWL

    组织简介

    用爱发电 (Programming With Love) 是一个以开源精神为核心的民间开源爱好者技术组织,“用爱发电”象征开源与贡献精神,加入组织,代表你将遵守组织的“个人开源爱好者”的各项条款。申请加入:用爱发电组织邀请帖
    用爱发电组织官网:https://programmingwithlove.stackoverflow.wiki/

    用爱发电组织的核心驱动力:

    • 遵守开源守则,体现开源&贡献精神:以分享为目的,拒绝非法牟利。
    • 自我保护:使用适当的 License 保护自己的原创作品。
    • 尊重他人:不以各种理由、各种漏洞进行未经允许的抄袭、散播、洩露;以礼相待,尊重所有对社区做出贡献的开发者;通过他人的分享习得知识,要留下足迹,表示感谢。
    • 热爱编程、热爱学习:加入组织,热爱编程是首当其要的。我们欢迎热爱讨论、分享、提问的朋友,也同样欢迎默默成就的朋友。
    • 倾听:正确并恳切对待、处理问题与建议,及时修复开源项目的 Bug ,及时与反馈者沟通。不抬杠、不无视、不辱骂。
    • 平视:不诋毁、轻视、嘲讽其他开发者,主动提出建议、施以帮助,以和谐为本。只要他人肯努力,你也可能会被昔日小看的人所超越,所以请保持谦虚。
    • 乐观且活跃:你的努力决定了你的高度。不要放弃,多年后回头俯瞰,才会发现自己已经成就往日所仰望的水平。积极地将项目开源,帮助他人学习、改进,自己也会获得相应的提升、成就与成就感。
    1 引用 • 487 回帖 • 2 关注
  • 架构

    我们平时所说的“架构”主要是指软件架构,这是有关软件整体结构与组件的抽象描述,用于指导软件系统各个方面的设计。另外还有“业务架构”、“网络架构”、“硬件架构”等细分领域。

    142 引用 • 442 回帖
  • 深度学习

    深度学习(Deep Learning)是机器学习的分支,是一种试图使用包含复杂结构或由多重非线性变换构成的多个处理层对数据进行高层抽象的算法。

    43 引用 • 44 回帖
  • 代码片段

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

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

    227 引用 • 1638 回帖
  • WordPress

    WordPress 是一个使用 PHP 语言开发的博客平台,用户可以在支持 PHP 和 MySQL 数据库的服务器上架设自己的博客。也可以把 WordPress 当作一个内容管理系统(CMS)来使用。WordPress 是一个免费的开源项目,在 GNU 通用公共许可证(GPLv2)下授权发布。

    46 引用 • 114 回帖 • 156 关注
  • Scala

    Scala 是一门多范式的编程语言,集成面向对象编程和函数式编程的各种特性。

    13 引用 • 11 回帖 • 165 关注